Auto merge of #120863 - saethlin:slice-get-checked, r=the8472
Use intrinsics::debug_assertions in debug_assert_nounwind This is the first item in https://github.com/rust-lang/rust/issues/120848. Based on the benchmarking in this PR, it looks like, for the programs in our benchmark suite, enabling all these additional checks does not introduce significant compile-time overhead, with the single exception of `Alignment::new_unchecked`. Therefore, I've added `#[cfg(debug_assertions)]` to that one call site, so that it remains compiled out in the distributed standard library. The trailing commas in the previous calls to `debug_assert_nounwind!` were causing the macro to expand to `panic_nouwnind_fmt`, which requires more work to set up its arguments, and that overhead alone is measured between this perf run and the next: https://github.com/rust-lang/rust/pull/120863#issuecomment-1937423502
This commit is contained in:
commit
2b43e75c98
10 changed files with 67 additions and 367 deletions
|
@ -139,27 +139,38 @@ pub macro unreachable_2021 {
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Asserts that a boolean expression is `true`, and perform a non-unwinding panic otherwise.
|
/// Like `assert_unsafe_precondition!` the defining features of this macro are that its
|
||||||
|
/// checks are enabled when they are monomorphized with debug assertions enabled, and upon failure
|
||||||
|
/// a non-unwinding panic is launched so that failures cannot compromise unwind safety.
|
||||||
///
|
///
|
||||||
/// This macro is similar to `debug_assert!`, but is intended to be used in code that should not
|
/// But there are many differences from `assert_unsafe_precondition!`. This macro does not use
|
||||||
/// unwind. For example, checks in `_unchecked` functions that are intended for debugging but should
|
/// `const_eval_select` internally, and therefore it is sound to call this with an expression
|
||||||
/// not compromise unwind safety.
|
/// that evaluates to `false`. Also unlike `assert_unsafe_precondition!` the condition being
|
||||||
|
/// checked here is not put in an outlined function. If the check compiles to a lot of IR, this
|
||||||
|
/// can cause code bloat if the check is monomorphized many times. But it also means that the checks
|
||||||
|
/// from this macro can be deduplicated or otherwise optimized out.
|
||||||
|
///
|
||||||
|
/// In general, this macro should be used to check all public-facing preconditions. But some
|
||||||
|
/// preconditions may be called too often or instantiated too often to make the overhead of the
|
||||||
|
/// checks tolerable. In such cases, place `#[cfg(debug_assertions)]` on the macro call. That will
|
||||||
|
/// disable the check in our precompiled standard library, but if a user wishes, they can still
|
||||||
|
/// enable the check by recompiling the standard library with debug assertions enabled.
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
#[unstable(feature = "panic_internals", issue = "none")]
|
#[unstable(feature = "panic_internals", issue = "none")]
|
||||||
#[allow_internal_unstable(panic_internals, const_format_args)]
|
#[allow_internal_unstable(panic_internals, delayed_debug_assertions)]
|
||||||
#[rustc_macro_transparency = "semitransparent"]
|
#[rustc_macro_transparency = "semitransparent"]
|
||||||
pub macro debug_assert_nounwind {
|
pub macro debug_assert_nounwind {
|
||||||
($cond:expr $(,)?) => {
|
($cond:expr $(,)?) => {
|
||||||
if $crate::cfg!(debug_assertions) {
|
if $crate::intrinsics::debug_assertions() {
|
||||||
if !$cond {
|
if !$cond {
|
||||||
$crate::panicking::panic_nounwind($crate::concat!("assertion failed: ", $crate::stringify!($cond)));
|
$crate::panicking::panic_nounwind($crate::concat!("assertion failed: ", $crate::stringify!($cond)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
($cond:expr, $($arg:tt)+) => {
|
($cond:expr, $message:expr) => {
|
||||||
if $crate::cfg!(debug_assertions) {
|
if $crate::intrinsics::debug_assertions() {
|
||||||
if !$cond {
|
if !$cond {
|
||||||
$crate::panicking::panic_nounwind_fmt($crate::const_format_args!($($arg)+), false);
|
$crate::panicking::panic_nounwind($message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
|
@ -76,6 +76,7 @@ impl Alignment {
|
||||||
#[rustc_const_unstable(feature = "ptr_alignment_type", issue = "102070")]
|
#[rustc_const_unstable(feature = "ptr_alignment_type", issue = "102070")]
|
||||||
#[inline]
|
#[inline]
|
||||||
pub const unsafe fn new_unchecked(align: usize) -> Self {
|
pub const unsafe fn new_unchecked(align: usize) -> Self {
|
||||||
|
#[cfg(debug_assertions)]
|
||||||
crate::panic::debug_assert_nounwind!(
|
crate::panic::debug_assert_nounwind!(
|
||||||
align.is_power_of_two(),
|
align.is_power_of_two(),
|
||||||
"Alignment::new_unchecked requires a power of two"
|
"Alignment::new_unchecked requires a power of two"
|
||||||
|
|
|
@ -13,7 +13,7 @@ where
|
||||||
{
|
{
|
||||||
type Output = I::Output;
|
type Output = I::Output;
|
||||||
|
|
||||||
#[inline]
|
#[inline(always)]
|
||||||
fn index(&self, index: I) -> &I::Output {
|
fn index(&self, index: I) -> &I::Output {
|
||||||
index.index(self)
|
index.index(self)
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,7 @@ impl<T, I> ops::IndexMut<I> for [T]
|
||||||
where
|
where
|
||||||
I: SliceIndex<[T]>,
|
I: SliceIndex<[T]>,
|
||||||
{
|
{
|
||||||
#[inline]
|
#[inline(always)]
|
||||||
fn index_mut(&mut self, index: I) -> &mut I::Output {
|
fn index_mut(&mut self, index: I) -> &mut I::Output {
|
||||||
index.index_mut(self)
|
index.index_mut(self)
|
||||||
}
|
}
|
||||||
|
@ -227,14 +227,16 @@ unsafe impl<T> SliceIndex<[T]> for usize {
|
||||||
unsafe fn get_unchecked(self, slice: *const [T]) -> *const T {
|
unsafe fn get_unchecked(self, slice: *const [T]) -> *const T {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self < slice.len(),
|
self < slice.len(),
|
||||||
"slice::get_unchecked requires that the index is within the slice",
|
"slice::get_unchecked requires that the index is within the slice"
|
||||||
);
|
);
|
||||||
// SAFETY: the caller guarantees that `slice` is not dangling, so it
|
// SAFETY: the caller guarantees that `slice` is not dangling, so it
|
||||||
// cannot be longer than `isize::MAX`. They also guarantee that
|
// cannot be longer than `isize::MAX`. They also guarantee that
|
||||||
// `self` is in bounds of `slice` so `self` cannot overflow an `isize`,
|
// `self` is in bounds of `slice` so `self` cannot overflow an `isize`,
|
||||||
// so the call to `add` is safe.
|
// so the call to `add` is safe.
|
||||||
unsafe {
|
unsafe {
|
||||||
crate::hint::assert_unchecked(self < slice.len());
|
// 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());
|
||||||
slice.as_ptr().add(self)
|
slice.as_ptr().add(self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -243,7 +245,7 @@ unsafe impl<T> SliceIndex<[T]> for usize {
|
||||||
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut T {
|
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut T {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self < slice.len(),
|
self < slice.len(),
|
||||||
"slice::get_unchecked_mut requires that the index is within the slice",
|
"slice::get_unchecked_mut requires that the index is within the slice"
|
||||||
);
|
);
|
||||||
// SAFETY: see comments for `get_unchecked` above.
|
// SAFETY: see comments for `get_unchecked` above.
|
||||||
unsafe { slice.as_mut_ptr().add(self) }
|
unsafe { slice.as_mut_ptr().add(self) }
|
||||||
|
@ -305,8 +307,9 @@ unsafe impl<T> SliceIndex<[T]> for ops::IndexRange {
|
||||||
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] {
|
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self.end() <= slice.len(),
|
self.end() <= slice.len(),
|
||||||
"slice::get_unchecked_mut requires that the index is within the slice",
|
"slice::get_unchecked_mut requires that the index is within the slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: see comments for `get_unchecked` above.
|
// SAFETY: see comments for `get_unchecked` above.
|
||||||
unsafe { ptr::slice_from_raw_parts_mut(slice.as_mut_ptr().add(self.start()), self.len()) }
|
unsafe { ptr::slice_from_raw_parts_mut(slice.as_mut_ptr().add(self.start()), self.len()) }
|
||||||
}
|
}
|
||||||
|
@ -361,8 +364,9 @@ unsafe impl<T> SliceIndex<[T]> for ops::Range<usize> {
|
||||||
unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] {
|
unsafe fn get_unchecked(self, slice: *const [T]) -> *const [T] {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self.end >= self.start && self.end <= slice.len(),
|
self.end >= self.start && self.end <= slice.len(),
|
||||||
"slice::get_unchecked requires that the range is within the slice",
|
"slice::get_unchecked requires that the range is within the slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: the caller guarantees that `slice` is not dangling, so it
|
// SAFETY: the caller guarantees that `slice` is not dangling, so it
|
||||||
// cannot be longer than `isize::MAX`. They also guarantee that
|
// cannot be longer than `isize::MAX`. They also guarantee that
|
||||||
// `self` is in bounds of `slice` so `self` cannot overflow an `isize`,
|
// `self` is in bounds of `slice` so `self` cannot overflow an `isize`,
|
||||||
|
@ -377,7 +381,7 @@ unsafe impl<T> SliceIndex<[T]> for ops::Range<usize> {
|
||||||
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] {
|
unsafe fn get_unchecked_mut(self, slice: *mut [T]) -> *mut [T] {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self.end >= self.start && self.end <= slice.len(),
|
self.end >= self.start && self.end <= slice.len(),
|
||||||
"slice::get_unchecked_mut requires that the range is within the slice",
|
"slice::get_unchecked_mut requires that the range is within the slice"
|
||||||
);
|
);
|
||||||
// SAFETY: see comments for `get_unchecked` above.
|
// SAFETY: see comments for `get_unchecked` above.
|
||||||
unsafe {
|
unsafe {
|
||||||
|
@ -386,7 +390,7 @@ unsafe impl<T> SliceIndex<[T]> for ops::Range<usize> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline(always)]
|
||||||
fn index(self, slice: &[T]) -> &[T] {
|
fn index(self, slice: &[T]) -> &[T] {
|
||||||
if self.start > self.end {
|
if self.start > self.end {
|
||||||
slice_index_order_fail(self.start, self.end);
|
slice_index_order_fail(self.start, self.end);
|
||||||
|
@ -436,7 +440,7 @@ unsafe impl<T> SliceIndex<[T]> for ops::RangeTo<usize> {
|
||||||
unsafe { (0..self.end).get_unchecked_mut(slice) }
|
unsafe { (0..self.end).get_unchecked_mut(slice) }
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline(always)]
|
||||||
fn index(self, slice: &[T]) -> &[T] {
|
fn index(self, slice: &[T]) -> &[T] {
|
||||||
(0..self.end).index(slice)
|
(0..self.end).index(slice)
|
||||||
}
|
}
|
||||||
|
|
|
@ -938,7 +938,7 @@ impl<T> [T] {
|
||||||
pub const unsafe fn swap_unchecked(&mut self, a: usize, b: usize) {
|
pub const unsafe fn swap_unchecked(&mut self, a: usize, b: usize) {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
a < self.len() && b < self.len(),
|
a < self.len() && b < self.len(),
|
||||||
"slice::swap_unchecked requires that the indices are within the slice",
|
"slice::swap_unchecked requires that the indices are within the slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
let ptr = self.as_mut_ptr();
|
let ptr = self.as_mut_ptr();
|
||||||
|
@ -1278,7 +1278,7 @@ impl<T> [T] {
|
||||||
pub const unsafe fn as_chunks_unchecked<const N: usize>(&self) -> &[[T; N]] {
|
pub const unsafe fn as_chunks_unchecked<const N: usize>(&self) -> &[[T; N]] {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
N != 0 && self.len() % N == 0,
|
N != 0 && self.len() % N == 0,
|
||||||
"slice::as_chunks_unchecked requires `N != 0` and the slice to split exactly into `N`-element chunks",
|
"slice::as_chunks_unchecked requires `N != 0` and the slice to split exactly into `N`-element chunks"
|
||||||
);
|
);
|
||||||
// SAFETY: Caller must guarantee that `N` is nonzero and exactly divides the slice length
|
// SAFETY: Caller must guarantee that `N` is nonzero and exactly divides the slice length
|
||||||
let new_len = unsafe { exact_div(self.len(), N) };
|
let new_len = unsafe { exact_div(self.len(), N) };
|
||||||
|
@ -1432,7 +1432,7 @@ impl<T> [T] {
|
||||||
pub const unsafe fn as_chunks_unchecked_mut<const N: usize>(&mut self) -> &mut [[T; N]] {
|
pub const unsafe fn as_chunks_unchecked_mut<const N: usize>(&mut self) -> &mut [[T; N]] {
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
N != 0 && self.len() % N == 0,
|
N != 0 && self.len() % N == 0,
|
||||||
"slice::as_chunks_unchecked requires `N != 0` and the slice to split exactly into `N`-element chunks",
|
"slice::as_chunks_unchecked requires `N != 0` and the slice to split exactly into `N`-element chunks"
|
||||||
);
|
);
|
||||||
// SAFETY: Caller must guarantee that `N` is nonzero and exactly divides the slice length
|
// SAFETY: Caller must guarantee that `N` is nonzero and exactly divides the slice length
|
||||||
let new_len = unsafe { exact_div(self.len(), N) };
|
let new_len = unsafe { exact_div(self.len(), N) };
|
||||||
|
@ -1964,7 +1964,7 @@ impl<T> [T] {
|
||||||
|
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
mid <= len,
|
mid <= len,
|
||||||
"slice::split_at_unchecked requires the index to be within the slice",
|
"slice::split_at_unchecked requires the index to be within the slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: Caller has to check that `0 <= mid <= self.len()`
|
// SAFETY: Caller has to check that `0 <= mid <= self.len()`
|
||||||
|
@ -2014,7 +2014,7 @@ impl<T> [T] {
|
||||||
|
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
mid <= len,
|
mid <= len,
|
||||||
"slice::split_at_mut_unchecked requires the index to be within the slice",
|
"slice::split_at_mut_unchecked requires the index to be within the slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: Caller has to check that `0 <= mid <= self.len()`.
|
// SAFETY: Caller has to check that `0 <= mid <= self.len()`.
|
||||||
|
|
|
@ -200,7 +200,7 @@ unsafe impl SliceIndex<str> for ops::Range<usize> {
|
||||||
// `str::get_unchecked` without adding a special function
|
// `str::get_unchecked` without adding a special function
|
||||||
// to `SliceIndex` just for this.
|
// to `SliceIndex` just for this.
|
||||||
self.end >= self.start && self.end <= slice.len(),
|
self.end >= self.start && self.end <= slice.len(),
|
||||||
"str::get_unchecked requires that the range is within the string slice",
|
"str::get_unchecked requires that the range is within the string slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: the caller guarantees that `self` is in bounds of `slice`
|
// SAFETY: the caller guarantees that `self` is in bounds of `slice`
|
||||||
|
@ -215,7 +215,7 @@ unsafe impl SliceIndex<str> for ops::Range<usize> {
|
||||||
|
|
||||||
debug_assert_nounwind!(
|
debug_assert_nounwind!(
|
||||||
self.end >= self.start && self.end <= slice.len(),
|
self.end >= self.start && self.end <= slice.len(),
|
||||||
"str::get_unchecked_mut requires that the range is within the string slice",
|
"str::get_unchecked_mut requires that the range is within the string slice"
|
||||||
);
|
);
|
||||||
|
|
||||||
// SAFETY: see comments for `get_unchecked`.
|
// SAFETY: see comments for `get_unchecked`.
|
||||||
|
|
|
@ -7,89 +7,13 @@ fn slice_get_mut_usize(_1: &mut [u32], _2: usize) -> Option<&mut u32> {
|
||||||
scope 1 (inlined core::slice::<impl [u32]>::get_mut::<usize>) {
|
scope 1 (inlined core::slice::<impl [u32]>::get_mut::<usize>) {
|
||||||
debug self => _1;
|
debug self => _1;
|
||||||
debug index => _2;
|
debug index => _2;
|
||||||
scope 2 (inlined <usize as SliceIndex<[u32]>>::get_mut) {
|
|
||||||
debug self => _2;
|
|
||||||
debug slice => _1;
|
|
||||||
let mut _3: usize;
|
|
||||||
let mut _4: bool;
|
|
||||||
let mut _5: *mut [u32];
|
|
||||||
let mut _7: *mut u32;
|
|
||||||
let mut _8: &mut u32;
|
|
||||||
scope 3 {
|
|
||||||
scope 4 (inlined <usize as SliceIndex<[u32]>>::get_unchecked_mut) {
|
|
||||||
debug self => _2;
|
|
||||||
debug slice => _5;
|
|
||||||
let mut _6: *mut u32;
|
|
||||||
let mut _9: *mut [u32];
|
|
||||||
let mut _10: &[&str];
|
|
||||||
scope 5 {
|
|
||||||
scope 10 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::as_mut_ptr) {
|
|
||||||
debug self => _5;
|
|
||||||
}
|
|
||||||
scope 11 (inlined std::ptr::mut_ptr::<impl *mut u32>::add) {
|
|
||||||
debug self => _6;
|
|
||||||
debug count => _2;
|
|
||||||
scope 12 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 6 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::len) {
|
|
||||||
debug self => _9;
|
|
||||||
let mut _11: *const [u32];
|
|
||||||
scope 7 (inlined std::ptr::metadata::<[u32]>) {
|
|
||||||
debug ptr => _11;
|
|
||||||
scope 8 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 9 (inlined Arguments::<'_>::new_const) {
|
|
||||||
debug pieces => _10;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bb0: {
|
bb0: {
|
||||||
StorageLive(_7);
|
_0 = <usize as SliceIndex<[u32]>>::get_mut(move _2, move _1) -> [return: bb1, unwind unreachable];
|
||||||
StorageLive(_4);
|
|
||||||
StorageLive(_3);
|
|
||||||
_3 = Len((*_1));
|
|
||||||
_4 = Lt(_2, move _3);
|
|
||||||
switchInt(move _4) -> [0: bb1, otherwise: bb2];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bb1: {
|
bb1: {
|
||||||
StorageDead(_3);
|
|
||||||
_0 = const Option::<&mut u32>::None;
|
|
||||||
goto -> bb3;
|
|
||||||
}
|
|
||||||
|
|
||||||
bb2: {
|
|
||||||
StorageDead(_3);
|
|
||||||
StorageLive(_8);
|
|
||||||
StorageLive(_5);
|
|
||||||
_5 = &raw mut (*_1);
|
|
||||||
StorageLive(_9);
|
|
||||||
StorageLive(_10);
|
|
||||||
StorageLive(_11);
|
|
||||||
StorageLive(_6);
|
|
||||||
_6 = _5 as *mut u32 (PtrToPtr);
|
|
||||||
_7 = Offset(_6, _2);
|
|
||||||
StorageDead(_6);
|
|
||||||
StorageDead(_11);
|
|
||||||
StorageDead(_10);
|
|
||||||
StorageDead(_9);
|
|
||||||
StorageDead(_5);
|
|
||||||
_8 = &mut (*_7);
|
|
||||||
_0 = Option::<&mut u32>::Some(move _8);
|
|
||||||
StorageDead(_8);
|
|
||||||
goto -> bb3;
|
|
||||||
}
|
|
||||||
|
|
||||||
bb3: {
|
|
||||||
StorageDead(_4);
|
|
||||||
StorageDead(_7);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,89 +7,13 @@ fn slice_get_mut_usize(_1: &mut [u32], _2: usize) -> Option<&mut u32> {
|
||||||
scope 1 (inlined core::slice::<impl [u32]>::get_mut::<usize>) {
|
scope 1 (inlined core::slice::<impl [u32]>::get_mut::<usize>) {
|
||||||
debug self => _1;
|
debug self => _1;
|
||||||
debug index => _2;
|
debug index => _2;
|
||||||
scope 2 (inlined <usize as SliceIndex<[u32]>>::get_mut) {
|
|
||||||
debug self => _2;
|
|
||||||
debug slice => _1;
|
|
||||||
let mut _3: usize;
|
|
||||||
let mut _4: bool;
|
|
||||||
let mut _5: *mut [u32];
|
|
||||||
let mut _7: *mut u32;
|
|
||||||
let mut _8: &mut u32;
|
|
||||||
scope 3 {
|
|
||||||
scope 4 (inlined <usize as SliceIndex<[u32]>>::get_unchecked_mut) {
|
|
||||||
debug self => _2;
|
|
||||||
debug slice => _5;
|
|
||||||
let mut _6: *mut u32;
|
|
||||||
let mut _9: *mut [u32];
|
|
||||||
let mut _10: &[&str];
|
|
||||||
scope 5 {
|
|
||||||
scope 10 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::as_mut_ptr) {
|
|
||||||
debug self => _5;
|
|
||||||
}
|
|
||||||
scope 11 (inlined std::ptr::mut_ptr::<impl *mut u32>::add) {
|
|
||||||
debug self => _6;
|
|
||||||
debug count => _2;
|
|
||||||
scope 12 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 6 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::len) {
|
|
||||||
debug self => _9;
|
|
||||||
let mut _11: *const [u32];
|
|
||||||
scope 7 (inlined std::ptr::metadata::<[u32]>) {
|
|
||||||
debug ptr => _11;
|
|
||||||
scope 8 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 9 (inlined Arguments::<'_>::new_const) {
|
|
||||||
debug pieces => _10;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bb0: {
|
bb0: {
|
||||||
StorageLive(_7);
|
_0 = <usize as SliceIndex<[u32]>>::get_mut(move _2, move _1) -> [return: bb1, unwind continue];
|
||||||
StorageLive(_4);
|
|
||||||
StorageLive(_3);
|
|
||||||
_3 = Len((*_1));
|
|
||||||
_4 = Lt(_2, move _3);
|
|
||||||
switchInt(move _4) -> [0: bb1, otherwise: bb2];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bb1: {
|
bb1: {
|
||||||
StorageDead(_3);
|
|
||||||
_0 = const Option::<&mut u32>::None;
|
|
||||||
goto -> bb3;
|
|
||||||
}
|
|
||||||
|
|
||||||
bb2: {
|
|
||||||
StorageDead(_3);
|
|
||||||
StorageLive(_8);
|
|
||||||
StorageLive(_5);
|
|
||||||
_5 = &raw mut (*_1);
|
|
||||||
StorageLive(_9);
|
|
||||||
StorageLive(_10);
|
|
||||||
StorageLive(_11);
|
|
||||||
StorageLive(_6);
|
|
||||||
_6 = _5 as *mut u32 (PtrToPtr);
|
|
||||||
_7 = Offset(_6, _2);
|
|
||||||
StorageDead(_6);
|
|
||||||
StorageDead(_11);
|
|
||||||
StorageDead(_10);
|
|
||||||
StorageDead(_9);
|
|
||||||
StorageDead(_5);
|
|
||||||
_8 = &mut (*_7);
|
|
||||||
_0 = Option::<&mut u32>::Some(move _8);
|
|
||||||
StorageDead(_8);
|
|
||||||
goto -> bb3;
|
|
||||||
}
|
|
||||||
|
|
||||||
bb3: {
|
|
||||||
StorageDead(_4);
|
|
||||||
StorageDead(_7);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,106 +4,24 @@ fn slice_get_unchecked_mut_range(_1: &mut [u32], _2: std::ops::Range<usize>) ->
|
||||||
debug slice => _1;
|
debug slice => _1;
|
||||||
debug index => _2;
|
debug index => _2;
|
||||||
let mut _0: &mut [u32];
|
let mut _0: &mut [u32];
|
||||||
let mut _3: usize;
|
|
||||||
let mut _4: usize;
|
|
||||||
scope 1 (inlined core::slice::<impl [u32]>::get_unchecked_mut::<std::ops::Range<usize>>) {
|
scope 1 (inlined core::slice::<impl [u32]>::get_unchecked_mut::<std::ops::Range<usize>>) {
|
||||||
debug self => _1;
|
debug self => _1;
|
||||||
debug ((index: std::ops::Range<usize>).0: usize) => _3;
|
debug index => _2;
|
||||||
debug ((index: std::ops::Range<usize>).1: usize) => _4;
|
let mut _3: *mut [u32];
|
||||||
let mut _5: *mut [u32];
|
let mut _4: *mut [u32];
|
||||||
let mut _13: *mut [u32];
|
|
||||||
scope 2 {
|
scope 2 {
|
||||||
scope 3 (inlined <std::ops::Range<usize> as SliceIndex<[u32]>>::get_unchecked_mut) {
|
|
||||||
debug ((self: std::ops::Range<usize>).0: usize) => _3;
|
|
||||||
debug ((self: std::ops::Range<usize>).1: usize) => _4;
|
|
||||||
debug slice => _5;
|
|
||||||
let mut _7: *mut u32;
|
|
||||||
let mut _8: *mut u32;
|
|
||||||
let mut _14: *mut [u32];
|
|
||||||
let mut _15: &[&str];
|
|
||||||
scope 4 {
|
|
||||||
let _6: usize;
|
|
||||||
scope 5 {
|
|
||||||
debug new_len => _6;
|
|
||||||
scope 10 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::as_mut_ptr) {
|
|
||||||
debug self => _5;
|
|
||||||
}
|
|
||||||
scope 11 (inlined std::ptr::mut_ptr::<impl *mut u32>::add) {
|
|
||||||
debug self => _7;
|
|
||||||
debug count => _3;
|
|
||||||
scope 12 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 13 (inlined slice_from_raw_parts_mut::<u32>) {
|
|
||||||
debug data => _8;
|
|
||||||
debug len => _6;
|
|
||||||
let mut _9: *mut ();
|
|
||||||
scope 14 (inlined std::ptr::mut_ptr::<impl *mut u32>::cast::<()>) {
|
|
||||||
debug self => _8;
|
|
||||||
}
|
|
||||||
scope 15 (inlined std::ptr::from_raw_parts_mut::<[u32]>) {
|
|
||||||
debug data_pointer => _9;
|
|
||||||
debug metadata => _6;
|
|
||||||
let mut _10: *const ();
|
|
||||||
let mut _11: std::ptr::metadata::PtrComponents<[u32]>;
|
|
||||||
let mut _12: std::ptr::metadata::PtrRepr<[u32]>;
|
|
||||||
scope 16 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 6 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::len) {
|
|
||||||
debug self => _14;
|
|
||||||
let mut _16: *const [u32];
|
|
||||||
scope 7 (inlined std::ptr::metadata::<[u32]>) {
|
|
||||||
debug ptr => _16;
|
|
||||||
scope 8 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 9 (inlined Arguments::<'_>::new_const) {
|
|
||||||
debug pieces => _15;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bb0: {
|
bb0: {
|
||||||
_3 = move (_2.0: usize);
|
StorageLive(_3);
|
||||||
_4 = move (_2.1: usize);
|
_3 = &raw mut (*_1);
|
||||||
StorageLive(_5);
|
_4 = <std::ops::Range<usize> as SliceIndex<[u32]>>::get_unchecked_mut(move _2, move _3) -> [return: bb1, unwind unreachable];
|
||||||
_5 = &raw mut (*_1);
|
}
|
||||||
StorageLive(_14);
|
|
||||||
StorageLive(_15);
|
bb1: {
|
||||||
StorageLive(_6);
|
StorageDead(_3);
|
||||||
StorageLive(_16);
|
_0 = &mut (*_4);
|
||||||
_6 = SubUnchecked(_4, _3);
|
|
||||||
StorageLive(_8);
|
|
||||||
StorageLive(_7);
|
|
||||||
_7 = _5 as *mut u32 (PtrToPtr);
|
|
||||||
_8 = Offset(_7, _3);
|
|
||||||
StorageDead(_7);
|
|
||||||
StorageLive(_9);
|
|
||||||
_9 = _8 as *mut () (PtrToPtr);
|
|
||||||
StorageLive(_12);
|
|
||||||
StorageLive(_11);
|
|
||||||
StorageLive(_10);
|
|
||||||
_10 = _8 as *const () (PtrToPtr);
|
|
||||||
_11 = std::ptr::metadata::PtrComponents::<[u32]> { data_pointer: move _10, metadata: _6 };
|
|
||||||
StorageDead(_10);
|
|
||||||
_12 = std::ptr::metadata::PtrRepr::<[u32]> { const_ptr: move _11 };
|
|
||||||
StorageDead(_11);
|
|
||||||
_13 = (_12.1: *mut [u32]);
|
|
||||||
StorageDead(_12);
|
|
||||||
StorageDead(_9);
|
|
||||||
StorageDead(_8);
|
|
||||||
StorageDead(_16);
|
|
||||||
StorageDead(_6);
|
|
||||||
StorageDead(_15);
|
|
||||||
StorageDead(_14);
|
|
||||||
StorageDead(_5);
|
|
||||||
_0 = &mut (*_13);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,106 +4,24 @@ fn slice_get_unchecked_mut_range(_1: &mut [u32], _2: std::ops::Range<usize>) ->
|
||||||
debug slice => _1;
|
debug slice => _1;
|
||||||
debug index => _2;
|
debug index => _2;
|
||||||
let mut _0: &mut [u32];
|
let mut _0: &mut [u32];
|
||||||
let mut _3: usize;
|
|
||||||
let mut _4: usize;
|
|
||||||
scope 1 (inlined core::slice::<impl [u32]>::get_unchecked_mut::<std::ops::Range<usize>>) {
|
scope 1 (inlined core::slice::<impl [u32]>::get_unchecked_mut::<std::ops::Range<usize>>) {
|
||||||
debug self => _1;
|
debug self => _1;
|
||||||
debug ((index: std::ops::Range<usize>).0: usize) => _3;
|
debug index => _2;
|
||||||
debug ((index: std::ops::Range<usize>).1: usize) => _4;
|
let mut _3: *mut [u32];
|
||||||
let mut _5: *mut [u32];
|
let mut _4: *mut [u32];
|
||||||
let mut _13: *mut [u32];
|
|
||||||
scope 2 {
|
scope 2 {
|
||||||
scope 3 (inlined <std::ops::Range<usize> as SliceIndex<[u32]>>::get_unchecked_mut) {
|
|
||||||
debug ((self: std::ops::Range<usize>).0: usize) => _3;
|
|
||||||
debug ((self: std::ops::Range<usize>).1: usize) => _4;
|
|
||||||
debug slice => _5;
|
|
||||||
let mut _7: *mut u32;
|
|
||||||
let mut _8: *mut u32;
|
|
||||||
let mut _14: *mut [u32];
|
|
||||||
let mut _15: &[&str];
|
|
||||||
scope 4 {
|
|
||||||
let _6: usize;
|
|
||||||
scope 5 {
|
|
||||||
debug new_len => _6;
|
|
||||||
scope 10 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::as_mut_ptr) {
|
|
||||||
debug self => _5;
|
|
||||||
}
|
|
||||||
scope 11 (inlined std::ptr::mut_ptr::<impl *mut u32>::add) {
|
|
||||||
debug self => _7;
|
|
||||||
debug count => _3;
|
|
||||||
scope 12 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 13 (inlined slice_from_raw_parts_mut::<u32>) {
|
|
||||||
debug data => _8;
|
|
||||||
debug len => _6;
|
|
||||||
let mut _9: *mut ();
|
|
||||||
scope 14 (inlined std::ptr::mut_ptr::<impl *mut u32>::cast::<()>) {
|
|
||||||
debug self => _8;
|
|
||||||
}
|
|
||||||
scope 15 (inlined std::ptr::from_raw_parts_mut::<[u32]>) {
|
|
||||||
debug data_pointer => _9;
|
|
||||||
debug metadata => _6;
|
|
||||||
let mut _10: *const ();
|
|
||||||
let mut _11: std::ptr::metadata::PtrComponents<[u32]>;
|
|
||||||
let mut _12: std::ptr::metadata::PtrRepr<[u32]>;
|
|
||||||
scope 16 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 6 (inlined std::ptr::mut_ptr::<impl *mut [u32]>::len) {
|
|
||||||
debug self => _14;
|
|
||||||
let mut _16: *const [u32];
|
|
||||||
scope 7 (inlined std::ptr::metadata::<[u32]>) {
|
|
||||||
debug ptr => _16;
|
|
||||||
scope 8 {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope 9 (inlined Arguments::<'_>::new_const) {
|
|
||||||
debug pieces => _15;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bb0: {
|
bb0: {
|
||||||
_3 = move (_2.0: usize);
|
StorageLive(_3);
|
||||||
_4 = move (_2.1: usize);
|
_3 = &raw mut (*_1);
|
||||||
StorageLive(_5);
|
_4 = <std::ops::Range<usize> as SliceIndex<[u32]>>::get_unchecked_mut(move _2, move _3) -> [return: bb1, unwind continue];
|
||||||
_5 = &raw mut (*_1);
|
}
|
||||||
StorageLive(_14);
|
|
||||||
StorageLive(_15);
|
bb1: {
|
||||||
StorageLive(_6);
|
StorageDead(_3);
|
||||||
StorageLive(_16);
|
_0 = &mut (*_4);
|
||||||
_6 = SubUnchecked(_4, _3);
|
|
||||||
StorageLive(_8);
|
|
||||||
StorageLive(_7);
|
|
||||||
_7 = _5 as *mut u32 (PtrToPtr);
|
|
||||||
_8 = Offset(_7, _3);
|
|
||||||
StorageDead(_7);
|
|
||||||
StorageLive(_9);
|
|
||||||
_9 = _8 as *mut () (PtrToPtr);
|
|
||||||
StorageLive(_12);
|
|
||||||
StorageLive(_11);
|
|
||||||
StorageLive(_10);
|
|
||||||
_10 = _8 as *const () (PtrToPtr);
|
|
||||||
_11 = std::ptr::metadata::PtrComponents::<[u32]> { data_pointer: move _10, metadata: _6 };
|
|
||||||
StorageDead(_10);
|
|
||||||
_12 = std::ptr::metadata::PtrRepr::<[u32]> { const_ptr: move _11 };
|
|
||||||
StorageDead(_11);
|
|
||||||
_13 = (_12.1: *mut [u32]);
|
|
||||||
StorageDead(_12);
|
|
||||||
StorageDead(_9);
|
|
||||||
StorageDead(_8);
|
|
||||||
StorageDead(_16);
|
|
||||||
StorageDead(_6);
|
|
||||||
StorageDead(_15);
|
|
||||||
StorageDead(_14);
|
|
||||||
StorageDead(_5);
|
|
||||||
_0 = &mut (*_13);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
//@ run-fail
|
//@ run-fail
|
||||||
//@ compile-flags: -Copt-level=3 -Cdebug-assertions=yes
|
//@ compile-flags: -Copt-level=3 -Cdebug-assertions=yes
|
||||||
//@ error-pattern: unsafe precondition(s) violated: hint::assert_unchecked
|
//@ error-pattern: slice::get_unchecked requires
|
||||||
//@ ignore-debug
|
//@ ignore-debug
|
||||||
//@ ignore-wasm32-bare no panic messages
|
//@ ignore-wasm32-bare no panic messages
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue