Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Check type of place base (not projection) for Freeze in IndirectlyMutableLocals #65030

Closed
Closed
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 7 additions & 5 deletions src/librustc_mir/dataflow/impls/indirect_mutation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -104,15 +104,17 @@ impl<'tcx> TransferFunction<'_, '_, 'tcx> {
kind: mir::BorrowKind,
borrowed_place: &mir::Place<'tcx>,
) -> bool {
let borrowed_ty = borrowed_place.ty(self.body, self.tcx).ty;
let base_ty = borrowed_place.base.ty(self.body).ty;

// Zero-sized types cannot be mutated, since there is nothing inside to mutate.
//
// FIXME: For now, we only exempt arrays of length zero. We need to carefully
// consider the effects before extending this to all ZSTs.
if let ty::Array(_, len) = borrowed_ty.kind {
if len.try_eval_usize(self.tcx, self.param_env) == Some(0) {
return false;
if borrowed_place.projection.is_empty() {
if let ty::Array(_, len) = base_ty.kind {
if len.try_eval_usize(self.tcx, self.param_env) == Some(0) {
return false;
}
}
}

Expand All @@ -122,7 +124,7 @@ impl<'tcx> TransferFunction<'_, '_, 'tcx> {
| mir::BorrowKind::Shared
| mir::BorrowKind::Shallow
| mir::BorrowKind::Unique
=> !borrowed_ty.is_freeze(self.tcx, self.param_env, DUMMY_SP),
=> !base_ty.is_freeze(self.tcx, self.param_env, DUMMY_SP),
}
}
}
Expand Down
2 changes: 1 addition & 1 deletion src/librustc_mir/transform/rustc_peek.rs
Original file line number Diff line number Diff line change
Expand Up @@ -276,7 +276,7 @@ impl<'tcx> RustcPeekAt<'tcx> for IndirectlyMutableLocals<'_, 'tcx> {
flow_state: &BitSet<Local>,
call: PeekCall,
) {
warn!("peek_at: place={:?}", place);
trace!("peek_at: place={:?}", place);
let local = match place {
mir::Place { base: mir::PlaceBase::Local(l), projection: box [] } => *l,
_ => {
Expand Down
33 changes: 33 additions & 0 deletions src/test/ui/mir-dataflow/indirect-mutation-1.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
#![feature(core_intrinsics, rustc_attrs, const_raw_ptr_deref)]
oli-obk marked this conversation as resolved.
Show resolved Hide resolved

use std::cell::Cell;
use std::intrinsics::rustc_peek;

#[rustc_mir(rustc_peek_indirectly_mutable, stop_after_dataflow)]
pub fn mut_ref(flag: bool) -> i32 {
let mut i = 0;
let cell = Cell::new(0);

if flag {
let p = &mut i;
unsafe { rustc_peek(i) };
*p = 1;

let p = &cell;
unsafe { rustc_peek(cell) };
p.set(2);
} else {
unsafe { rustc_peek(i) }; //~ ERROR rustc_peek: bit not set
unsafe { rustc_peek(cell) }; //~ ERROR rustc_peek: bit not set

let p = &mut cell;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

cell is immutable, how does this line not error?

Copy link
Contributor Author

@ecstatic-morse ecstatic-morse Oct 3, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

stop_after_dataflow prevents later all subsequent compiler passes from running, so we never actually get to the pass that would fail on this broken code. I'll make sure these tests compile outside of rustc_peek mode when possible though.

unsafe { rustc_peek(cell) };
*p = Cell::new(3);
}

unsafe { rustc_peek(i) };
unsafe { rustc_peek(cell) };
i + cell.get()
}

fn main() {}
16 changes: 16 additions & 0 deletions src/test/ui/mir-dataflow/indirect-mutation-1.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
error: rustc_peek: bit not set
--> $DIR/indirect-mutation-1.rs:20:18
|
LL | unsafe { rustc_peek(i) };
| ^^^^^^^^^^^^^

error: rustc_peek: bit not set
--> $DIR/indirect-mutation-1.rs:21:18
|
LL | unsafe { rustc_peek(cell) };
| ^^^^^^^^^^^^^^^^

error: stop_after_dataflow ended compilation

error: aborting due to 3 previous errors

30 changes: 30 additions & 0 deletions src/test/ui/mir-dataflow/indirect-mutation-2.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
#![feature(core_intrinsics, rustc_attrs, const_raw_ptr_deref)]

use std::cell::Cell;
use std::intrinsics::rustc_peek;

struct Arr {
inner: [Cell<i32>; 0],
_peek: (),
}

// Zero-sized arrays are never mutable.
#[rustc_mir(rustc_peek_indirectly_mutable, stop_after_dataflow)]
pub fn zst(flag: bool) {
{
let arr: [i32; 0] = [];
let s: &mut [i32] = &mut arr;
unsafe { rustc_peek(arr) }; //~ ERROR rustc_peek: bit not set
}

{
let arr: [Cell<i32>; 0] = [];
let s: &[Cell<i32>] = &arr;
unsafe { rustc_peek(arr) }; //~ ERROR rustc_peek: bit not set

let ss = &s;
unsafe { rustc_peek(arr) }; //~ ERROR rustc_peek: bit not set
}
}

fn main() {}
22 changes: 22 additions & 0 deletions src/test/ui/mir-dataflow/indirect-mutation-2.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
error: rustc_peek: bit not set
--> $DIR/indirect-mutation-2.rs:17:18
|
LL | unsafe { rustc_peek(arr) };
| ^^^^^^^^^^^^^^^

error: rustc_peek: bit not set
--> $DIR/indirect-mutation-2.rs:23:18
|
LL | unsafe { rustc_peek(arr) };
| ^^^^^^^^^^^^^^^

error: rustc_peek: bit not set
--> $DIR/indirect-mutation-2.rs:26:18
|
LL | unsafe { rustc_peek(arr) };
| ^^^^^^^^^^^^^^^

error: stop_after_dataflow ended compilation

error: aborting due to 4 previous errors

11 changes: 7 additions & 4 deletions src/test/ui/mir-dataflow/indirect-mutation-offset.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,14 +12,17 @@ struct PartialInteriorMut {
}

#[rustc_mir(rustc_peek_indirectly_mutable,stop_after_dataflow)]
#[rustc_mir(borrowck_graphviz_postflow="indirect.dot")]
const BOO: i32 = {
let x = PartialInteriorMut {
zst: [],
cell: UnsafeCell::new(0),
};

let p_zst: *const _ = &x.zst ; // Doesn't cause `x` to get marked as indirectly mutable.
unsafe { rustc_peek(x) }; //~ ERROR rustc_peek: bit not set

let p_zst: *const _ = &x.zst ;

unsafe { rustc_peek(x) };

let rmut_cell = unsafe {
// Take advantage of the fact that `zst` and `cell` are at the same location in memory.
Expand All @@ -30,9 +33,9 @@ const BOO: i32 = {
&mut *pmut_cell
};

*rmut_cell = 42; // Mutates `x` indirectly even though `x` is not marked indirectly mutable!!!
*rmut_cell = 42;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm a bit confused. This line is UB. You obtained a reference to cell from a reference to zst. I'm certain miri will slap this code around if it were runtime code

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm confused about rust's memory model then. If you're not allowed to obtain a reference to one part of a struct and convert it to a reference to a disjoint part of that same struct, then there's no need to consider this in the analysis.

Copy link
Contributor Author

@ecstatic-morse ecstatic-morse Oct 3, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The following quote is from the ptr::offset docs (non-normative, I know). I don't actually use ptr::offset, because it's not supported in MIRI. Instead I use a zero-sized type in a #[repr(C)] struct to ensure that the two fields have the same address.

Both the starting and resulting pointer must be either in bounds or one byte past the end of the same allocated object. Note that in Rust, every (stack-allocated) variable is considered a separate allocated object.

I was assuming that the stack-allocated instance of PartialInteriorMut was a single "allocated object" in rust's memory model, and thus converting a pointer to one of its fields into a pointer to another was not UB.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure at all. Maybe @RalfJung can have a look?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm confused about rust's memory model then. If you're not allowed to obtain a reference to one part of a struct and convert it to a reference to a disjoint part of that same struct, then there's no need to consider this in the analysis.

Stacked Borrows disallows this, to keep aliasing under control. But we don't have a final memory model yet. See this document for further details on Stacked Borrows.

Also, with Stacked Borrows you can do things like this through raw pointers. Like, &x as *mut T and then do whatever you want with the resulting raw pointer -- anything that C allows, we also allow.

let val = *rmut_cell;
unsafe { rustc_peek(x) }; //~ ERROR rustc_peek: bit not set
unsafe { rustc_peek(x) };

val
};
Expand Down
2 changes: 1 addition & 1 deletion src/test/ui/mir-dataflow/indirect-mutation-offset.stderr
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
error: rustc_peek: bit not set
--> $DIR/indirect-mutation-offset.rs:35:14
--> $DIR/indirect-mutation-offset.rs:21:14
|
LL | unsafe { rustc_peek(x) };
| ^^^^^^^^^^^^^
Expand Down