xref: /wasmtime-44.0.1/crates/core/src/alloc/vec.rs (revision 72dccdfd)
1 use crate::alloc::{TryClone, try_realloc};
2 use crate::error::OutOfMemory;
3 use core::borrow::Borrow;
4 use core::{
5     cmp::Ordering,
6     fmt, mem,
7     num::NonZeroUsize,
8     ops::{Deref, DerefMut, Index, IndexMut},
9     slice::SliceIndex,
10 };
11 #[cfg(feature = "serde")]
12 use serde::ser::SerializeSeq;
13 use std_alloc::alloc::Layout;
14 use std_alloc::boxed::Box;
15 use std_alloc::vec::Vec as StdVec;
16 
17 /// Same as the [`std::vec!`] macro but returns an error on allocation failure.
18 #[macro_export]
19 macro_rules! try_vec {
20     ( $( $elem:expr ),* ) => {{
21         let len = $crate::private_len!( $( $elem ),* );
22         $crate::alloc::TryVec::with_capacity(len).and_then(|mut v| {
23             $( v.push($elem)?; )*
24             let _ = &mut v;
25             Ok(v)
26         })
27     }};
28 
29     ( $elem:expr; $len:expr ) => {{
30         let len: usize = $len;
31         if let Some(len) = ::core::num::NonZeroUsize::new(len) {
32             let elem = $elem;
33             $crate::alloc::TryVec::from_elem(elem, len)
34         } else {
35             Ok($crate::alloc::TryVec::new())
36         }
37     }};
38 
39 }
40 
41 // Only for use by the `vec!` macro.
42 #[doc(hidden)]
43 #[macro_export]
44 macro_rules! private_len {
45     ( ) => { 0 };
46     ( $e:expr $( , $es:expr )* ) => { 1 + $crate::private_len!( $( $es ),* ) };
47 }
48 
49 /// Like `std::vec::Vec` but all methods that allocate force handling allocation
50 /// failure.
51 #[derive(PartialEq, Eq, PartialOrd, Ord, Hash)]
52 pub struct TryVec<T> {
53     inner: StdVec<T>,
54 }
55 
56 impl<T> Default for TryVec<T> {
57     fn default() -> Self {
58         Self {
59             inner: Default::default(),
60         }
61     }
62 }
63 
64 impl<T: fmt::Debug> fmt::Debug for TryVec<T> {
65     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
66         fmt::Debug::fmt(&self.inner, f)
67     }
68 }
69 
70 impl<T> TryClone for TryVec<T>
71 where
72     T: TryClone,
73 {
74     fn try_clone(&self) -> Result<Self, OutOfMemory> {
75         let mut v = TryVec::with_capacity(self.len())?;
76         for x in self {
77             v.push(x.try_clone()?).expect("reserved capacity");
78         }
79         Ok(v)
80     }
81 }
82 
83 impl<T> TryVec<T> {
84     /// Same as [`std::vec::Vec::new`].
85     pub const fn new() -> Self {
86         Self {
87             inner: StdVec::new(),
88         }
89     }
90 
91     /// Same as [`std::vec::Vec::with_capacity`] but returns an error on
92     /// allocation failure.
93     pub fn with_capacity(capacity: usize) -> Result<Self, OutOfMemory> {
94         let mut v = Self::new();
95         v.reserve(capacity)?;
96         Ok(v)
97     }
98 
99     // For use with the `vec!` macro.
100     #[doc(hidden)]
101     #[inline]
102     pub fn from_elem(elem: T, len: NonZeroUsize) -> Result<Self, OutOfMemory>
103     where
104         T: TryClone,
105     {
106         let mut v = Self::with_capacity(len.get())?;
107 
108         // Minimize calls to `TryClone` by always pushing `elem` itself as the
109         // last element.
110         for _ in 0..len.get() - 1 {
111             v.push(elem.try_clone()?)?;
112         }
113         v.push(elem)?;
114 
115         Ok(v)
116     }
117 
118     /// Same as [`std::vec::Vec::reserve`] but returns an error on allocation
119     /// failure.
120     pub fn reserve(&mut self, additional: usize) -> Result<(), OutOfMemory> {
121         self.inner.try_reserve(additional).map_err(|_| {
122             OutOfMemory::new(
123                 self.len()
124                     .saturating_add(additional)
125                     .saturating_mul(mem::size_of::<T>()),
126             )
127         })
128     }
129 
130     /// Same as [`std::vec::Vec::reserve_exact`] but returns an error on allocation
131     /// failure.
132     pub fn reserve_exact(&mut self, additional: usize) -> Result<(), OutOfMemory> {
133         self.inner
134             .try_reserve_exact(additional)
135             .map_err(|_| OutOfMemory::new(self.len().saturating_add(additional)))
136     }
137 
138     /// Same as [`std::vec::Vec::len`].
139     pub fn len(&self) -> usize {
140         self.inner.len()
141     }
142 
143     /// Same as [`std::vec::Vec::capacity`].
144     pub fn capacity(&self) -> usize {
145         self.inner.capacity()
146     }
147 
148     /// Same as [`std::vec::Vec::is_empty`].
149     pub fn is_empty(&self) -> bool {
150         self.inner.is_empty()
151     }
152 
153     /// Same as [`std::vec::Vec::push`] but returns an error on allocation
154     /// failure.
155     pub fn push(&mut self, value: T) -> Result<(), OutOfMemory> {
156         self.reserve(1)?;
157         self.inner.push(value);
158         Ok(())
159     }
160 
161     /// Same as [`std::vec::Vec::pop`].
162     pub fn pop(&mut self) -> Option<T> {
163         self.inner.pop()
164     }
165 
166     /// Same as [`std::vec::Vec::truncate`].
167     pub fn truncate(&mut self, len: usize) {
168         self.inner.truncate(len);
169     }
170 
171     /// Same as [`std::vec::Vec::resize`] but returns an error on allocation
172     /// failure.
173     pub fn resize(&mut self, new_len: usize, value: T) -> Result<(), OutOfMemory>
174     where
175         T: TryClone,
176     {
177         match new_len.cmp(&self.len()) {
178             Ordering::Less => self.truncate(new_len),
179             Ordering::Equal => {}
180             Ordering::Greater => {
181                 let delta = new_len - self.len();
182                 self.reserve(delta)?;
183                 // Minimize `try_clone` calls by always pushing `value` directly
184                 // as the last element.
185                 for _ in 0..delta - 1 {
186                     self.push(value.try_clone()?)?;
187                 }
188                 self.push(value)?;
189             }
190         }
191         Ok(())
192     }
193 
194     /// Same as [`std::vec::Vec::into_raw_parts`].
195     pub fn into_raw_parts(mut self) -> (*mut T, usize, usize) {
196         // NB: Can't use `Vec::into_raw_parts` until our MSRV is >= 1.93.
197         #[cfg(not(miri))]
198         {
199             let ptr = self.as_mut_ptr();
200             let len = self.len();
201             let cap = self.capacity();
202             mem::forget(self);
203             (ptr, len, cap)
204         }
205         // NB: Miri requires using `into_raw_parts`, but always run on nightly,
206         // so it's fine to use there.
207         #[cfg(miri)]
208         {
209             let _ = &mut self;
210             self.inner.into_raw_parts()
211         }
212     }
213 
214     /// Same as [`std::vec::Vec::from_raw_parts`].
215     pub unsafe fn from_raw_parts(ptr: *mut T, length: usize, capacity: usize) -> Self {
216         TryVec {
217             // Safety: Same as our unsafe contract.
218             inner: unsafe { StdVec::from_raw_parts(ptr, length, capacity) },
219         }
220     }
221 
222     /// Same as [`std::vec::Vec::drain`].
223     pub fn drain<R>(&mut self, range: R) -> std_alloc::vec::Drain<'_, T>
224     where
225         R: core::ops::RangeBounds<usize>,
226     {
227         self.inner.drain(range)
228     }
229 
230     /// Same as [`std::vec::Vec::shrink_to_fit`] but returns an error on
231     /// allocation failure.
232     pub fn shrink_to_fit(&mut self) -> Result<(), OutOfMemory> {
233         // If our length is already equal to our capacity, then there is nothing
234         // to shrink.
235         if self.len() == self.capacity() {
236             return Ok(());
237         }
238 
239         // `realloc` requires a non-zero original layout as well as a non-zero
240         // destination layout, so this guard ensures that the sizes below are
241         // all nonzero. This handles a few cases:
242         //
243         // * If `len == cap == 0` then no allocation has ever been made.
244         // * If `len == 0` and `cap != 0` then this function effectively frees
245         //   the memory.
246         // * If `T` is a zero-sized type then nothing's been allocated either.
247         //
248         // In all of these cases delegate to the standard library's
249         // `shrink_to_fit` which is guaranteed to not perform a `realloc`.
250         if self.is_empty() || mem::size_of::<T>() == 0 {
251             self.inner.shrink_to_fit();
252             return Ok(());
253         }
254 
255         let (ptr, len, cap) = mem::take(self).into_raw_parts();
256         let layout = Layout::array::<T>(cap).unwrap();
257         let new_size = Layout::array::<T>(len).unwrap().size();
258 
259         // SAFETY: `ptr` was previously allocated in the global allocator,
260         // `layout` has a nonzero size and matches the current allocation of
261         // `ptr`, `new_size` is nonzero, and `new_size` is a valid array size
262         // for `len` elements given its constructor.
263         let result = unsafe { try_realloc(ptr.cast(), layout, new_size) };
264 
265         match result {
266             Ok(ptr) => {
267                 // SAFETY: `result` is allocated with the global allocator and
268                 // has room for exactly `[T; len]`.
269                 *self = unsafe { Self::from_raw_parts(ptr.cast::<T>().as_ptr(), len, len) };
270                 Ok(())
271             }
272             Err(oom) => {
273                 // SAFETY: If reallocation fails then it's guaranteed that the
274                 // original allocation is not tampered with, so it's safe to
275                 // reassemble the original vector.
276                 *self = unsafe { TryVec::from_raw_parts(ptr, len, cap) };
277                 Err(oom)
278             }
279         }
280     }
281 
282     /// Same as [`std::vec::Vec::into_boxed_slice`] but returns an error on
283     /// allocation failure.
284     pub fn into_boxed_slice(mut self) -> Result<Box<[T]>, OutOfMemory> {
285         self.shrink_to_fit()?;
286 
287         // Once we've shrunken the allocation to just the actual length, we can
288         // use `std`'s `into_boxed_slice` without fear of `realloc`.
289         Ok(self.inner.into_boxed_slice())
290     }
291 
292     /// Same as [`std::vec::Vec::clear`].
293     pub fn clear(&mut self) {
294         self.inner.clear();
295     }
296 }
297 
298 impl<T> Deref for TryVec<T> {
299     type Target = [T];
300 
301     fn deref(&self) -> &Self::Target {
302         &self.inner
303     }
304 }
305 
306 impl<T> DerefMut for TryVec<T> {
307     fn deref_mut(&mut self) -> &mut Self::Target {
308         &mut self.inner
309     }
310 }
311 
312 impl<T> AsRef<[T]> for TryVec<T> {
313     fn as_ref(&self) -> &[T] {
314         self
315     }
316 }
317 
318 impl<T> Borrow<[T]> for TryVec<T> {
319     fn borrow(&self) -> &[T] {
320         self
321     }
322 }
323 
324 impl<T, I> Index<I> for TryVec<T>
325 where
326     I: SliceIndex<[T]>,
327 {
328     type Output = <I as SliceIndex<[T]>>::Output;
329 
330     fn index(&self, index: I) -> &Self::Output {
331         &self.inner[index]
332     }
333 }
334 
335 impl<T, I> IndexMut<I> for TryVec<T>
336 where
337     I: SliceIndex<[T]>,
338 {
339     fn index_mut(&mut self, index: I) -> &mut Self::Output {
340         &mut self.inner[index]
341     }
342 }
343 
344 impl<T> IntoIterator for TryVec<T> {
345     type Item = T;
346     type IntoIter = std_alloc::vec::IntoIter<T>;
347 
348     fn into_iter(self) -> Self::IntoIter {
349         self.inner.into_iter()
350     }
351 }
352 
353 impl<'a, T> IntoIterator for &'a TryVec<T> {
354     type Item = &'a T;
355 
356     type IntoIter = core::slice::Iter<'a, T>;
357 
358     fn into_iter(self) -> Self::IntoIter {
359         (**self).iter()
360     }
361 }
362 
363 impl<'a, T> IntoIterator for &'a mut TryVec<T> {
364     type Item = &'a mut T;
365 
366     type IntoIter = core::slice::IterMut<'a, T>;
367 
368     fn into_iter(self) -> Self::IntoIter {
369         (**self).iter_mut()
370     }
371 }
372 
373 impl<T> From<TryVec<T>> for StdVec<T> {
374     fn from(v: TryVec<T>) -> Self {
375         v.inner
376     }
377 }
378 
379 impl<T> From<StdVec<T>> for TryVec<T> {
380     fn from(inner: StdVec<T>) -> Self {
381         Self { inner }
382     }
383 }
384 
385 impl<T> From<Box<[T]>> for TryVec<T> {
386     fn from(boxed_slice: Box<[T]>) -> Self {
387         Self::from(StdVec::from(boxed_slice))
388     }
389 }
390 
391 #[cfg(feature = "serde")]
392 impl<T> serde::ser::Serialize for TryVec<T>
393 where
394     T: serde::ser::Serialize,
395 {
396     fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
397     where
398         S: serde::Serializer,
399     {
400         let mut seq = serializer.serialize_seq(Some(self.len()))?;
401         for elem in self {
402             seq.serialize_element(elem)?;
403         }
404         seq.end()
405     }
406 }
407 
408 #[cfg(feature = "serde")]
409 impl<'de, T> serde::de::Deserialize<'de> for TryVec<T>
410 where
411     T: serde::de::Deserialize<'de>,
412 {
413     fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
414     where
415         D: serde::Deserializer<'de>,
416     {
417         use core::marker::PhantomData;
418 
419         struct Visitor<T>(PhantomData<fn() -> TryVec<T>>);
420 
421         impl<'de, T> serde::de::Visitor<'de> for Visitor<T>
422         where
423             T: serde::de::Deserialize<'de>,
424         {
425             type Value = TryVec<T>;
426 
427             fn expecting(&self, f: &mut fmt::Formatter) -> fmt::Result {
428                 f.write_str("a `wasmtime_core::alloc::Vec` sequence")
429             }
430 
431             fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
432             where
433                 A: serde::de::SeqAccess<'de>,
434             {
435                 use serde::de::Error as _;
436 
437                 let mut v = TryVec::new();
438 
439                 if let Some(len) = seq.size_hint() {
440                     v.reserve_exact(len).map_err(|oom| A::Error::custom(oom))?;
441                 }
442 
443                 while let Some(elem) = seq.next_element()? {
444                     v.push(elem).map_err(|oom| A::Error::custom(oom))?;
445                 }
446 
447                 Ok(v)
448             }
449         }
450 
451         deserializer.deserialize_seq(Visitor(PhantomData))
452     }
453 }
454 
455 #[cfg(test)]
456 mod tests {
457     use super::TryVec;
458     use crate::error::OutOfMemory;
459 
460     #[test]
461     fn test_into_boxed_slice() -> Result<(), OutOfMemory> {
462         assert_eq!(*TryVec::<i32>::new().into_boxed_slice()?, []);
463 
464         let mut vec = TryVec::new();
465         vec.push(1)?;
466         assert_eq!(*vec.into_boxed_slice()?, [1]);
467 
468         let mut vec = TryVec::with_capacity(2)?;
469         vec.push(1)?;
470         assert_eq!(*vec.into_boxed_slice()?, [1]);
471 
472         let mut vec = TryVec::with_capacity(2)?;
473         vec.push(1_u128)?;
474         assert_eq!(*vec.into_boxed_slice()?, [1]);
475 
476         assert_eq!(*TryVec::<()>::new().into_boxed_slice()?, []);
477 
478         let mut vec = TryVec::new();
479         vec.push(())?;
480         assert_eq!(*vec.into_boxed_slice()?, [()]);
481 
482         let vec = TryVec::<i32>::with_capacity(2)?;
483         assert_eq!(*vec.into_boxed_slice()?, []);
484         Ok(())
485     }
486 }
487