Skip to content

Commit 79ae6ee

Browse files
authored
Rollup merge of rust-lang#71665 - RalfJung:miri-intern-no-ice, r=oli-obk
Miri interning: replace ICEs by proper errors Fixes rust-lang#71316 I also did some refactoring, as I kept being confused by all the parameters to `intern_shallow`, some of which have invalid combinations (such as a mutable const). So instead `InternMode` now contains all the information that is needed and invalid combinations are ruled out by the type system. Also I removed interpreter errors from interning. We already ignored almost all errors, and the `ValidationFailure` errors that we handled separately actually cannot ever happen here. The only interpreter failure that was actually reachable was the UB on dangling pointers -- and arguably, a dangling raw pointer is not UB, so the error was not even correct. It's just that the rest of the compiler does not like "dangling" `AllocId`. It should be possible to review the 3 commits separately. r? @oli-obk Cc @rust-lang/wg-const-eval
2 parents 41f701a + e73ee41 commit 79ae6ee

23 files changed

+331
-336
lines changed

src/librustc_mir/const_eval/eval_queries.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ fn eval_body_using_ecx<'mir, 'tcx>(
6666
intern_kind,
6767
ret,
6868
body.ignore_interior_mut_in_const_validation,
69-
)?;
69+
);
7070

7171
debug!("eval_body_using_ecx done: {:?}", *ret);
7272
Ok(ret)

src/librustc_mir/const_eval/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ pub(crate) fn const_caller_location(
5353
let mut ecx = mk_eval_cx(tcx, DUMMY_SP, ty::ParamEnv::reveal_all(), false);
5454

5555
let loc_place = ecx.alloc_caller_location(file, line, col);
56-
intern_const_alloc_recursive(&mut ecx, InternKind::Constant, loc_place, false).unwrap();
56+
intern_const_alloc_recursive(&mut ecx, InternKind::Constant, loc_place, false);
5757
ConstValue::Scalar(loc_place.ptr)
5858
}
5959

src/librustc_mir/interpret/eval_context.rs

+3
Original file line numberDiff line numberDiff line change
@@ -871,6 +871,9 @@ impl<'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> InterpCx<'mir, 'tcx, M> {
871871
// Our result will later be validated anyway, and there seems no good reason
872872
// to have to fail early here. This is also more consistent with
873873
// `Memory::get_static_alloc` which has to use `const_eval_raw` to avoid cycles.
874+
// FIXME: We can hit delay_span_bug if this is an invalid const, interning finds
875+
// that problem, but we never run validation to show an error. Can we ensure
876+
// this does not happen?
874877
let val = self.tcx.const_eval_raw(param_env.and(gid))?;
875878
self.raw_const_to_mplace(val)
876879
}

src/librustc_mir/interpret/intern.rs

+177-153
Large diffs are not rendered by default.

src/librustc_mir/transform/const_prop.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -702,8 +702,7 @@ impl<'mir, 'tcx> ConstPropagator<'mir, 'tcx> {
702702
)) => l.is_bits() && r.is_bits(),
703703
interpret::Operand::Indirect(_) if mir_opt_level >= 2 => {
704704
let mplace = op.assert_mem_place(&self.ecx);
705-
intern_const_alloc_recursive(&mut self.ecx, InternKind::ConstProp, mplace, false)
706-
.expect("failed to intern alloc");
705+
intern_const_alloc_recursive(&mut self.ecx, InternKind::ConstProp, mplace, false);
707706
true
708707
}
709708
_ => false,

src/test/ui/consts/dangling-alloc-id-ice.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,13 @@
11
// https://github.com/rust-lang/rust/issues/55223
2+
#![allow(const_err)]
23

34
union Foo<'a> {
45
y: &'a (),
56
long_live_the_unit: &'static (),
67
}
78

