Skip to content

Commit 6fef5f1

Browse files
Rollup merge of #97219 - RalfJung:ptr-invalid, r=thomcc
make ptr::invalid not the same as a regular int2ptr cast In Miri, we would like to distinguish `ptr::invalid` from `ptr::from_exposed_provenance`, so that we can provide better diagnostics issues like rust-lang/miri#2134, and so that we can detect the UB in programs like ```rust fn main() { let x = 0u8; let original_ptr = &x as *const u8; let addr = original_ptr.expose_addr(); let new_ptr: *const u8 = core::ptr::invalid(addr); unsafe { dbg!(*new_ptr); } } ``` To achieve that, the two functions need to have different implementations. Currently, both are just `as` casts. We *could* add an intrinsic for this, but it turns out `transmute` already has the right behavior, at least as far as Miri is concerned. So I propose we just use that. Cc `@Gankra`
2 parents e5c7b21 + 31c3c04 commit 6fef5f1

File tree

1 file changed

+10
-2
lines changed

1 file changed

+10
-2
lines changed

library/core/src/ptr/mod.rs

+10-2
Original file line numberDiff line numberDiff line change
@@ -555,7 +555,11 @@ pub const fn null_mut<T>() -> *mut T {
555555
#[unstable(feature = "strict_provenance", issue = "95228")]
556556
pub const fn invalid<T>(addr: usize) -> *const T {
557557
// FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic.
558-
addr as *const T
558+
// We use transmute rather than a cast so tools like Miri can tell that this
559+
// is *not* the same as from_exposed_addr.
560+
// SAFETY: every valid integer is also a valid pointer (as long as you don't dereference that
561+
// pointer).
562+
unsafe { mem::transmute(addr) }
559563
}
560564

561565
/// Creates an invalid mutable pointer with the given address.
@@ -582,7 +586,11 @@ pub const fn invalid<T>(addr: usize) -> *const T {
582586
#[unstable(feature = "strict_provenance", issue = "95228")]
583587
pub const fn invalid_mut<T>(addr: usize) -> *mut T {
584588
// FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic.
585-
addr as *mut T
589+
// We use transmute rather than a cast so tools like Miri can tell that this
590+
// is *not* the same as from_exposed_addr.
591+
// SAFETY: every valid integer is also a valid pointer (as long as you don't dereference that
592+
// pointer).
593+
unsafe { mem::transmute(addr) }
586594
}
587595

588596
/// Convert an address back to a pointer, picking up a previously 'exposed' provenance.

0 commit comments

Comments
 (0)