//! Indexing implementations for `[T]`. use crate::panic::const_panic; use crate::ub_checks::assert_unsafe_precondition; use crate::{ops, range}; #[stable(feature = "rust1", since = "1.0.0")] impl ops::Index for [T] where I: SliceIndex<[T]>, { type Output = I::Output; #[inline(always)] fn index(&self, index: I) -> &I::Output { index.index(self) } } #[stable(feature = "rust1", since = "1.0.0")] impl ops::IndexMut for [T] where I: SliceIndex<[T]>, { #[inline(always)] fn index_mut(&mut self, index: I) -> &mut I::Output { index.index_mut(self) } } #[cfg_attr(not(feature = "panic_immediate_abort"), inline(never), cold)] #[cfg_attr(feature = "panic_immediate_abort", inline)] #[track_caller] const fn slice_start_index_len_fail(index: usize, len: usize) -> ! { const_panic!( "slice start index is out of range for slice", "range start index {index} out of range for slice of length {len}", index: usize, len: usize, ) } #[cfg_attr(not(feature = "panic_immediate_abort"), inline(never), cold)] #[cfg_attr(feature = "panic_immediate_abort", inline)] #[track_caller] const fn slice_end_index_len_fail(index: usize, len: usize) -> ! { const_panic!( "slice end index is out of range for slice", "range end index {index} out of range for slice of length {len}", index: usize, len: usize, ) } #[cfg_attr(not(feature = "panic_immediate_abort"), inline(never), cold)] #[cfg_attr(feature = "panic_immediate_abort", inline)] #[track_caller] const fn slice_index_order_fail(index: usize, end: usize) -> ! { const_panic!( "slice index start is larger than end", "slice index starts at {index} but ends at {end}", index: usize, end: usize, ) } #[cfg_attr(not(feature = "panic_immediate_abort"), inline(never), cold)] #[cfg_attr(feature = "panic_immediate_abort", inline)] #[track_caller] const fn slice_start_index_overflow_fail() -> ! { panic!("attempted to index slice from after maximum usize"); } #[cfg_attr(not(feature = "panic_immediate_abort"), inline(never), cold)] #[cfg_attr(feature = "panic_immediate_abort", inline)] #[track_caller] const fn slice_end_index_overflow_fail() -> ! { panic!("attempted to index slice up to maximum usize"); } // The UbChecks are great for catching bugs in the unsafe methods, but including // them in safe indexing is unnecessary and hurts inlining and debug runtime perf. // Both the safe and unsafe public methods share these helpers, // which use intrinsics directly to get *no* extra checks. #[inline(always)] const unsafe fn get_noubcheck(ptr: *const [T], index: usize) -> *const T { let ptr = ptr as *const T; // SAFETY: The caller already checked these preconditions unsafe { crate::intrinsics::offset(ptr, index) } } #[inline(always)] const unsafe fn get_mut_noubcheck(ptr: *mut [T], index: usize) -> *mut T { let ptr = ptr as *mut T; // SAFETY: The caller already checked these preconditions unsafe { crate::intrinsics::offset(ptr, index) } } #[inline(always)] const unsafe fn get_offset_len_noubcheck( ptr: *const [T], offset: usize, len: usize, ) -> *const [T] { // SAFETY: The caller already checked these preconditions let ptr = unsafe { get_noubcheck(ptr, offset) }; crate::intrinsics::aggregate_raw_ptr(ptr, len) } #[inline(always)] const unsafe fn get_offset_len_mut_noubcheck( ptr: *mut [T], offset: usize, len: usize, ) -> *mut [T] { // SAFETY: The caller already checked these preconditions let ptr = unsafe { get_mut_noubcheck(ptr, offset) }; crate::intrinsics::aggregate_raw_ptr(ptr, len) } mod private_slice_index { use super::{ops, range}; #[stable(feature = "slice_get_slice", since = "1.28.0")] pub trait Sealed {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for usize {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::Range {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::RangeTo {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::RangeFrom {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::RangeFull {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::RangeInclusive {} #[stable(feature = "slice_get_slice", since = "1.28.0")] impl Sealed for ops::RangeToInclusive {} #[stable(feature = "slice_index_with_ops_bound_pair", since = "1.53.0")] impl Sealed for (ops::Bound, ops::Bound) {} #[unstable(feature = "new_range_api", issue = "125687")] impl Sealed for range::Range {} #[unstable(feature = "new_range_api", issue = "125687")] impl Sealed for range::RangeInclusive {} #[unstable(feature = "new_range_api", issue = "125687")] impl Sealed for range::RangeFrom {} impl Sealed for ops::IndexRange {} } /// A helper trait used for indexing operations. /// /// Implementations of this trait have to promise that if the argument /// to `get_unchecked(_mut)` is a safe reference, then so is the result. #[stable(feature = "slice_get_slice", since = "1.28.0")] #[rustc_diagnostic_item = "SliceIndex"] #[rustc_on_unimplemented( on(T = "str", label = "string indices are ranges of `usize`",), on( all(any(T = "str", T = "&str", T = "alloc::string::String"), _Self = "{integer}"), note = "you can use `.chars().nth()` or `.bytes().nth()`\n\ for more information, see chapter 8 in The Book: \ " ), message = "the type `{T}` cannot be indexed by `{Self}`", label = "slice indices are of type `usize` or ranges of `usize`" )] pub unsafe trait SliceIndex: private_slice_index::Sealed { /// The output type returned by methods. #[stable(feature = "slice_get_slice", since = "1.28.0")] type Output: ?Sized; /// Returns a shared reference to the output at this location, if in /// bounds. #[unstable(feature = "slice_index_methods", issue = "none")] fn get(self, slice: &T) -> Option<&Self::Output>; /// Returns a mutable reference to the output at this location, if in /// bounds. #[unstable(feature = "slice_index_methods", issue = "none")] fn get_mut(self, slice: &mut T) -> Option<&mut Self::Output>; /// Returns a pointer to the output at this location, without /// performing any bounds checking. /// /// Calling this method with an out-of-bounds index or a dangling `slice` pointer /// is *[undefined behavior]* even if the resulting pointer is not used. /// /// [undefined behavior]: https://doc.rust-lang.org/reference/behavior-considered-undefined.html #[unstable(feature = "slice_index_methods", issue = "none")] unsafe fn get_unchecked(self, slice: *const T) -> *const Self::Output; /// Returns a mutable pointer to the output at this location, without /// performing any bounds checking. /// /// Calling this method with an out-of-bounds index or a dangling `slice` pointer /// is *[undefined behavior]* even if the resulting pointer is not used. /// /// [undefined behavior]: https://doc.rust-lang.org/reference/behavior-considered-undefined.html #[unstable(feature = "slice_index_methods", issue = "none")] unsafe fn get_unchecked_mut(self, slice: *mut T) -> *mut Self::Output; /// Returns a shared reference to the output at this location, panicking /// if out of bounds. #[unstable(feature = "slice_index_methods", issue = "none")] #[track_caller] fn index(self, slice: &T) -> &Self::Output; /// Returns a mutable reference to the output at this location, panicking /// if out of bounds. #[unstable(feature = "slice_index_methods", issue = "none")] #[track_caller] fn index_mut(self, slice: &mut T) -> &mut Self::Output; } /// The methods `index` and `index_mut` panic if the index is out of bounds. #[stable(feature = "slice_get_slice_impls", since = "1.15.0")] unsafe impl SliceIndex<[T]> for usize { type Output = T; #[inline] fn get(self, slice: &[T]) -> Option<&T> { // SAFETY: `self` is checked to be in bounds. if self < slice.len() { unsafe { Some(&*get_noubcheck(slice, self)) } } else { None } } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut T> { if self < slice.len() { // SAFETY: `self` is checked to be in bounds. unsafe { Some(&mut *get_mut_noubcheck(slice, self)) } } else { None } } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const T { assert_unsafe_precondition!( check_language_ub, "slice::get_unchecked requires that the index is within the slice", (this: usize = self, len: usize = slice.len()) => this < len ); // SAFETY: the caller guarantees that `slice` is not dangling, so it // cannot be longer than `isize::MAX`. They also guarantee that // `self` is in bounds of `slice` so `self` cannot overflow an `isize`, // so the call to `add` is safe. unsafe { // Use intrinsics::assume instead of hint::assert_unchecked so that we don't check the // precondition of this function twice. crate::intrinsics::assume(self < slice.len()); get_noubcheck(slice, self) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut T { assert_unsafe_precondition!( check_library_ub, "slice::get_unchecked_mut requires that the index is within the slice", (this: usize = self, len: usize = slice.len()) => this < len ); // SAFETY: see comments for `get_unchecked` above. unsafe { get_mut_noubcheck(slice, self) } } #[inline] fn index(self, slice: &[T]) -> &T { // N.B., use intrinsic indexing &(*slice)[self] } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut T { // N.B., use intrinsic indexing &mut (*slice)[self] } } /// Because `IndexRange` guarantees `start <= end`, fewer checks are needed here /// than there are for a general `Range` (which might be `100..3`). unsafe impl SliceIndex<[T]> for ops::IndexRange { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { if self.end() <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { Some(&*get_offset_len_noubcheck(slice, self.start(), self.len())) } } else { None } } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { if self.end() <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { Some(&mut *get_offset_len_mut_noubcheck(slice, self.start(), self.len())) } } else { None } } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { assert_unsafe_precondition!( check_library_ub, "slice::get_unchecked requires that the index is within the slice", (end: usize = self.end(), len: usize = slice.len()) => end <= len ); // SAFETY: the caller guarantees that `slice` is not dangling, so it // cannot be longer than `isize::MAX`. They also guarantee that // `self` is in bounds of `slice` so `self` cannot overflow an `isize`, // so the call to `add` is safe. unsafe { get_offset_len_noubcheck(slice, self.start(), self.len()) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { assert_unsafe_precondition!( check_library_ub, "slice::get_unchecked_mut requires that the index is within the slice", (end: usize = self.end(), len: usize = slice.len()) => end <= len ); // SAFETY: see comments for `get_unchecked` above. unsafe { get_offset_len_mut_noubcheck(slice, self.start(), self.len()) } } #[inline] fn index(self, slice: &[T]) -> &[T] { if self.end() <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &*get_offset_len_noubcheck(slice, self.start(), self.len()) } } else { slice_end_index_len_fail(self.end(), slice.len()) } } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { if self.end() <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &mut *get_offset_len_mut_noubcheck(slice, self.start(), self.len()) } } else { slice_end_index_len_fail(self.end(), slice.len()) } } } /// The methods `index` and `index_mut` panic if: /// - the start of the range is greater than the end of the range or /// - the end of the range is out of bounds. #[stable(feature = "slice_get_slice_impls", since = "1.15.0")] unsafe impl SliceIndex<[T]> for ops::Range { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { // Using checked_sub is a safe way to get `SubUnchecked` in MIR if let Some(new_len) = usize::checked_sub(self.end, self.start) && self.end <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { Some(&*get_offset_len_noubcheck(slice, self.start, new_len)) } } else { None } } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { if let Some(new_len) = usize::checked_sub(self.end, self.start) && self.end <= slice.len() { // SAFETY: `self` is checked to be valid and in bounds above. unsafe { Some(&mut *get_offset_len_mut_noubcheck(slice, self.start, new_len)) } } else { None } } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { assert_unsafe_precondition!( check_library_ub, "slice::get_unchecked requires that the range is within the slice", ( start: usize = self.start, end: usize = self.end, len: usize = slice.len() ) => end >= start && end <= len ); // SAFETY: the caller guarantees that `slice` is not dangling, so it // cannot be longer than `isize::MAX`. They also guarantee that // `self` is in bounds of `slice` so `self` cannot overflow an `isize`, // so the call to `add` is safe and the length calculation cannot overflow. unsafe { // Using the intrinsic avoids a superfluous UB check, // since the one on this method already checked `end >= start`. let new_len = crate::intrinsics::unchecked_sub(self.end, self.start); get_offset_len_noubcheck(slice, self.start, new_len) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { assert_unsafe_precondition!( check_library_ub, "slice::get_unchecked_mut requires that the range is within the slice", ( start: usize = self.start, end: usize = self.end, len: usize = slice.len() ) => end >= start && end <= len ); // SAFETY: see comments for `get_unchecked` above. unsafe { let new_len = crate::intrinsics::unchecked_sub(self.end, self.start); get_offset_len_mut_noubcheck(slice, self.start, new_len) } } #[inline(always)] fn index(self, slice: &[T]) -> &[T] { // Using checked_sub is a safe way to get `SubUnchecked` in MIR let Some(new_len) = usize::checked_sub(self.end, self.start) else { slice_index_order_fail(self.start, self.end) }; if self.end > slice.len() { slice_end_index_len_fail(self.end, slice.len()); } // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &*get_offset_len_noubcheck(slice, self.start, new_len) } } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { let Some(new_len) = usize::checked_sub(self.end, self.start) else { slice_index_order_fail(self.start, self.end) }; if self.end > slice.len() { slice_end_index_len_fail(self.end, slice.len()); } // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &mut *get_offset_len_mut_noubcheck(slice, self.start, new_len) } } } #[unstable(feature = "new_range_api", issue = "125687")] unsafe impl SliceIndex<[T]> for range::Range { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { ops::Range::from(self).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { ops::Range::from(self).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { ops::Range::from(self).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { ops::Range::from(self).get_unchecked_mut(slice) } } #[inline(always)] fn index(self, slice: &[T]) -> &[T] { ops::Range::from(self).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { ops::Range::from(self).index_mut(slice) } } /// The methods `index` and `index_mut` panic if the end of the range is out of bounds. #[stable(feature = "slice_get_slice_impls", since = "1.15.0")] unsafe impl SliceIndex<[T]> for ops::RangeTo { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { (0..self.end).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { (0..self.end).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { (0..self.end).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { (0..self.end).get_unchecked_mut(slice) } } #[inline(always)] fn index(self, slice: &[T]) -> &[T] { (0..self.end).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { (0..self.end).index_mut(slice) } } /// The methods `index` and `index_mut` panic if the start of the range is out of bounds. #[stable(feature = "slice_get_slice_impls", since = "1.15.0")] unsafe impl SliceIndex<[T]> for ops::RangeFrom { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { (self.start..slice.len()).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { (self.start..slice.len()).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { (self.start..slice.len()).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { (self.start..slice.len()).get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &[T] { if self.start > slice.len() { slice_start_index_len_fail(self.start, slice.len()); } // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &*self.get_unchecked(slice) } } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { if self.start > slice.len() { slice_start_index_len_fail(self.start, slice.len()); } // SAFETY: `self` is checked to be valid and in bounds above. unsafe { &mut *self.get_unchecked_mut(slice) } } } #[unstable(feature = "new_range_api", issue = "125687")] unsafe impl SliceIndex<[T]> for range::RangeFrom { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { ops::RangeFrom::from(self).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { ops::RangeFrom::from(self).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { ops::RangeFrom::from(self).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { ops::RangeFrom::from(self).get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &[T] { ops::RangeFrom::from(self).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { ops::RangeFrom::from(self).index_mut(slice) } } #[stable(feature = "slice_get_slice_impls", since = "1.15.0")] unsafe impl SliceIndex<[T]> for ops::RangeFull { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { Some(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { Some(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { slice } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { slice } #[inline] fn index(self, slice: &[T]) -> &[T] { slice } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { slice } } /// The methods `index` and `index_mut` panic if: /// - the end of the range is `usize::MAX` or /// - the start of the range is greater than the end of the range or /// - the end of the range is out of bounds. #[stable(feature = "inclusive_range", since = "1.26.0")] unsafe impl SliceIndex<[T]> for ops::RangeInclusive { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { if *self.end() == usize::MAX { None } else { self.into_slice_range().get(slice) } } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { if *self.end() == usize::MAX { None } else { self.into_slice_range().get_mut(slice) } } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { self.into_slice_range().get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { self.into_slice_range().get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &[T] { if *self.end() == usize::MAX { slice_end_index_overflow_fail(); } self.into_slice_range().index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { if *self.end() == usize::MAX { slice_end_index_overflow_fail(); } self.into_slice_range().index_mut(slice) } } #[unstable(feature = "new_range_api", issue = "125687")] unsafe impl SliceIndex<[T]> for range::RangeInclusive { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { ops::RangeInclusive::from(self).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { ops::RangeInclusive::from(self).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { ops::RangeInclusive::from(self).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { ops::RangeInclusive::from(self).get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &[T] { ops::RangeInclusive::from(self).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { ops::RangeInclusive::from(self).index_mut(slice) } } /// The methods `index` and `index_mut` panic if the end of the range is out of bounds. #[stable(feature = "inclusive_range", since = "1.26.0")] unsafe impl SliceIndex<[T]> for ops::RangeToInclusive { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&[T]> { (0..=self.end).get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut [T]> { (0..=self.end).get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { (0..=self.end).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { (0..=self.end).get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &[T] { (0..=self.end).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut [T] { (0..=self.end).index_mut(slice) } } /// Performs bounds checking of a range. /// /// This method is similar to [`Index::index`] for slices, but it returns a /// [`Range`] equivalent to `range`. You can use this method to turn any range /// into `start` and `end` values. /// /// `bounds` is the range of the slice to use for bounds checking. It should /// be a [`RangeTo`] range that ends at the length of the slice. /// /// The returned [`Range`] is safe to pass to [`slice::get_unchecked`] and /// [`slice::get_unchecked_mut`] for slices with the given range. /// /// [`Range`]: ops::Range /// [`RangeTo`]: ops::RangeTo /// [`slice::get_unchecked`]: slice::get_unchecked /// [`slice::get_unchecked_mut`]: slice::get_unchecked_mut /// /// # Panics /// /// Panics if `range` would be out of bounds. /// /// # Examples /// /// ``` /// #![feature(slice_range)] /// /// use std::slice; /// /// let v = [10, 40, 30]; /// assert_eq!(1..2, slice::range(1..2, ..v.len())); /// assert_eq!(0..2, slice::range(..2, ..v.len())); /// assert_eq!(1..3, slice::range(1.., ..v.len())); /// ``` /// /// Panics when [`Index::index`] would panic: /// /// ```should_panic /// #![feature(slice_range)] /// /// use std::slice; /// /// let _ = slice::range(2..1, ..3); /// ``` /// /// ```should_panic /// #![feature(slice_range)] /// /// use std::slice; /// /// let _ = slice::range(1..4, ..3); /// ``` /// /// ```should_panic /// #![feature(slice_range)] /// /// use std::slice; /// /// let _ = slice::range(1..=usize::MAX, ..3); /// ``` /// /// [`Index::index`]: ops::Index::index #[track_caller] #[unstable(feature = "slice_range", issue = "76393")] #[must_use] pub fn range(range: R, bounds: ops::RangeTo) -> ops::Range where R: ops::RangeBounds, { let len = bounds.end; let start = match range.start_bound() { ops::Bound::Included(&start) => start, ops::Bound::Excluded(start) => { start.checked_add(1).unwrap_or_else(|| slice_start_index_overflow_fail()) } ops::Bound::Unbounded => 0, }; let end = match range.end_bound() { ops::Bound::Included(end) => { end.checked_add(1).unwrap_or_else(|| slice_end_index_overflow_fail()) } ops::Bound::Excluded(&end) => end, ops::Bound::Unbounded => len, }; if start > end { slice_index_order_fail(start, end); } if end > len { slice_end_index_len_fail(end, len); } ops::Range { start, end } } /// Performs bounds checking of a range without panicking. /// /// This is a version of [`range()`] that returns [`None`] instead of panicking. /// /// # Examples /// /// ``` /// #![feature(slice_range)] /// /// use std::slice; /// /// let v = [10, 40, 30]; /// assert_eq!(Some(1..2), slice::try_range(1..2, ..v.len())); /// assert_eq!(Some(0..2), slice::try_range(..2, ..v.len())); /// assert_eq!(Some(1..3), slice::try_range(1.., ..v.len())); /// ``` /// /// Returns [`None`] when [`Index::index`] would panic: /// /// ``` /// #![feature(slice_range)] /// /// use std::slice; /// /// assert_eq!(None, slice::try_range(2..1, ..3)); /// assert_eq!(None, slice::try_range(1..4, ..3)); /// assert_eq!(None, slice::try_range(1..=usize::MAX, ..3)); /// ``` /// /// [`Index::index`]: ops::Index::index #[unstable(feature = "slice_range", issue = "76393")] #[must_use] pub fn try_range(range: R, bounds: ops::RangeTo) -> Option> where R: ops::RangeBounds, { let len = bounds.end; let start = match range.start_bound() { ops::Bound::Included(&start) => start, ops::Bound::Excluded(start) => start.checked_add(1)?, ops::Bound::Unbounded => 0, }; let end = match range.end_bound() { ops::Bound::Included(end) => end.checked_add(1)?, ops::Bound::Excluded(&end) => end, ops::Bound::Unbounded => len, }; if start > end || end > len { None } else { Some(ops::Range { start, end }) } } /// Converts a pair of `ops::Bound`s into `ops::Range` without performing any /// bounds checking or (in debug) overflow checking. pub(crate) fn into_range_unchecked( len: usize, (start, end): (ops::Bound, ops::Bound), ) -> ops::Range { use ops::Bound; let start = match start { Bound::Included(i) => i, Bound::Excluded(i) => i + 1, Bound::Unbounded => 0, }; let end = match end { Bound::Included(i) => i + 1, Bound::Excluded(i) => i, Bound::Unbounded => len, }; start..end } /// Converts pair of `ops::Bound`s into `ops::Range`. /// Returns `None` on overflowing indices. pub(crate) fn into_range( len: usize, (start, end): (ops::Bound, ops::Bound), ) -> Option> { use ops::Bound; let start = match start { Bound::Included(start) => start, Bound::Excluded(start) => start.checked_add(1)?, Bound::Unbounded => 0, }; let end = match end { Bound::Included(end) => end.checked_add(1)?, Bound::Excluded(end) => end, Bound::Unbounded => len, }; // Don't bother with checking `start < end` and `end <= len` // since these checks are handled by `Range` impls Some(start..end) } /// Converts pair of `ops::Bound`s into `ops::Range`. /// Panics on overflowing indices. pub(crate) fn into_slice_range( len: usize, (start, end): (ops::Bound, ops::Bound), ) -> ops::Range { use ops::Bound; let start = match start { Bound::Included(start) => start, Bound::Excluded(start) => { start.checked_add(1).unwrap_or_else(|| slice_start_index_overflow_fail()) } Bound::Unbounded => 0, }; let end = match end { Bound::Included(end) => { end.checked_add(1).unwrap_or_else(|| slice_end_index_overflow_fail()) } Bound::Excluded(end) => end, Bound::Unbounded => len, }; // Don't bother with checking `start < end` and `end <= len` // since these checks are handled by `Range` impls start..end } #[stable(feature = "slice_index_with_ops_bound_pair", since = "1.53.0")] unsafe impl SliceIndex<[T]> for (ops::Bound, ops::Bound) { type Output = [T]; #[inline] fn get(self, slice: &[T]) -> Option<&Self::Output> { into_range(slice.len(), self)?.get(slice) } #[inline] fn get_mut(self, slice: &mut [T]) -> Option<&mut Self::Output> { into_range(slice.len(), self)?.get_mut(slice) } #[inline] unsafe fn get_unchecked(self, slice: *const [T]) -> *const Self::Output { // SAFETY: the caller has to uphold the safety contract for `get_unchecked`. unsafe { into_range_unchecked(slice.len(), self).get_unchecked(slice) } } #[inline] unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut Self::Output { // SAFETY: the caller has to uphold the safety contract for `get_unchecked_mut`. unsafe { into_range_unchecked(slice.len(), self).get_unchecked_mut(slice) } } #[inline] fn index(self, slice: &[T]) -> &Self::Output { into_slice_range(slice.len(), self).index(slice) } #[inline] fn index_mut(self, slice: &mut [T]) -> &mut Self::Output { into_slice_range(slice.len(), self).index_mut(slice) } }