use core::ptr; use crate::alloc::Allocator; use crate::raw_vec::RawVec; use super::{IsZero, Vec}; // Specialization trait used for Vec::from_elem pub(super) trait SpecFromElem: Sized { fn from_elem(elem: Self, n: usize, alloc: A) -> Vec; } impl SpecFromElem for T { default fn from_elem(elem: Self, n: usize, alloc: A) -> Vec { let mut v = Vec::with_capacity_in(n, alloc); v.extend_with(n, elem); v } } impl SpecFromElem for T { #[inline] default fn from_elem(elem: T, n: usize, alloc: A) -> Vec { if elem.is_zero() { return Vec { buf: RawVec::with_capacity_zeroed_in(n, alloc), len: n }; } let mut v = Vec::with_capacity_in(n, alloc); v.extend_with(n, elem); v } } impl SpecFromElem for i8 { #[inline] fn from_elem(elem: i8, n: usize, alloc: A) -> Vec { if elem == 0 { return Vec { buf: RawVec::with_capacity_zeroed_in(n, alloc), len: n }; } let mut v = Vec::with_capacity_in(n, alloc); unsafe { ptr::write_bytes(v.as_mut_ptr(), elem as u8, n); v.set_len(n); } v } } impl SpecFromElem for u8 { #[inline] fn from_elem(elem: u8, n: usize, alloc: A) -> Vec { if elem == 0 { return Vec { buf: RawVec::with_capacity_zeroed_in(n, alloc), len: n }; } let mut v = Vec::with_capacity_in(n, alloc); unsafe { ptr::write_bytes(v.as_mut_ptr(), elem, n); v.set_len(n); } v } } // A better way would be to implement this for all ZSTs which are `Copy` and have trivial `Clone` // but the latter cannot be detected currently impl SpecFromElem for () { #[inline] fn from_elem(_elem: (), n: usize, alloc: A) -> Vec<(), A> { let mut v = Vec::with_capacity_in(n, alloc); // SAFETY: the capacity has just been set to `n` // and `()` is a ZST with trivial `Clone` implementation unsafe { v.set_len(n); } v } }