Skip to content

Commit 2f721c2

Browse files
committed
Auto merge of rust-lang#120594 - saethlin:delayed-debug-asserts, r=<try>
Toggle assert_unsafe_precondition in codegen instead of expansion r? `@ghost` rust-lang#120539 (comment)
2 parents b11fbfb + 3b4fe4a commit 2f721c2

File tree

32 files changed

+1020
-1206
lines changed

32 files changed

+1020
-1206
lines changed

compiler/rustc_codegen_cranelift/src/intrinsics/mod.rs

+8
Original file line numberDiff line numberDiff line change
@@ -438,6 +438,14 @@ fn codegen_regular_intrinsic_call<'tcx>(
438438
fx.bcx.ins().trap(TrapCode::User(0));
439439
return;
440440
}
441+
sym::debug_assertions => {
442+
let bool_layout = fx.layout_of(fx.tcx.types.bool);
443+
let val = CValue::by_val(
444+
fx.bcx.ins().iconst(types::I8, fx.tcx.sess.opts.debug_assertions as i64),
445+
bool_layout,
446+
);
447+
ret.write_cvalue(fx, val);
448+
}
441449
sym::likely | sym::unlikely => {
442450
intrinsic_args!(fx, args => (a); intrinsic);
443451

compiler/rustc_codegen_ssa/src/mir/intrinsic.rs

+1
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,7 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
8484
return;
8585
}
8686

