@@ -66,8 +66,8 @@ impl SimplifyCfg {
66
66
}
67
67
}
68
68
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 ( ) ;
71
71
remove_dead_blocks ( body) ;
72
72
73
73
// FIXME: Should probably be moved into some kind of pass manager
@@ -79,9 +79,9 @@ impl<'tcx> crate::MirPass<'tcx> for SimplifyCfg {
79
79
self . name ( )
80
80
}
81
81
82
- fn run_pass ( & self , _ : TyCtxt < ' tcx > , body : & mut Body < ' tcx > ) {
82
+ fn run_pass ( & self , tcx : TyCtxt < ' tcx > , body : & mut Body < ' tcx > ) {
83
83
debug ! ( "SimplifyCfg({:?}) - simplifying {:?}" , self . name( ) , body. source) ;
84
- simplify_cfg ( body) ;
84
+ simplify_cfg ( tcx , body) ;
85
85
}
86
86
87
87
fn is_required ( & self ) -> bool {
@@ -90,12 +90,13 @@ impl<'tcx> crate::MirPass<'tcx> for SimplifyCfg {
90
90
}
91
91
92
92
struct CfgSimplifier < ' a , ' tcx > {
93
+ preserve_switch_reads : bool ,
93
94
basic_blocks : & ' a mut IndexSlice < BasicBlock , BasicBlockData < ' tcx > > ,
94
95
pred_count : IndexVec < BasicBlock , u32 > ,
95
96
}
96
97
97
98
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 {
99
100
let mut pred_count = IndexVec :: from_elem ( 0u32 , & body. basic_blocks ) ;
100
101
101
102
// we can't use mir.predecessors() here because that counts
@@ -110,9 +111,13 @@ impl<'a, 'tcx> CfgSimplifier<'a, 'tcx> {
110
111
}
111
112
}
112
113
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 ;
113
118
let basic_blocks = body. basic_blocks_mut ( ) ;
114
119
115
- CfgSimplifier { basic_blocks, pred_count }
120
+ CfgSimplifier { preserve_switch_reads , basic_blocks, pred_count }
116
121
}
117
122
118
123
fn simplify ( mut self ) {
@@ -253,9 +258,15 @@ impl<'a, 'tcx> CfgSimplifier<'a, 'tcx> {
253
258
254
259
// turn a branch with all successors identical to a goto
255
260
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 ;
259
270
} ;
260
271
261
272
let first_succ = {
0 commit comments