Skip to content

Commit c39ad27

Browse files
authored
Unrolled build for rust-lang#122249
Rollup merge of rust-lang#122249 - RalfJung:machine-read-hook, r=oli-obk interpret: do not call machine read hooks during validation Fixes rust-lang/miri#3347 r? ``@oli-obk``
2 parents 65cd843 + bf47df8 commit c39ad27

File tree

6 files changed

+107
-12
lines changed

6 files changed

+107
-12
lines changed

compiler/rustc_const_eval/src/const_eval/eval_queries.rs

+2-6
Original file line numberDiff line numberDiff line change
@@ -380,16 +380,12 @@ pub fn eval_in_interpreter<'mir, 'tcx>(
380380
}
381381
Ok(mplace) => {
382382
// Since evaluation had no errors, validate the resulting constant.
383-
384-
// Temporarily allow access to the static_root_alloc_id for the purpose of validation.
385-
let static_root_alloc_id = ecx.machine.static_root_alloc_id.take();
386-
let validation = const_validate_mplace(&ecx, &mplace, cid);
387-
ecx.machine.static_root_alloc_id = static_root_alloc_id;
383+
let res = const_validate_mplace(&ecx, &mplace, cid);
388384

389385
let alloc_id = mplace.ptr().provenance.unwrap().alloc_id();
390386

391387
// Validation failed, report an error.
392-
if let Err(error) = validation {
388+
if let Err(error) = res {
393389
Err(const_report_error(&ecx, error, alloc_id))
394390
} else {
395391
// Convert to raw constant

compiler/rustc_const_eval/src/interpret/machine.rs

+4
Original file line numberDiff line numberDiff line change
@@ -391,6 +391,8 @@ pub trait Machine<'mir, 'tcx: 'mir>: Sized {
391391

392392
/// Hook for performing extra checks on a memory read access.
393393
///
394+
/// This will *not* be called during validation!
395+
///
394396
/// Takes read-only access to the allocation so we can keep all the memory read
395397
/// operations take `&self`. Use a `RefCell` in `AllocExtra` if you
396398
/// need to mutate.
@@ -410,6 +412,8 @@ pub trait Machine<'mir, 'tcx: 'mir>: Sized {
410412
/// Hook for performing extra checks on any memory read access,
411413
/// that involves an allocation, even ZST reads.
412414
///
415+
/// This will *not* be called during validation!
416+
///
413417
/// Used to prevent statics from self-initializing by reading from their own memory
414418
/// as it is being initialized.
415419
fn before_alloc_read(

compiler/rustc_const_eval/src/interpret/memory.rs

+38-5
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
99
use std::assert_matches::assert_matches;
1010
use std::borrow::Cow;
11+
use std::cell::Cell;
1112
use std::collections::VecDeque;
1213
use std::fmt;
1314
use std::ptr;
@@ -111,6 +112,11 @@ pub struct Memory<'mir, 'tcx, M: Machine<'mir, 'tcx>> {
111112
/// that do not exist any more.
112113
// FIXME: this should not be public, but interning currently needs access to it
113114
pub(super) dead_alloc_map: FxIndexMap<AllocId, (Size, Align)>,
115+
116+
/// This stores whether we are currently doing reads purely for the purpose of validation.
117+
/// Those reads do not trigger the machine's hooks for memory reads.
118+
/// Needless to say, this must only be set with great care!
119+
validation_in_progress: Cell<bool>,
114120
}
115121

116122
/// A reference to some allocation that was already bounds-checked for the given region
@@ -137,6 +143,7 @@ impl<'mir, 'tcx, M: Machine<'mir, 'tcx>> Memory<'mir, 'tcx, M> {
137143
alloc_map: M::MemoryMap::default(),
138144
extra_fn_ptr_map: FxIndexMap::default(),
139145
dead_alloc_map: FxIndexMap::default(),
146+
validation_in_progress: Cell::new(false),
140147
}
141148
}
142149

@@ -624,18 +631,28 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
624631
size,
625632
CheckInAllocMsg::MemoryAccessTest,
626633
|alloc_id, offset, prov| {
627-
// We want to call the hook on *all* accesses that involve an AllocId,
628-
// including zero-sized accesses. That means we have to do it here
629-
// rather than below in the `Some` branch.
630-
M::before_alloc_read(self, alloc_id)?;
634+
if !self.memory.validation_in_progress.get() {
635+
// We want to call the hook on *all* accesses that involve an AllocId,
636+
// including zero-sized accesses. That means we have to do it here
637+
// rather than below in the `Some` branch.
638+
M::before_alloc_read(self, alloc_id)?;
639+
}
631640
let alloc = self.get_alloc_raw(alloc_id)?;
632641
Ok((alloc.size(), alloc.align, (alloc_id, offset, prov, alloc)))
633642
},
634643
)?;
635644

636645
if let Some((alloc_id, offset, prov, alloc)) = ptr_and_alloc {
637646
let range = alloc_range(offset, size);
638-
M::before_memory_read(self.tcx, &self.machine, &alloc.extra, (alloc_id, prov), range)?;
647+
if !self.memory.validation_in_progress.get() {
648+
M::before_memory_read(
649+
self.tcx,
650+
&self.machine,
651+
&alloc.extra,
652+
(alloc_id, prov),
653+
range,
654+
)?;
655+
}
639656
Ok(Some(AllocRef { alloc, range, tcx: *self.tcx, alloc_id }))
640657
} else {
641658
Ok(None)
@@ -909,6 +926,21 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
909926
}
910927
})
911928
}
929+
930+
/// Runs the close in "validation" mode, which means the machine's memory read hooks will be
931+
/// suppressed. Needless to say, this must only be set with great care! Cannot be nested.
932+
pub(super) fn run_for_validation<R>(&self, f: impl FnOnce() -> R) -> R {
933+
assert!(
934+
self.memory.validation_in_progress.replace(true) == false,
935+
"`validation_in_progress` was already set"
936+
);
937+
let res = f();
938+
assert!(
939+
self.memory.validation_in_progress.replace(false) == true,
940+
"`validation_in_progress` was unset by someone else"
941+
);
942+
res
943+
}
912944
}
913945

