Skip to content

Commit 848090d

Browse files
committed
Auto merge of #98017 - RalfJung:dereferenceable, r=nikic
do not mark interior mutable shared refs as dereferenceable My proposed solution to #55005.
2 parents ffa7733 + 35c6dec commit 848090d

File tree

4 files changed

+57
-21
lines changed

4 files changed

+57
-21
lines changed

compiler/rustc_middle/src/ty/layout.rs

+12-7
Original file line numberDiff line numberDiff line change
@@ -2618,14 +2618,14 @@ where
26182618
// Use conservative pointer kind if not optimizing. This saves us the
26192619
// Freeze/Unpin queries, and can save time in the codegen backend (noalias
26202620
// attributes in LLVM have compile-time cost even in unoptimized builds).
2621-
PointerKind::Shared
2621+
PointerKind::SharedMutable
26222622
} else {
26232623
match mt {
26242624
hir::Mutability::Not => {
26252625
if ty.is_freeze(tcx.at(DUMMY_SP), cx.param_env()) {
26262626
PointerKind::Frozen
26272627
} else {
2628-
PointerKind::Shared
2628+
PointerKind::SharedMutable
26292629
}
26302630
}
26312631
hir::Mutability::Mut => {
@@ -2636,7 +2636,7 @@ where
26362636
if ty.is_unpin(tcx.at(DUMMY_SP), cx.param_env()) {
26372637
PointerKind::UniqueBorrowed
26382638
} else {
2639-
PointerKind::Shared
2639+
PointerKind::UniqueBorrowedPinned
26402640
}
26412641
}
26422642
}
@@ -3255,10 +3255,13 @@ impl<'tcx> LayoutCx<'tcx, TyCtxt<'tcx>> {
32553255

32563256
// `Box` (`UniqueBorrowed`) are not necessarily dereferenceable
32573257
// for the entire duration of the function as they can be deallocated
3258-
// at any time. Set their valid size to 0.
3258+
// at any time. Same for shared mutable references. If LLVM had a
3259+
// way to say "dereferenceable on entry" we could use it here.
32593260
attrs.pointee_size = match kind {
3260-
PointerKind::UniqueOwned => Size::ZERO,
3261-
_ => pointee.size,
3261+
PointerKind::UniqueBorrowed
3262+
| PointerKind::UniqueBorrowedPinned
3263+
| PointerKind::Frozen => pointee.size,
3264+
PointerKind::SharedMutable | PointerKind::UniqueOwned => Size::ZERO,
32623265
};
32633266

32643267
// `Box`, `&T`, and `&mut T` cannot be undef.
@@ -3285,7 +3288,9 @@ impl<'tcx> LayoutCx<'tcx, TyCtxt<'tcx>> {
32853288
// or not to actually emit the attribute. It can also be controlled with the
32863289
// `-Zmutable-noalias` debugging option.
32873290
let no_alias = match kind {
3288-
PointerKind::Shared | PointerKind::UniqueBorrowed => false,
3291+
PointerKind::SharedMutable
3292+
| PointerKind::UniqueBorrowed
3293+
| PointerKind::UniqueBorrowedPinned => false,
32893294
PointerKind::UniqueOwned => noalias_for_box,
32903295
PointerKind::Frozen => !is_return,
32913296
};

compiler/rustc_target/src/abi/mod.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -1350,15 +1350,19 @@ impl<'a, Ty> Deref for TyAndLayout<'a, Ty> {
13501350
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
13511351
pub enum PointerKind {
13521352
/// Most general case, we know no restrictions to tell LLVM.
1353-
Shared,
1353+
SharedMutable,
13541354

1355-
/// `&T` where `T` contains no `UnsafeCell`, is `noalias` and `readonly`.
1355+
/// `&T` where `T` contains no `UnsafeCell`, is `dereferenceable`, `noalias` and `readonly`.
13561356
Frozen,
13571357

1358-
/// `&mut T` which is `noalias` but not `readonly`.
1358+
/// `&mut T` which is `dereferenceable` and `noalias` but not `readonly`.
13591359
UniqueBorrowed,
13601360

1361-
/// `Box<T>`, unlike `UniqueBorrowed`, it also has `noalias` on returns.
1361+
/// `&mut !Unpin`, which is `dereferenceable` but neither `noalias` nor `readonly`.
1362+
UniqueBorrowedPinned,
1363+
1364+
/// `Box<T>`, which is `noalias` (even on return types, unlike the above) but neither `readonly`
1365+
/// nor `dereferenceable`.
13621366
UniqueOwned,
13631367
}
13641368

library/core/src/cell.rs

+18-9
Original file line numberDiff line numberDiff line change
@@ -1766,15 +1766,24 @@ impl<T: ?Sized + fmt::Display> fmt::Display for RefMut<'_, T> {
17661766
///
17671767
/// The precise Rust aliasing rules are somewhat in flux, but the main points are not contentious:
17681768
///
1769-
/// - If you create a safe reference with lifetime `'a` (either a `&T` or `&mut T`
1770-
/// reference) that is accessible by safe code (for example, because you returned it),
1771-
/// then you must not access the data in any way that contradicts that reference for the
1772-
/// remainder of `'a`. For example, this means that if you take the `*mut T` from an
1773-
/// `UnsafeCell<T>` and cast it to an `&T`, then the data in `T` must remain immutable
1774-
/// (modulo any `UnsafeCell` data found within `T`, of course) until that reference's
1775-
/// lifetime expires. Similarly, if you create a `&mut T` reference that is released to
1776-
/// safe code, then you must not access the data within the `UnsafeCell` until that
1777-
/// reference expires.
1769+
/// - If you create a safe reference with lifetime `'a` (either a `&T` or `&mut T` reference), then
1770+
/// you must not access the data in any way that contradicts that reference for the remainder of
1771+
/// `'a`. For example, this means that if you take the `*mut T` from an `UnsafeCell<T>` and cast it
1772+
/// to an `&T`, then the data in `T` must remain immutable (modulo any `UnsafeCell` data found
1773+
/// within `T`, of course) until that reference's lifetime expires. Similarly, if you create a `&mut
1774+
/// T` reference that is released to safe code, then you must not access the data within the
1775+
/// `UnsafeCell` until that reference expires.
1776+
///
1777+
/// - For both `&T` without `UnsafeCell<_>` and `&mut T`, you must also not deallocate the data
1778+
/// until the reference expires. As a special exception, given an `&T`, any part of it that is
1779+
/// inside an `UnsafeCell<_>` may be deallocated during the lifetime of the reference, after the
1780+
/// last time the reference is used (dereferenced or reborrowed). Since you cannot deallocate a part
1781+
/// of what a reference points to, this means the memory an `&T` points to can be deallocted only if
1782+
/// *every part of it* (including padding) is inside an `UnsafeCell`.
1783+
///
1784+
/// However, whenever a `&UnsafeCell<T>` is constructed or dereferenced, it must still point to
1785+
/// live memory and the compiler is allowed to insert spurious reads if it can prove that this
1786+
/// memory has not yet been deallocated.
17781787
///
17791788
/// - At all times, you must avoid data races. If multiple threads have access to
17801789
/// the same `UnsafeCell`, then any writes must have a proper happens-before relation to all other

src/test/codegen/function-arguments.rs

+19-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55

66
use std::mem::MaybeUninit;
77
use std::num::NonZeroU64;
8+
use std::marker::PhantomPinned;
89

910
pub struct S {
1011
_field: [i32; 8],
@@ -14,6 +15,11 @@ pub struct UnsafeInner {
1415
_field: std::cell::UnsafeCell<i16>,
1516
}
1617

18+
pub struct NotUnpin {
19+
_field: i32,
20+
_marker: PhantomPinned,
21+
}
22+
1723
pub enum MyBool {
1824
True,
1925
False,
@@ -91,7 +97,7 @@ pub fn static_borrow(_: &'static i32) {
9197
pub fn named_borrow<'r>(_: &'r i32) {
9298
}
9399

94-
// CHECK: @unsafe_borrow({{i16\*|ptr}} noundef align 2 dereferenceable(2) %_1)
100+
// CHECK: @unsafe_borrow({{i16\*|ptr}} noundef nonnull align 2 %_1)
95101
// unsafe interior means this isn't actually readonly and there may be aliases ...
96102
#[no_mangle]
97103
pub fn unsafe_borrow(_: &UnsafeInner) {
@@ -109,6 +115,18 @@ pub fn mutable_unsafe_borrow(_: &mut UnsafeInner) {
109115
pub fn mutable_borrow(_: &mut i32) {
110116
}
111117

118+
#[no_mangle]
119+
// CHECK: @mutable_notunpin_borrow({{i32\*|ptr}} noundef align 4 dereferenceable(4) %_1)
120+
// This one is *not* `noalias` because it might be self-referential.
121+
pub fn mutable_notunpin_borrow(_: &mut NotUnpin) {
122+
}
123+
124+
// CHECK: @notunpin_borrow({{i32\*|ptr}} noalias noundef readonly align 4 dereferenceable(4) %_1)
125+
// But `&NotUnpin` behaves perfectly normal.
126+
#[no_mangle]
127+
pub fn notunpin_borrow(_: &NotUnpin) {
128+
}
129+
112130
// CHECK: @indirect_struct({{%S\*|ptr}} noalias nocapture noundef dereferenceable(32) %_1)
113131
#[no_mangle]
114132
pub fn indirect_struct(_: S) {

0 commit comments

Comments
 (0)