icu_locid/shortvec/
mod.rs
1mod litemap;
43
44use alloc::boxed::Box;
45use alloc::vec;
46use alloc::vec::Vec;
47use core::ops::Deref;
48use core::ops::DerefMut;
49
50#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
53pub(crate) enum ShortBoxSliceInner<T> {
54 ZeroOne(Option<T>),
55 Multi(Box<[T]>),
56}
57
58impl<T> Default for ShortBoxSliceInner<T> {
59 fn default() -> Self {
60 use ShortBoxSliceInner::*;
61 ZeroOne(None)
62 }
63}
64
65#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
69pub(crate) struct ShortBoxSlice<T>(ShortBoxSliceInner<T>);
70
71impl<T> Default for ShortBoxSlice<T> {
72 fn default() -> Self {
73 Self(Default::default())
74 }
75}
76
77impl<T> ShortBoxSlice<T> {
78 #[inline]
80 pub const fn new() -> Self {
81 use ShortBoxSliceInner::*;
82 Self(ZeroOne(None))
83 }
84
85 #[inline]
87 pub const fn new_single(item: T) -> Self {
88 use ShortBoxSliceInner::*;
89 Self(ZeroOne(Some(item)))
90 }
91
92 pub fn push(&mut self, item: T) {
96 use ShortBoxSliceInner::*;
97 self.0 = match core::mem::replace(&mut self.0, ZeroOne(None)) {
98 ZeroOne(None) => ZeroOne(Some(item)),
99 ZeroOne(Some(prev_item)) => Multi(vec![prev_item, item].into_boxed_slice()),
100 Multi(items) => {
101 let mut items = items.into_vec();
102 items.push(item);
103 Multi(items.into_boxed_slice())
104 }
105 };
106 }
107
108 #[inline]
112 pub const fn single(&self) -> Option<&T> {
113 use ShortBoxSliceInner::*;
114 match self.0 {
115 ZeroOne(Some(ref v)) => Some(v),
116 _ => None,
117 }
118 }
119
120 #[inline]
122 pub fn len(&self) -> usize {
123 use ShortBoxSliceInner::*;
124 match self.0 {
125 ZeroOne(None) => 0,
126 ZeroOne(_) => 1,
127 Multi(ref v) => v.len(),
128 }
129 }
130
131 #[inline]
133 pub fn is_empty(&self) -> bool {
134 use ShortBoxSliceInner::*;
135 matches!(self.0, ZeroOne(None))
136 }
137
138 pub fn insert(&mut self, index: usize, elt: T) {
142 use ShortBoxSliceInner::*;
143 assert!(
144 index <= self.len(),
145 "insertion index (is {}) should be <= len (is {})",
146 index,
147 self.len()
148 );
149
150 self.0 = match core::mem::replace(&mut self.0, ZeroOne(None)) {
151 ZeroOne(None) => ZeroOne(Some(elt)),
152 ZeroOne(Some(item)) => {
153 let items = if index == 0 {
154 vec![elt, item].into_boxed_slice()
155 } else {
156 vec![item, elt].into_boxed_slice()
157 };
158 Multi(items)
159 }
160 Multi(items) => {
161 let mut items = items.into_vec();
162 items.insert(index, elt);
163 Multi(items.into_boxed_slice())
164 }
165 }
166 }
167
168 pub fn remove(&mut self, index: usize) -> T {
172 use ShortBoxSliceInner::*;
173 assert!(
174 index < self.len(),
175 "removal index (is {}) should be < len (is {})",
176 index,
177 self.len()
178 );
179
180 let (replaced, removed_item) = match core::mem::replace(&mut self.0, ZeroOne(None)) {
181 ZeroOne(None) => unreachable!(),
182 ZeroOne(Some(v)) => (ZeroOne(None), v),
183 Multi(v) => {
184 let mut v = v.into_vec();
185 let removed_item = v.remove(index);
186 match v.len() {
187 #[allow(clippy::unwrap_used)]
188 1 => (ZeroOne(Some(v.pop().unwrap())), removed_item),
190 _ => (Multi(v.into_boxed_slice()), removed_item),
192 }
193 }
194 };
195 self.0 = replaced;
196 removed_item
197 }
198
199 #[inline]
201 pub fn clear(&mut self) {
202 use ShortBoxSliceInner::*;
203 let _ = core::mem::replace(&mut self.0, ZeroOne(None));
204 }
205
206 pub fn retain<F>(&mut self, mut f: F)
208 where
209 F: FnMut(&T) -> bool,
210 {
211 use ShortBoxSliceInner::*;
212 match core::mem::take(&mut self.0) {
213 ZeroOne(Some(one)) if f(&one) => self.0 = ZeroOne(Some(one)),
214 ZeroOne(_) => self.0 = ZeroOne(None),
215 Multi(slice) => {
216 let mut vec = slice.into_vec();
217 vec.retain(f);
218 *self = ShortBoxSlice::from(vec)
219 }
220 };
221 }
222}
223
224impl<T> Deref for ShortBoxSlice<T> {
225 type Target = [T];
226
227 fn deref(&self) -> &Self::Target {
228 use ShortBoxSliceInner::*;
229 match self.0 {
230 ZeroOne(None) => &[],
231 ZeroOne(Some(ref v)) => core::slice::from_ref(v),
232 Multi(ref v) => v,
233 }
234 }
235}
236
237impl<T> DerefMut for ShortBoxSlice<T> {
238 fn deref_mut(&mut self) -> &mut Self::Target {
239 use ShortBoxSliceInner::*;
240 match self.0 {
241 ZeroOne(None) => &mut [],
242 ZeroOne(Some(ref mut v)) => core::slice::from_mut(v),
243 Multi(ref mut v) => v,
244 }
245 }
246}
247
248impl<T> From<Vec<T>> for ShortBoxSlice<T> {
249 fn from(v: Vec<T>) -> Self {
250 use ShortBoxSliceInner::*;
251 match v.len() {
252 0 => Self(ZeroOne(None)),
253 #[allow(clippy::unwrap_used)] 1 => Self(ZeroOne(Some(v.into_iter().next().unwrap()))),
255 _ => Self(Multi(v.into_boxed_slice())),
256 }
257 }
258}
259
260impl<T> FromIterator<T> for ShortBoxSlice<T> {
261 fn from_iter<I: IntoIterator<Item = T>>(iter: I) -> Self {
262 use ShortBoxSliceInner::*;
263 let mut iter = iter.into_iter();
264 match (iter.next(), iter.next()) {
265 (Some(first), Some(second)) => {
266 let mut vec = Vec::with_capacity(iter.size_hint().0.saturating_add(3));
268 vec.push(first);
269 vec.push(second);
270 vec.extend(iter);
271 Self(Multi(vec.into_boxed_slice()))
272 }
273 (first, _) => Self(ZeroOne(first)),
274 }
275 }
276}
277
278#[derive(Debug)]
280pub struct ShortBoxSliceIntoIter<T>(ShortBoxSliceIntoIterInner<T>);
281
282#[derive(Debug)]
283pub(crate) enum ShortBoxSliceIntoIterInner<T> {
284 ZeroOne(Option<T>),
285 Multi(alloc::vec::IntoIter<T>),
286}
287
288impl<T> Iterator for ShortBoxSliceIntoIter<T> {
289 type Item = T;
290 fn next(&mut self) -> Option<T> {
291 use ShortBoxSliceIntoIterInner::*;
292 match &mut self.0 {
293 ZeroOne(option) => option.take(),
294 Multi(into_iter) => into_iter.next(),
295 }
296 }
297}
298
299impl<T> IntoIterator for ShortBoxSlice<T> {
300 type Item = T;
301 type IntoIter = ShortBoxSliceIntoIter<T>;
302
303 fn into_iter(self) -> Self::IntoIter {
304 match self.0 {
305 ShortBoxSliceInner::ZeroOne(option) => {
306 ShortBoxSliceIntoIter(ShortBoxSliceIntoIterInner::ZeroOne(option))
307 }
308 ShortBoxSliceInner::Multi(boxed_slice) => ShortBoxSliceIntoIter(
311 ShortBoxSliceIntoIterInner::Multi(boxed_slice.into_vec().into_iter()),
312 ),
313 }
314 }
315}
316
317#[cfg(test)]
318mod tests {
319 use super::*;
320
321 #[test]
322 #[allow(clippy::get_first)]
323 fn test_new_single_const() {
324 const MY_CONST_SLICE: ShortBoxSlice<i32> = ShortBoxSlice::new_single(42);
325
326 assert_eq!(MY_CONST_SLICE.len(), 1);
327 assert_eq!(MY_CONST_SLICE.get(0), Some(&42));
328 }
329
330 #[test]
331 #[allow(clippy::redundant_pattern_matching)]
332 fn test_get_single() {
333 let mut vec = ShortBoxSlice::new();
334 assert!(matches!(vec.single(), None));
335
336 vec.push(100);
337 assert!(matches!(vec.single(), Some(_)));
338
339 vec.push(200);
340 assert!(matches!(vec.single(), None));
341 }
342}