87+
sym::debug_assertions => bx.const_bool(bx.tcx().sess.opts.debug_assertions),
8788
sym::va_start => bx.va_start(args[0].immediate()),
8889
sym::va_end => bx.va_end(args[0].immediate()),
8990
sym::size_of_val => {

compiler/rustc_const_eval/src/const_eval/machine.rs

+5
Original file line numberDiff line numberDiff line change
@@ -536,6 +536,11 @@ impl<'mir, 'tcx> interpret::Machine<'mir, 'tcx> for CompileTimeInterpreter<'mir,
536536
// (We know the value here in the machine of course, but this is the runtime of that code,
537537
// not the optimization stage.)
538538
sym::is_val_statically_known => ecx.write_scalar(Scalar::from_bool(false), dest)?,
539+
540+
sym::debug_assertions => {
541+
ecx.write_scalar(Scalar::from_bool(ecx.tcx.sess.opts.debug_assertions), dest)?
542+
}
543+
539544
_ => {
540545
throw_unsup_format!(
541546
"intrinsic `{intrinsic_name}` is not supported at compile-time"

compiler/rustc_hir_analysis/src/check/intrinsic.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -112,7 +112,8 @@ pub fn intrinsic_operation_unsafety(tcx: TyCtxt<'_>, intrinsic_id: DefId) -> hir
112112
| sym::forget
113113
| sym::black_box
114114
| sym::variant_count
115-
| sym::ptr_mask => hir::Unsafety::Normal,
115+
| sym::ptr_mask
116+
| sym::debug_assertions => hir::Unsafety::Normal,
116117
_ => hir::Unsafety::Unsafe,
117118
};
118119

@@ -461,6 +462,8 @@ pub fn check_intrinsic_type(tcx: TyCtxt<'_>, it: &hir::ForeignItem<'_>) {
461462
(0, vec![Ty::new_imm_ptr(tcx, Ty::new_unit(tcx))], tcx.types.usize)
462463
}
463464

465+
sym::debug_assertions => (0, Vec::new(), tcx.types.bool),
466+
464467
other => {
465468
tcx.dcx().emit_err(UnrecognizedIntrinsicFunction { span: it.span, name: other });
466469
return;

library/core/src/intrinsics.rs

+23-3
Original file line numberDiff line numberDiff line change
@@ -2569,6 +2569,12 @@ extern "rust-intrinsic" {
25692569
#[rustc_nounwind]
25702570
#[cfg(not(bootstrap))]
25712571
pub fn is_val_statically_known<T: Copy>(arg: T) -> bool;
2572+
2573+
#[rustc_const_unstable(feature = "delayed_debug_assertions", issue = "none")]
2574+
#[rustc_safe_intrinsic]
2575+
#[rustc_nounwind]
2576+
#[cfg(not(bootstrap))]
2577+
pub(crate) fn debug_assertions() -> bool;
25722578
}
25732579

25742580
// FIXME: Seems using `unstable` here completely ignores `rustc_allow_const_fn_unstable`
@@ -2604,10 +2610,18 @@ pub const unsafe fn is_val_statically_known<T: Copy>(_arg: T) -> bool {
26042610
///
26052611
/// So in a sense it is UB if this macro is useful, but we expect callers of `unsafe fn` to make
26062612
/// the occasional mistake, and this check should help them figure things out.
2607-
#[allow_internal_unstable(const_eval_select)] // permit this to be called in stably-const fn
2613+
#[allow_internal_unstable(const_eval_select, delayed_debug_assertions)] // permit this to be called in stably-const fn
26082614
macro_rules! assert_unsafe_precondition {
26092615
($name:expr, $([$($tt:tt)*])?($($i:ident:$ty:ty),*$(,)?) => $e:expr $(,)?) => {
2610-
if cfg!(debug_assertions) {
2616+
{
2617+
#[cfg(bootstrap)]
2618+
let should_check = cfg!(debug_assertions);
2619+
2620+
// Turn assertions off in Miri, but otherwise check in codegen
2621+
#[cfg(not(bootstrap))]
2622+
let should_check = !cfg!(miri) && ::core::intrinsics::debug_assertions();
2623+
2624+
if should_check {
26112625
// allow non_snake_case to allow capturing const generics
26122626
#[allow(non_snake_case)]
26132627
#[inline(always)]
@@ -2625,6 +2639,7 @@ macro_rules! assert_unsafe_precondition {
26252639

26262640
::core::intrinsics::const_eval_select(($($i,)*), comptime, runtime);
26272641
}
2642+
}
26282643
};
26292644
}
26302645
pub(crate) use assert_unsafe_precondition;
@@ -2633,9 +2648,13 @@ pub(crate) use assert_unsafe_precondition;
26332648
/// `align_of::<T>()`.
26342649
#[inline]
26352650
pub(crate) fn is_aligned_and_not_null<T>(ptr: *const T) -> bool {
2636-
!ptr.is_null() && ptr.is_aligned()
2651+
let mask = const { crate::mem::align_of::<T>() - 1 };
2652+
let is_aligned = (ptr.addr() & mask) == 0;
2653+
let not_null = ptr.addr() != 0;
2654+
is_aligned && not_null
26372655
}
26382656

2657+
/*
26392658
/// Checks whether an allocation of `len` instances of `T` exceeds
26402659
/// the maximum allowed allocation size.
26412660
#[inline]
@@ -2646,6 +2665,7 @@ pub(crate) fn is_valid_allocation_size<T>(len: usize) -> bool {
26462665
};
26472666
len <= max_len
26482667
}
2668+
*/
26492669

26502670
/// Checks whether the regions of memory starting at `src` and `dst` of size
26512671
/// `count * size_of::<T>()` do *not* overlap.

library/core/src/ptr/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -1208,10 +1208,12 @@ pub const unsafe fn read<T>(src: *const T) -> T {
12081208

12091209
// SAFETY: the caller must guarantee that `src` is valid for reads.
12101210
unsafe {
1211+
/*
12111212
assert_unsafe_precondition!(
12121213
"ptr::read requires that the pointer argument is aligned and non-null",
12131214
[T](src: *const T) => is_aligned_and_not_null(src)
12141215
);
1216+
*/
12151217
crate::intrinsics::read_via_copy(src)
12161218
}
12171219
}
@@ -1408,10 +1410,12 @@ pub const unsafe fn write<T>(dst: *mut T, src: T) {
14081410
// `dst` cannot overlap `src` because the caller has mutable access
14091411
// to `dst` while `src` is owned by this function.
14101412
unsafe {
1413+
/*
14111414
assert_unsafe_precondition!(
14121415
"ptr::write requires that the pointer argument is aligned and non-null",
14131416
[T](dst: *mut T) => is_aligned_and_not_null(dst)
14141417
);
1418+
*/
14151419
intrinsics::write_via_move(dst, src)
14161420
}
14171421
}

library/core/src/slice/raw.rs

+6
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
//! Free functions to create `&[T]` and `&mut [T]`.
22
33
use crate::array;
4+
/*
45
use crate::intrinsics::{
56
assert_unsafe_precondition, is_aligned_and_not_null, is_valid_allocation_size,
67
};
8+
*/
79
use crate::ops::Range;
810
use crate::ptr;
911

@@ -94,11 +96,13 @@ use crate::ptr;
9496
pub const unsafe fn from_raw_parts<'a, T>(data: *const T, len: usize) -> &'a [T] {
9597
// SAFETY: the caller must uphold the safety contract for `from_raw_parts`.
9698
unsafe {
99+
/*
97100
assert_unsafe_precondition!(
98101
"slice::from_raw_parts requires the pointer to be aligned and non-null, and the total size of the slice not to exceed `isize::MAX`",
99102
[T](data: *const T, len: usize) => is_aligned_and_not_null(data)
100103
&& is_valid_allocation_size::<T>(len)
101104
);
105+
*/
102106
&*ptr::slice_from_raw_parts(data, len)
103107
}
104108
}
@@ -141,11 +145,13 @@ pub const unsafe fn from_raw_parts<'a, T>(data: *const T, len: usize) -> &'a [T]
141145
pub const unsafe fn from_raw_parts_mut<'a, T>(data: *mut T, len: usize) -> &'a mut [T] {
142146
// SAFETY: the caller must uphold the safety contract for `from_raw_parts_mut`.
143147
unsafe {
148+
/*
144149
assert_unsafe_precondition!(
145150
"slice::from_raw_parts_mut requires the pointer to be aligned and non-null, and the total size of the slice not to exceed `isize::MAX`",
146151
[T](data: *mut T, len: usize) => is_aligned_and_not_null(data)
147152
&& is_valid_allocation_size::<T>(len)
148153
);
154+
*/
149155
&mut *ptr::slice_from_raw_parts_mut(data, len)
150156
}
151157
}

tests/mir-opt/dataflow-const-prop/default_boxed_slice.main.DataflowConstProp.32bit.panic-abort.diff

+58-22
Original file line numberDiff line numberDiff line change
@@ -24,22 +24,28 @@
2424
debug ptr => _6;
2525
scope 11 (inlined NonNull::<[bool; 0]>::new_unchecked) {
2626
debug ptr => _6;
27-
let mut _8: *const [bool; 0];
28-
let mut _9: *mut [bool; 0];
27+
let mut _9: *const [bool; 0];
2928
scope 12 {
30-
scope 13 (inlined NonNull::<T>::new_unchecked::runtime::<[bool; 0]>) {
31-
debug ptr => _9;
32-
scope 14 (inlined std::ptr::mut_ptr::<impl *mut [bool; 0]>::is_null) {
33-
debug self => _9;
34-
let mut _10: *mut u8;
35-
scope 15 {
36-
scope 16 (inlined std::ptr::mut_ptr::<impl *mut T>::is_null::runtime_impl) {
37-
debug ptr => _10;
38-
scope 17 (inlined std::ptr::mut_ptr::<impl *mut u8>::addr) {
39-
debug self => _10;
40-
scope 18 {
41-
scope 19 (inlined std::ptr::mut_ptr::<impl *mut u8>::cast::<()>) {
42-
debug self => _10;
29+
let _8: bool;
30+
scope 13 {
31+
debug should_check => _8;
32+
scope 14 (inlined NonNull::<T>::new_unchecked::runtime::<[bool; 0]>) {
33+
debug ptr => _6;
34+
let _10: !;
35+
scope 15 (inlined std::ptr::mut_ptr::<impl *mut [bool; 0]>::is_null) {
36+
debug self => _6;
37+
let mut _11: *mut u8;
38+
scope 16 {
39+
scope 17 (inlined std::ptr::mut_ptr::<impl *mut T>::is_null::runtime_impl) {
40+
debug ptr => _11;
41+
let mut _12: usize;
42+
scope 18 (inlined std::ptr::mut_ptr::<impl *mut u8>::addr) {
43+
debug self => _11;
44+
let mut _13: *mut ();
45+
scope 19 {
46+
scope 20 (inlined std::ptr::mut_ptr::<impl *mut u8>::cast::<()>) {
47+
debug self => _11;
48+
}
4349
}
4450
}
4551
}
@@ -66,6 +72,7 @@
6672
StorageLive(_1);
6773
StorageLive(_2);
6874
StorageLive(_3);
75+
StorageLive(_10);
6976
StorageLive(_4);
7077
StorageLive(_5);
7178
StorageLive(_6);
@@ -75,10 +82,33 @@
7582
StorageDead(_7);
7683
StorageLive(_8);
7784
StorageLive(_9);
78-
StorageLive(_10);
79-
_8 = const {0x1 as *mut [bool; 0]} as *const [bool; 0] (PointerCoercion(MutToConstPointer));
80-
_5 = NonNull::<[bool; 0]> { pointer: _8 };
81-
StorageDead(_10);
85+
_8 = intrinsics::debug_assertions() -> [return: bb2, unwind unreachable];
86+
}
87+
88+
bb1: {
89+
StorageDead(_1);
90+
return;
91+
}
92+
93+
bb2: {
94+
switchInt(_8) -> [0: bb4, otherwise: bb3];
95+
}
96+
97+
bb3: {
98+
StorageLive(_11);
99+
_11 = const {0x1 as *mut [bool; 0]} as *mut u8 (PtrToPtr);
100+
StorageLive(_12);
101+
StorageLive(_13);
102+
_13 = _11 as *mut () (PtrToPtr);
103+
_12 = move _13 as usize (Transmute);
104+
StorageDead(_13);
105+
StorageDead(_11);
106+
switchInt(move _12) -> [0: bb5, otherwise: bb6];
107+
}
108+
109+
bb4: {
110+
_9 = const {0x1 as *mut [bool; 0]} as *const [bool; 0] (PointerCoercion(MutToConstPointer));
111+
_5 = NonNull::<[bool; 0]> { pointer: _9 };
82112
StorageDead(_9);
83113
StorageDead(_8);
84114
StorageDead(_6);
@@ -87,16 +117,22 @@
87117
_3 = move _4 as std::ptr::Unique<[bool]> (PointerCoercion(Unsize));
88118
StorageDead(_4);
89119
_2 = Box::<[bool]>(_3, const std::alloc::Global);
120+
StorageDead(_10);
90121
StorageDead(_3);
91122
_1 = A { foo: move _2 };
92123
StorageDead(_2);
93124
_0 = const ();
94125
drop(_1) -> [return: bb1, unwind unreachable];
95126
}
96127

97-
bb1: {
98-
StorageDead(_1);
99-
return;
128+
bb5: {
129+
StorageDead(_12);
130+
_10 = core::panicking::panic_nounwind(const "unsafe precondition(s) violated: NonNull::new_unchecked requires that the pointer is non-null") -> unwind unreachable;
131+
}
132+
133+
bb6: {
134+
StorageDead(_12);
135+
goto -> bb4;
100136
}
101137
}
102138

0 commit comments

Comments
 (0)