914946
#[doc(hidden)]
@@ -1154,6 +1186,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
11541186
};
11551187
let src_alloc = self.get_alloc_raw(src_alloc_id)?;
11561188
let src_range = alloc_range(src_offset, size);
1189+
assert!(!self.memory.validation_in_progress.get(), "we can't be copying during validation");
11571190
M::before_memory_read(
11581191
tcx,
11591192
&self.machine,

compiler/rustc_const_eval/src/interpret/validity.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -971,7 +971,7 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
971971
let mut visitor = ValidityVisitor { path, ref_tracking, ctfe_mode, ecx: self };
972972

973973
// Run it.
974-
match visitor.visit_value(op) {
974+
match self.run_for_validation(|| visitor.visit_value(op)) {
975975
Ok(()) => Ok(()),
976976
// Pass through validation failures and "invalid program" issues.
977977
Err(err)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
#![feature(start)]
2+
#![no_std]
3+
//@compile-flags: -Zmiri-track-alloc-id=17 -Zmiri-track-alloc-accesses -Cpanic=abort
4+
//@only-target-linux: alloc IDs differ between OSes for some reason
5+
6+
extern "Rust" {
7+
fn miri_alloc(size: usize, align: usize) -> *mut u8;
8+
fn miri_dealloc(ptr: *mut u8, size: usize, align: usize);
9+
}
10+
11+
#[start]
12+
fn start(_: isize, _: *const *const u8) -> isize {
13+
unsafe {
14+
let ptr = miri_alloc(123, 1);
15+
*ptr = 42; // Crucially, only a write is printed here, no read!
16+
assert_eq!(*ptr, 42);
17+
miri_dealloc(ptr, 123, 1);
18+
}
19+
0
20+
}
21+
22+
#[panic_handler]
23+
fn panic_handler(_: &core::panic::PanicInfo) -> ! {
24+
loop {}
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
note: tracking was triggered
2+
--> $DIR/alloc-access-tracking.rs:LL:CC
3+
|
4+
LL | let ptr = miri_alloc(123, 1);
5+
| ^^^^^^^^^^^^^^^^^^ created Miri bare-metal heap allocation of 123 bytes (alignment ALIGN bytes) with id 17
6+
|
7+
= note: BACKTRACE:
8+
= note: inside `start` at $DIR/alloc-access-tracking.rs:LL:CC
9+
10+
note: tracking was triggered
11+
--> $DIR/alloc-access-tracking.rs:LL:CC
12+
|
13+
LL | *ptr = 42; // Crucially, only a write is printed here, no read!
14+
| ^^^^^^^^^ write access to allocation with id 17
15+
|
16+
= note: BACKTRACE:
17+
= note: inside `start` at $DIR/alloc-access-tracking.rs:LL:CC
18+
19+
note: tracking was triggered
20+
--> $DIR/alloc-access-tracking.rs:LL:CC
21+
|
22+
LL | assert_eq!(*ptr, 42);
23+
| ^^^^^^^^^^^^^^^^^^^^ read access to allocation with id 17
24+
|
25+
= note: BACKTRACE:
26+
= note: inside `start` at RUSTLIB/core/src/macros/mod.rs:LL:CC
27+
= note: this note originates in the macro `assert_eq` (in Nightly builds, run with -Z macro-backtrace for more info)
28+
29+
note: tracking was triggered
30+
--> $DIR/alloc-access-tracking.rs:LL:CC
31+
|
32+
LL | miri_dealloc(ptr, 123, 1);
33+
| ^^^^^^^^^^^^^^^^^^^^^^^^^ freed allocation with id 17
34+
|
35+
= note: BACKTRACE:
36+
= note: inside `start` at $DIR/alloc-access-tracking.rs:LL:CC
37+

0 commit comments

Comments
 (0)