Skip to content

Commit 01ee75d

Browse files
Do not optimize out SwitchInt before borrowck, or if Zmir-preserve-ub
1 parent 19cab6b commit 01ee75d

16 files changed

+192
-51
lines changed

compiler/rustc_interface/src/tests.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -816,8 +816,8 @@ fn test_unstable_options_tracking_hash() {
816816
tracked!(min_function_alignment, Some(Align::EIGHT));
817817
tracked!(mir_emit_retag, true);
818818
tracked!(mir_enable_passes, vec![("DestProp".to_string(), false)]);
819-
tracked!(mir_keep_place_mention, true);
820819
tracked!(mir_opt_level, Some(4));
820+
tracked!(mir_preserve_ub, true);
821821
tracked!(move_size_limit, Some(4096));
822822
tracked!(mutable_noalias, false);
823823
tracked!(next_solver, NextSolverConfig { coherence: true, globally: true });

compiler/rustc_mir_transform/src/early_otherwise_branch.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -224,7 +224,7 @@ impl<'tcx> crate::MirPass<'tcx> for EarlyOtherwiseBranch {
224224
// Since this optimization adds new basic blocks and invalidates others,
225225
// clean up the cfg to make it nicer for other passes
226226
if should_cleanup {
227-
simplify_cfg(body);
227+
simplify_cfg(tcx, body);
228228
}
229229
}
230230

compiler/rustc_mir_transform/src/inline.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ impl<'tcx> crate::MirPass<'tcx> for Inline {
6363
let _guard = span.enter();
6464
if inline::<NormalInliner<'tcx>>(tcx, body) {
6565
debug!("running simplify cfg on {:?}", body.source);
66-
simplify_cfg(body);
66+
simplify_cfg(tcx, body);
6767
deref_finder(tcx, body);
6868
}
6969
}
@@ -99,7 +99,7 @@ impl<'tcx> crate::MirPass<'tcx> for ForceInline {
9999
let _guard = span.enter();
100100
if inline::<ForceInliner<'tcx>>(tcx, body) {
101101
debug!("running simplify cfg on {:?}", body.source);
102-
simplify_cfg(body);
102+
simplify_cfg(tcx, body);
103103
deref_finder(tcx, body);
104104
}
105105
}

compiler/rustc_mir_transform/src/match_branches.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ impl<'tcx> crate::MirPass<'tcx> for MatchBranchSimplification {
4545
}
4646

4747
if should_cleanup {
48-
simplify_cfg(body);
48+
simplify_cfg(tcx, body);
4949
}
5050
}
5151

compiler/rustc_mir_transform/src/remove_place_mention.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ pub(super) struct RemovePlaceMention;
88

