diff --git a/library/alloc/src/boxed.rs b/library/alloc/src/boxed.rs index a563b2587236c..0b196154f6002 100644 --- a/library/alloc/src/boxed.rs +++ b/library/alloc/src/boxed.rs @@ -156,7 +156,7 @@ use core::fmt; use core::future::Future; use core::hash::{Hash, Hasher}; #[cfg(not(no_global_oom_handling))] -use core::iter::FromIterator; +use core::iter::{FromIterator, TrustedLen}; use core::iter::{FusedIterator, Iterator}; use core::marker::Tuple; use core::marker::{Destruct, Unpin, Unsize}; @@ -2009,11 +2009,51 @@ impl, U: ?Sized, A: Allocator> CoerceUnsized> fo #[unstable(feature = "dispatch_from_dyn", issue = "none")] impl, U: ?Sized> DispatchFromDyn> for Box {} +#[cfg(not(no_global_oom_handling))] +trait SpecFromIter { + fn from_iter(iter: I) -> Self; +} + +#[cfg(not(no_global_oom_handling))] +impl SpecFromIter for Box<[T]> +where + I: Iterator, +{ + default fn from_iter(iter: I) -> Self { + iter.collect::>().into_boxed_slice() + } +} + +#[cfg(not(no_global_oom_handling))] +impl SpecFromIter for Box<[T]> +where + I: TrustedLen, +{ + fn from_iter(iter: I) -> Self { + match iter.size_hint() { + (low, Some(high)) if low == high => { + let mut result = Box::new_uninit_slice(high); + let ptr: *mut T = mem::MaybeUninit::slice_as_mut_ptr(&mut *result); + for (offset, elem) in iter.enumerate() { + // Safety: TrustedLen guarantees that offset is in bounds + unsafe { ptr.add(offset).write(elem) } + } + // Safety: TrustedLen guarantees that we initialized all the elements + unsafe { result.assume_init() } + } + _ => panic!( + ".size_hint() provided by TrustedLen iterator is not exact: {:?}", + iter.size_hint() + ), + } + } +} + #[cfg(not(no_global_oom_handling))] #[stable(feature = "boxed_slice_from_iter", since = "1.32.0")] -impl FromIterator for Box<[I]> { - fn from_iter>(iter: T) -> Self { - iter.into_iter().collect::>().into_boxed_slice() +impl FromIterator for Box<[T]> { + fn from_iter>(iter: I) -> Self { + SpecFromIter::from_iter(iter.into_iter()) } }