Skip to content

Commit bd7aded

Browse files
committed
Auto merge of rust-lang#121242 - joboet:ptr_invalid_codegen, r=scottmcm
Generate `getelementptr` instead of `inttoptr` for `ptr::invalid` Currently, `ptr::invalid` [generates an `inttoptr`](https://godbolt.org/z/3cj15dEG1), which means LLVM doesn't know that the pointer shouldn't have provenance. This PR changes the implementation so that a `getelementptr` relative to the null pointer is generated, which LLVM knows not to have provenance.
2 parents 6cbf092 + b6d7805 commit bd7aded

File tree

2 files changed

+32
-10
lines changed

2 files changed

+32
-10
lines changed

library/core/src/ptr/mod.rs

+16-10
Original file line numberDiff line numberDiff line change
@@ -590,11 +590,14 @@ pub const fn null_mut<T: ?Sized + Thin>() -> *mut T {
590590
#[unstable(feature = "strict_provenance", issue = "95228")]
591591
pub const fn without_provenance<T>(addr: usize) -> *const T {
592592
// FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic.
593-
// We use transmute rather than a cast so tools like Miri can tell that this
594-
// is *not* the same as from_exposed_addr.
595-
// SAFETY: every valid integer is also a valid pointer (as long as you don't dereference that
596-
// pointer).
597-
unsafe { mem::transmute(addr) }
593+
// FIXME(maybe): simply transmute the address once codegen lowers that correctly.
594+
// We offset the null pointer instead of using a cast so that this lowers to
595+
// getelementptr instead of inttoptr.
596+
// We use transmute instead of casting so that in miri the pointer truly
597+
// has no provenance.
598+
// SAFETY: on all current platforms, usize and pointers have the same layout,
599+
// and the validity invariant of pointers is the same as that of integers
600+
unsafe { mem::transmute::<_, *const T>(0usize).wrapping_byte_add(addr) }
598601
}
599602

600603
/// Creates a new pointer that is dangling, but well-aligned.
@@ -632,11 +635,14 @@ pub const fn dangling<T>() -> *const T {
632635
#[unstable(feature = "strict_provenance", issue = "95228")]
633636
pub const fn without_provenance_mut<T>(addr: usize) -> *mut T {
634637
// FIXME(strict_provenance_magic): I am magic and should be a compiler intrinsic.
635-
// We use transmute rather than a cast so tools like Miri can tell that this
636-
// is *not* the same as from_exposed_addr.
637-
// SAFETY: every valid integer is also a valid pointer (as long as you don't dereference that
638-
// pointer).
639-
unsafe { mem::transmute(addr) }
638+
// FIXME(maybe): simply transmute the address once codegen lowers that correctly.
639+
// We offset the null pointer instead of using a cast so that this lowers to
640+
// getelementptr instead of inttoptr.
641+
// We use transmute instead of casting so that in miri the pointer truly
642+
// has no provenance.
643+
// SAFETY: on all current platforms, usize and pointers have the same layout,
644+
// and the validity invariant of pointers is the same as that of integers
645+
unsafe { mem::transmute::<_, *mut T>(0usize).wrapping_byte_add(addr) }
640646
}
641647

642648
/// Creates a new pointer that is dangling, but well-aligned.

tests/codegen/strict-provenance.rs

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
//@ compile-flags: -O
2+
3+
#![crate_type = "lib"]
4+
#![feature(strict_provenance)]
5+
6+
use std::ptr;
7+
8+
// CHECK-LABEL: ptr @without_provenance(
9+
// CHECK-SAME: [[USIZE:i[0-9]+]] noundef %addr)
10+
#[no_mangle]
11+
fn without_provenance(addr: usize) -> *const () {
12+
// CHECK: start
13+
// CHECK-NEXT: %0 = getelementptr i8, ptr null, [[USIZE]] %addr
14+
// CHECK-NEXT: ret ptr %0
15+
ptr::without_provenance(addr)
16+
}

0 commit comments

Comments
 (0)