99
impl<'tcx> crate::MirPass<'tcx> for RemovePlaceMention {
1010
fn is_enabled(&self, sess: &rustc_session::Session) -> bool {
11-
!sess.opts.unstable_opts.mir_keep_place_mention
11+
!sess.opts.unstable_opts.mir_preserve_ub
1212
}
1313

1414
fn run_pass(&self, _: TyCtxt<'tcx>, body: &mut Body<'tcx>) {

compiler/rustc_mir_transform/src/remove_unneeded_drops.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ impl<'tcx> crate::MirPass<'tcx> for RemoveUnneededDrops {
3535
// if we applied optimizations, we potentially have some cfg to cleanup to
3636
// make it easier for further passes
3737
if should_simplify {
38-
simplify_cfg(body);
38+
simplify_cfg(tcx, body);
3939
}
4040
}
4141

compiler/rustc_mir_transform/src/simplify.rs

+20-9
Original file line numberDiff line numberDiff line change
@@ -66,8 +66,8 @@ impl SimplifyCfg {
6666
}
6767
}
6868

69-
pub(super) fn simplify_cfg(body: &mut Body<'_>) {
70-
CfgSimplifier::new(body).simplify();
69+
pub(super) fn simplify_cfg<'tcx>(tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) {
70+
CfgSimplifier::new(tcx, body).simplify();
7171
remove_dead_blocks(body);
7272

7373
// FIXME: Should probably be moved into some kind of pass manager
@@ -79,9 +79,9 @@ impl<'tcx> crate::MirPass<'tcx> for SimplifyCfg {
7979
self.name()
8080
}
8181

82-
fn run_pass(&self, _: TyCtxt<'tcx>, body: &mut Body<'tcx>) {
82+
fn run_pass(&self, tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) {
8383
debug!("SimplifyCfg({:?}) - simplifying {:?}", self.name(), body.source);
84-
simplify_cfg(body);
84+
simplify_cfg(tcx, body);
8585
}
8686

8787
fn is_required(&self) -> bool {
@@ -90,12 +90,13 @@ impl<'tcx> crate::MirPass<'tcx> for SimplifyCfg {
9090
}
9191

9292
struct CfgSimplifier<'a, 'tcx> {
93+
preserve_switch_reads: bool,
9394
basic_blocks: &'a mut IndexSlice<BasicBlock, BasicBlockData<'tcx>>,
9495
pred_count: IndexVec<BasicBlock, u32>,
9596
}
9697

9798
impl<'a, 'tcx> CfgSimplifier<'a, 'tcx> {
98-
fn new(body: &'a mut Body<'tcx>) -> Self {
99+
fn new(tcx: TyCtxt<'tcx>, body: &'a mut Body<'tcx>) -> Self {
99100
let mut pred_count = IndexVec::from_elem(0u32, &body.basic_blocks);
100101

101102
// we can't use mir.predecessors() here because that counts
@@ -110,9 +111,13 @@ impl<'a, 'tcx> CfgSimplifier<'a, 'tcx> {
110111
}
111112
}
112113

114+
// Preserve `SwitchInt` reads on the phase transition from Built -> Analysis(Initial)
115+
// or if `-Zmir-preserve-ub`.
116+
let preserve_switch_reads =
117+
matches!(body.phase, MirPhase::Built) | tcx.sess.opts.unstable_opts.mir_preserve_ub;
113118
let basic_blocks = body.basic_blocks_mut();
114119

115-
CfgSimplifier { basic_blocks, pred_count }
120+
CfgSimplifier { preserve_switch_reads, basic_blocks, pred_count }
116121
}
117122

118123
fn simplify(mut self) {
@@ -253,9 +258,15 @@ impl<'a, 'tcx> CfgSimplifier<'a, 'tcx> {
253258

254259
// turn a branch with all successors identical to a goto
255260
fn simplify_branch(&mut self, terminator: &mut Terminator<'tcx>) -> bool {
256-
match terminator.kind {
257-
TerminatorKind::SwitchInt { .. } => {}
258-
_ => return false,
261+
// Removing a `SwitchInt` terminator may remove reads that result in UB,
262+
// so we must not apply this optimization before borrowck or when
263+
// `-Zmir-preserve-ub` is set.
264+
if self.preserve_switch_reads {
265+
return false;
266+
}
267+
268+
let TerminatorKind::SwitchInt { .. } = terminator.kind else {
269+
return false;
259270
};
260271

261272
let first_succ = {

compiler/rustc_session/src/options.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -2313,12 +2313,12 @@ options! {
23132313
mir_include_spans: MirIncludeSpans = (MirIncludeSpans::default(), parse_mir_include_spans, [UNTRACKED],
23142314
"include extra comments in mir pretty printing, like line numbers and statement indices, \
23152315
details about types, etc. (boolean for all passes, 'nll' to enable in NLL MIR only, default: 'nll')"),
2316-
mir_keep_place_mention: bool = (false, parse_bool, [TRACKED],
2317-
"keep place mention MIR statements, interpreted e.g., by miri; implies -Zmir-opt-level=0 \
2318-
(default: no)"),
23192316
#[rustc_lint_opt_deny_field_access("use `Session::mir_opt_level` instead of this field")]
23202317
mir_opt_level: Option<usize> = (None, parse_opt_number, [TRACKED],
23212318
"MIR optimization level (0-4; default: 1 in non optimized builds and 2 in optimized builds)"),
2319+
mir_preserve_ub: bool = (false, parse_bool, [TRACKED],
2320+
"keep place mention statements and reads in trivial SwitchInt terminators, which are interpreted \
2321+
e.g., by miri; implies -Zmir-opt-level=0 (default: no)"),
23222322
mir_strip_debuginfo: MirStripDebugInfo = (MirStripDebugInfo::None, parse_mir_strip_debuginfo, [TRACKED],
23232323
"Whether to remove some of the MIR debug info from methods. Default: None"),
23242324
move_size_limit: Option<usize> = (None, parse_opt_number, [TRACKED],

src/tools/miri/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -169,7 +169,7 @@ pub const MIRI_DEFAULT_ARGS: &[&str] = &[
169169
"-Zalways-encode-mir",
170170
"-Zextra-const-ub-checks",
171171
"-Zmir-emit-retag",
172-
"-Zmir-keep-place-mention",
172+
"-Zmir-preserve-ub",
173173
"-Zmir-opt-level=0",
174174
"-Zmir-enable-passes=-CheckAlignment,-CheckNull",
175175
// Deduplicating diagnostics means we miss events when tracking what happens during an
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
use std::mem::MaybeUninit;
2+
3+
fn main() {
4+
let uninit: MaybeUninit<i32> = MaybeUninit::uninit();
5+
let bad_ref: &i32 = unsafe { uninit.assume_init_ref() };
6+
let &(0 | _) = bad_ref;
7+
//~^ ERROR: Undefined Behavior: using uninitialized data, but this operation requires initialized memory
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
error: Undefined Behavior: using uninitialized data, but this operation requires initialized memory
2+
--> tests/fail/read_from_trivial_switch.rs:LL:CC
3+
|
4+
LL | let &(0 | _) = bad_ref;
5+
| ^^^^^^^^ using uninitialized data, but this operation requires initialized memory
6+
|
7+
= help: this indicates a bug in the program: it performed an invalid operation, and caused Undefined Behavior
8+
= help: see https://doc.rust-lang.org/nightly/reference/behavior-considered-undefined.html for further information
9+
= note: BACKTRACE:
10+
= note: inside `main` at tests/fail/read_from_trivial_switch.rs:LL:CC
11+
12+
note: some details are omitted, run with `MIRIFLAGS=-Zmiri-backtrace=full` for a verbose backtrace
13+
14+
error: aborting due to 1 previous error
15+

tests/mir-opt/building/match/exponential_or.match_tuple.SimplifyCfg-initial.after.mir

+19-15
Original file line numberDiff line numberDiff line change
@@ -24,43 +24,47 @@ fn match_tuple(_1: (u32, bool, Option<i32>, u32)) -> u32 {
2424

2525
bb1: {
2626
_0 = const 0_u32;
27-
goto -> bb10;
27+
goto -> bb11;
2828
}
2929

3030
bb2: {
31-
_2 = discriminant((_1.2: std::option::Option<i32>));
32-
switchInt(move _2) -> [0: bb4, 1: bb3, otherwise: bb1];
31+
switchInt(copy (_1.1: bool)) -> [0: bb3, otherwise: bb3];
3332
}
3433

3534
bb3: {
36-
switchInt(copy (((_1.2: std::option::Option<i32>) as Some).0: i32)) -> [1: bb4, 8: bb4, otherwise: bb1];
35+
_2 = discriminant((_1.2: std::option::Option<i32>));
36+
switchInt(move _2) -> [0: bb5, 1: bb4, otherwise: bb1];
3737
}
3838

3939
bb4: {
40-
_5 = Le(const 6_u32, copy (_1.3: u32));
41-
switchInt(move _5) -> [0: bb5, otherwise: bb7];
40+
switchInt(copy (((_1.2: std::option::Option<i32>) as Some).0: i32)) -> [1: bb5, 8: bb5, otherwise: bb1];
4241
}
4342

4443
bb5: {
45-
_3 = Le(const 13_u32, copy (_1.3: u32));
46-
switchInt(move _3) -> [0: bb1, otherwise: bb6];
44+
_5 = Le(const 6_u32, copy (_1.3: u32));
45+
switchInt(move _5) -> [0: bb6, otherwise: bb8];
4746
}
4847

4948
bb6: {
50-
_4 = Le(copy (_1.3: u32), const 16_u32);
51-
switchInt(move _4) -> [0: bb1, otherwise: bb8];
49+
_3 = Le(const 13_u32, copy (_1.3: u32));
50+
switchInt(move _3) -> [0: bb1, otherwise: bb7];
5251
}
5352

5453
bb7: {
55-
_6 = Le(copy (_1.3: u32), const 9_u32);
56-
switchInt(move _6) -> [0: bb5, otherwise: bb8];
54+
_4 = Le(copy (_1.3: u32), const 16_u32);
55+
switchInt(move _4) -> [0: bb1, otherwise: bb9];
5756
}
5857

5958
bb8: {
60-
falseEdge -> [real: bb9, imaginary: bb1];
59+
_6 = Le(copy (_1.3: u32), const 9_u32);
60+
switchInt(move _6) -> [0: bb6, otherwise: bb9];
6161
}
6262

6363
bb9: {
64+
falseEdge -> [real: bb10, imaginary: bb1];
65+
}
66+
67+
bb10: {
6468
StorageLive(_7);
6569
_7 = copy (_1.0: u32);
6670
StorageLive(_8);
@@ -74,10 +78,10 @@ fn match_tuple(_1: (u32, bool, Option<i32>, u32)) -> u32 {
7478
StorageDead(_9);
7579
StorageDead(_8);
7680
StorageDead(_7);
77-
goto -> bb10;
81+
goto -> bb11;
7882
}
7983

80-
bb10: {
84+
bb11: {
8185
return;
8286
}
8387
}

tests/mir-opt/dead-store-elimination/place_mention.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
// and don't remove it as a dead store.
33
//
44
//@ test-mir-pass: DeadStoreElimination-initial
5-
//@ compile-flags: -Zmir-keep-place-mention
5+
//@ compile-flags: -Zmir-preserve-ub
66

77
// EMIT_MIR place_mention.main.DeadStoreElimination-initial.diff
88
fn main() {

tests/mir-opt/or_pattern.single_switchint.SimplifyCfg-initial.after.mir

+23-15
Original file line numberDiff line numberDiff line change
@@ -14,59 +14,67 @@ fn single_switchint() -> () {
1414
}
1515

1616
bb1: {
17-
switchInt(copy (_2.0: i32)) -> [3: bb8, 4: bb8, otherwise: bb7];
17+
switchInt(copy (_2.0: i32)) -> [3: bb9, 4: bb9, otherwise: bb8];
1818
}
1919

2020
bb2: {
2121
switchInt(copy (_2.1: bool)) -> [0: bb6, otherwise: bb3];
2222
}
2323

2424
bb3: {
25-
falseEdge -> [real: bb12, imaginary: bb4];
25+
falseEdge -> [real: bb14, imaginary: bb4];
2626
}
2727

2828
bb4: {
2929
switchInt(copy (_2.1: bool)) -> [0: bb5, otherwise: bb6];
3030
}
3131

3232
bb5: {
33-
falseEdge -> [real: bb11, imaginary: bb6];
33+
falseEdge -> [real: bb13, imaginary: bb6];
3434
}
3535

3636
bb6: {
37-
falseEdge -> [real: bb10, imaginary: bb1];
37+
switchInt(copy (_2.1: bool)) -> [0: bb7, otherwise: bb7];
3838
}
3939

4040
bb7: {
41-
_1 = const 5_i32;
42-
goto -> bb13;
41+
falseEdge -> [real: bb12, imaginary: bb1];
4342
}
4443

4544
bb8: {
46-
falseEdge -> [real: bb9, imaginary: bb7];
45+
_1 = const 5_i32;
46+
goto -> bb15;
4747
}
4848

4949
bb9: {
50-
_1 = const 4_i32;
51-
goto -> bb13;
50+
switchInt(copy (_2.1: bool)) -> [0: bb10, otherwise: bb10];
5251
}
5352

5453
bb10: {
55-
_1 = const 3_i32;
56-
goto -> bb13;
54+
falseEdge -> [real: bb11, imaginary: bb8];
5755
}
5856

5957
bb11: {
60-
_1 = const 2_i32;
61-
goto -> bb13;
58+
_1 = const 4_i32;
59+
goto -> bb15;
6260
}
6361

6462
bb12: {
65-
_1 = const 1_i32;
66-
goto -> bb13;
63+
_1 = const 3_i32;
64+
goto -> bb15;
6765
}
6866

6967
bb13: {
68+
_1 = const 2_i32;
69+
goto -> bb15;
70+
}
71+
72+
bb14: {
73+
_1 = const 1_i32;
74+
goto -> bb15;
75+
}
76+
77+
bb15: {
7078
StorageDead(_2);
7179
StorageDead(_1);
7280
_0 = const ();

0 commit comments

Comments
 (0)