8-
const FOO: &() = { //~ ERROR any use of this value will cause an error
9+
const FOO: &() = { //~ ERROR it is undefined behavior to use this value
10+
//~^ ERROR encountered dangling pointer in final constant
911
let y = ();
1012
unsafe { Foo { y: &y }.long_live_the_unit }
1113
};
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,25 @@
1-
error: any use of this value will cause an error
2-
--> $DIR/dangling-alloc-id-ice.rs:8:1
1+
error: encountered dangling pointer in final constant
2+
--> $DIR/dangling-alloc-id-ice.rs:9:1
33
|
44
LL | / const FOO: &() = {
5+
LL | |
56
LL | | let y = ();
67
LL | | unsafe { Foo { y: &y }.long_live_the_unit }
78
LL | | };
8-
| |__^ encountered dangling pointer in final constant
9+
| |__^
10+
11+
error[E0080]: it is undefined behavior to use this value
12+
--> $DIR/dangling-alloc-id-ice.rs:9:1
13+
|
14+
LL | / const FOO: &() = {
15+
LL | |
16+
LL | | let y = ();
17+
LL | | unsafe { Foo { y: &y }.long_live_the_unit }
18+
LL | | };
19+
| |__^ type validation failed: encountered a dangling reference (use-after-free)
920
|
10-
= note: `#[deny(const_err)]` on by default
21+
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rustc repository if you believe it should not be considered undefined behavior.
1122

12-
error: aborting due to previous error
23+
error: aborting due to 2 previous errors
1324

25+
For more information about this error, try `rustc --explain E0080`.

src/test/ui/consts/dangling_raw_ptr.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
const FOO: *const u32 = { //~ ERROR any use of this value will cause an error
1+
const FOO: *const u32 = { //~ ERROR encountered dangling pointer in final constant
22
let x = 42;
33
&x
44
};
+2-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,11 @@
1-
error: any use of this value will cause an error
1+
error: encountered dangling pointer in final constant
22
--> $DIR/dangling_raw_ptr.rs:1:1
33
|
44
LL | / const FOO: *const u32 = {
55
LL | | let x = 42;
66
LL | | &x
77
LL | | };
8-
| |__^ encountered dangling pointer in final constant
9-
|
10-
= note: `#[deny(const_err)]` on by default
8+
| |__^
119

1210
error: aborting due to previous error
1311

src/test/ui/consts/miri_unleashed/mutable_const.rs

-25
This file was deleted.

src/test/ui/consts/miri_unleashed/mutable_const.stderr

-39
This file was deleted.

src/test/ui/consts/miri_unleashed/mutable_const2.rs

-16
This file was deleted.

src/test/ui/consts/miri_unleashed/mutable_const2.stderr

-29
This file was deleted.

src/test/ui/consts/miri_unleashed/mutable_references.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -17,12 +17,11 @@ struct Foo<T>(T);
1717
// this is fine for the same reason as `BAR`.
1818
static BOO: &mut Foo<()> = &mut Foo(());
1919

20+
// interior mutability is fine
2021
struct Meh {
2122
x: &'static UnsafeCell<i32>,
2223
}
23-
2424
unsafe impl Sync for Meh {}
25-
2625
static MEH: Meh = Meh {
2726
x: &UnsafeCell::new(42),
2827
};

src/test/ui/consts/miri_unleashed/mutable_references.stderr

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error[E0594]: cannot assign to `*OH_YES`, as `OH_YES` is an immutable static item
2-
--> $DIR/mutable_references.rs:37:5
2+
--> $DIR/mutable_references.rs:36:5
33
|
44
LL | *OH_YES = 99;
55
| ^^^^^^^^^^^^ cannot assign
@@ -22,12 +22,12 @@ help: skipping check for `const_mut_refs` feature
2222
LL | static BOO: &mut Foo<()> = &mut Foo(());
2323
| ^^^^^^^^^^^^
2424
help: skipping check that does not even have a feature gate
25-
--> $DIR/mutable_references.rs:27:8
25+
--> $DIR/mutable_references.rs:26:8
2626
|
2727
LL | x: &UnsafeCell::new(42),
2828
| ^^^^^^^^^^^^^^^^^^^^
2929
help: skipping check for `const_mut_refs` feature
30-
--> $DIR/mutable_references.rs:31:27
30+
--> $DIR/mutable_references.rs:30:27
3131
|
3232
LL | static OH_YES: &mut i32 = &mut 42;
3333
| ^^^^^^^
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
// compile-flags: -Zunleash-the-miri-inside-of-you
2+
3+
#![allow(const_err)]
4+
5+
use std::cell::UnsafeCell;
6+
7+
// this test ensures that our mutability story is sound
8+
9+
struct Meh {
10+
x: &'static UnsafeCell<i32>,
11+
}
12+
unsafe impl Sync for Meh {}
13+
14+
// the following will never be ok! no interior mut behind consts, because
15+
// all allocs interned here will be marked immutable.
16+
const MUH: Meh = Meh { //~ ERROR: mutable memory (`UnsafeCell`) is not allowed in constant
17+
x: &UnsafeCell::new(42),
18+
};
19+
20+
struct Synced {
21+
x: UnsafeCell<i32>,
22+
}
23+
unsafe impl Sync for Synced {}
24+
25+
// Make sure we also catch this behind a type-erased `dyn Trait` reference.
26+
const SNEAKY: &dyn Sync = &Synced { x: UnsafeCell::new(42) };
27+
//~^ ERROR: mutable memory (`UnsafeCell`) is not allowed in constant
28+
29+
// Make sure we also catch mutable references.
30+
const BLUNT: &mut i32 = &mut 42;
31+
//~^ ERROR: mutable memory (`&mut`) is not allowed in constant
32+
33+
fn main() {
34+
unsafe {
35+
*MUH.x.get() = 99;
36+
}
37+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
error: mutable memory (`UnsafeCell`) is not allowed in constant
2+
--> $DIR/mutable_references_err.rs:16:1
3+
|
4+
LL | / const MUH: Meh = Meh {
5+
LL | | x: &UnsafeCell::new(42),
6+
LL | | };
7+
| |__^
8+
9+
error: mutable memory (`UnsafeCell`) is not allowed in constant
10+
--> $DIR/mutable_references_err.rs:26:1
11+
|
12+
LL | const SNEAKY: &dyn Sync = &Synced { x: UnsafeCell::new(42) };
13+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
14+
15+
error: mutable memory (`&mut`) is not allowed in constant
16+
--> $DIR/mutable_references_err.rs:30:1
17+
|
18+
LL | const BLUNT: &mut i32 = &mut 42;
19+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
20+
21+
warning: skipping const checks
22+
|
23+
help: skipping check that does not even have a feature gate
24+
--> $DIR/mutable_references_err.rs:17:8
25+
|
26+
LL | x: &UnsafeCell::new(42),
27+
| ^^^^^^^^^^^^^^^^^^^^
28+
help: skipping check that does not even have a feature gate
29+
--> $DIR/mutable_references_err.rs:26:27
30+
|
31+
LL | const SNEAKY: &dyn Sync = &Synced { x: UnsafeCell::new(42) };
32+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
33+
help: skipping check for `const_mut_refs` feature
34+
--> $DIR/mutable_references_err.rs:30:25
35+
|
36+
LL | const BLUNT: &mut i32 = &mut 42;
37+
| ^^^^^^^
38+
39+
error: aborting due to 3 previous errors; 1 warning emitted
40+

src/test/ui/consts/miri_unleashed/mutable_references_ice.rs

-29
This file was deleted.

src/test/ui/consts/miri_unleashed/mutable_references_ice.stderr

-25
This file was deleted.

0 commit comments

Comments
 (0)