1
1
//! Provides the [`assert_unsafe_precondition`] macro as well as some utility functions that cover
2
2
//! common preconditions.
3
3
4
- use crate :: intrinsics:: const_eval_select;
4
+ use crate :: intrinsics:: { self , const_eval_select} ;
5
5
6
6
/// Check that the preconditions of an unsafe function are followed. The check is enabled at
7
7
/// runtime if debug assertions are enabled when the caller is monomorphized. In const-eval/Miri
@@ -45,7 +45,7 @@ use crate::intrinsics::const_eval_select;
45
45
/// order to call it. Since the precompiled standard library is built with full debuginfo and these
46
46
/// variables cannot be optimized out in MIR, an innocent-looking `let` can produce enough
47
47
/// debuginfo to have a measurable compile-time impact on debug builds.
48
- #[ allow_internal_unstable( ub_checks ) ] // permit this to be called in stably-const fn
48
+ #[ allow_internal_unstable( const_ub_checks ) ] // permit this to be called in stably-const fn
49
49
macro_rules! assert_unsafe_precondition {
50
50
( $kind: ident, $message: expr, ( $( $name: ident: $ty: ty = $arg: expr) ,* $( , ) ?) => $e: expr $( , ) ?) => {
51
51
{
@@ -69,7 +69,7 @@ macro_rules! assert_unsafe_precondition {
69
69
#[ cfg_attr( not( bootstrap) , rustc_no_mir_inline) ]
70
70
#[ cfg_attr( not( bootstrap) , inline) ]
71
71
#[ rustc_nounwind]
72
- #[ rustc_const_unstable( feature = "ub_checks " , issue = "none" ) ]
72
+ #[ rustc_const_unstable( feature = "const_ub_checks " , issue = "none" ) ]
73
73
const fn precondition_check( $( $name: $ty) ,* ) {
74
74
if !$e {
75
75
:: core:: panicking:: panic_nounwind(
@@ -78,14 +78,44 @@ macro_rules! assert_unsafe_precondition {
78
78
}
79
79
}
80
80
81
- if :: core:: intrinsics :: $kind( ) {
81
+ if :: core:: ub_checks :: $kind( ) {
82
82
precondition_check( $( $arg, ) * ) ;
83
83
}
84
84
}
85
85
} ;
86
86
}
87
87
pub ( crate ) use assert_unsafe_precondition;
88
88
89
+ /// Checking library UB is always enabled when UB-checking is done
90
+ /// (and we use a reexport so that there is no unnecessary wrapper function).
91
+ pub ( crate ) use intrinsics:: ub_checks as check_library_ub;
92
+
93
+ /// Determines whether we should check for language UB.
94
+ ///
95
+ /// The intention is to not do that when running in the interpreter, as that one has its own
96
+ /// language UB checks which generally produce better errors.
97
+ #[ rustc_const_unstable( feature = "const_ub_checks" , issue = "none" ) ]
98
+ pub ( crate ) const fn check_language_ub ( ) -> bool {
99
+ #[ inline]
100
+ fn runtime ( ) -> bool {
101
+ // Disable UB checks in Miri.
102
+ !cfg ! ( miri)
103
+ }
104
+
105
+ #[ inline]
106
+ const fn comptime ( ) -> bool {
107
+ // Always disable UB checks.
108
+ false
109
+ }
110
+
111
+ #[ cfg_attr( not( bootstrap) , allow( unused_unsafe) ) ] // on bootstrap bump, remove unsafe block
112
+ // SAFETY: `const_eval_select` is only used to toggle UB checks here, not to provide any
113
+ // observable behavior differences.
114
+ unsafe {
115
+ intrinsics:: ub_checks ( ) && const_eval_select ( ( ) , comptime, runtime)
116
+ }
117
+ }
118
+
89
119
/// Checks whether `ptr` is properly aligned with respect to
90
120
/// `align_of::<T>()`.
91
121
///
0 commit comments