Skip to content

Commit 21107e0

Browse files
committed
Simplify dataflow SwitchInt handling.
Current `SwitchInt` handling has complicated control flow. - The dataflow engine calls `Analysis::apply_switch_int_edge_effects`, passing in an "applier" that impls `SwitchIntEdgeEffects`. - `apply_switch_int_edge_effects` possibly calls `apply` on the applier, passing it a closure. - The `apply` method calls the closure on each `SwitchInt` edge. - The closure operates on the edge. I.e. control flow goes from the engine, to the analysis, to the applier (which came from the engine), to the closure (which came from the analysis). It took me a while to work this out. This commit changes to a simpler structure that maintains the important characteristics. - The dataflow engine calls `Analysis::get_switch_int_data`. - `get_switch_int_data` returns an `Option<Self::SwitchIntData>` value. - If that returned value was `Some`, the dataflow engine calls `Analysis::apply_switch_int_edge_effect` on each edge, passing the `Self::SwitchIntData` value. - `Analysis::apply_switch_int_edge_effect` operates on the edge. I.e. control flow goes from the engine, to the analysis, to the engine, to the analysis. Added: - The `Analysis::SwitchIntData` assoc type and the `Analysis::get_switch_int_data` method. Both only need to be defined by analyses that look at `SwitchInt` terminators. - The `MaybePlacesSwitchIntData` struct, which has three fields. Changes: - `Analysis::apply_switch_int_edge_effects` becomes `Analysis::apply_switch_int_edge_effect`, which is a little simpler because it's dealing with a single edge instead of all edges. Removed: - The `SwitchIntEdgeEffects` trait, and its two impls: `BackwardSwitchIntEdgeEffectsApplier` (which has six fields) and `ForwardSwitchIntEdgeEffectsApplier` structs (which has four fields). - The closure. The new structure is more concise and simpler.
1 parent 551dade commit 21107e0

File tree

5 files changed

+139
-183
lines changed

5 files changed

+139
-183
lines changed

compiler/rustc_borrowck/src/dataflow.rs

+1-11
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use rustc_middle::mir::{
99
use rustc_middle::ty::{RegionVid, TyCtxt};
1010
use rustc_mir_dataflow::fmt::DebugWithContext;
1111
use rustc_mir_dataflow::impls::{EverInitializedPlaces, MaybeUninitializedPlaces};
12-
use rustc_mir_dataflow::{Analysis, GenKill, JoinSemiLattice, SwitchIntEdgeEffects};
12+
use rustc_mir_dataflow::{Analysis, GenKill, JoinSemiLattice};
1313
use tracing::debug;
1414

1515
use crate::{BorrowSet, PlaceConflictBias, PlaceExt, RegionInferenceContext, places_conflict};
@@ -98,16 +98,6 @@ impl<'a, 'tcx> Analysis<'tcx> for Borrowck<'a, 'tcx> {
9898
// This is only reachable from `iterate_to_fixpoint`, which this analysis doesn't use.
9999
unreachable!();
100100
}
101-
102-
fn apply_switch_int_edge_effects(
103-
&mut self,
104-
_block: BasicBlock,
105-
_discr: &mir::Operand<'tcx>,
106-
_apply_edge_effects: &mut impl SwitchIntEdgeEffects<Self::Domain>,
107-
) {
108-
// This is only reachable from `iterate_to_fixpoint`, which this analysis doesn't use.
109-
unreachable!();
110-
}
111101
}
112102

113103
impl JoinSemiLattice for BorrowckDomain<'_, '_> {

compiler/rustc_mir_dataflow/src/framework/direction.rs

+34-96
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,6 @@
11
use std::ops::RangeInclusive;
22

3-
use rustc_middle::mir::{
4-
self, BasicBlock, CallReturnPlaces, Location, SwitchTargets, TerminatorEdges,
5-
};
3+
use rustc_middle::mir::{self, BasicBlock, CallReturnPlaces, Location, TerminatorEdges};
64

75
use super::visitor::ResultsVisitor;
86
use super::{Analysis, Effect, EffectIndex, Results, SwitchIntTarget};
@@ -115,18 +113,18 @@ impl Direction for Backward {
115113
}
116114

117115
mir::TerminatorKind::SwitchInt { targets: _, ref discr } => {
118-
let mut applier = BackwardSwitchIntEdgeEffectsApplier {
119-
body,
120-
pred,
121-
exit_state,
122-
block,
123-
propagate: &mut propagate,
124-
effects_applied: false,
125-
};
126-
127-
analysis.apply_switch_int_edge_effects(pred, discr, &mut applier);
128-
129-
if !applier.effects_applied {
116+
if let Some(mut data) = analysis.get_switch_int_data(block, discr) {
117+
let values = &body.basic_blocks.switch_sources()[&(block, pred)];
118+
let targets =
119+
values.iter().map(|&value| SwitchIntTarget { value, target: block });
120+
121+
let mut tmp = None;
122+
for target in targets {
123+
let tmp = opt_clone_from_or_clone(&mut tmp, exit_state);
124+
analysis.apply_switch_int_edge_effect(&mut data, tmp, target);
125+
propagate(pred, tmp);
126+
}
127+
} else {
130128
propagate(pred, exit_state)
131129
}
132130
}
@@ -245,37 +243,6 @@ impl Direction for Backward {
245243
}
246244
}
247245

248-
struct BackwardSwitchIntEdgeEffectsApplier<'mir, 'tcx, D, F> {
249-
body: &'mir mir::Body<'tcx>,
250-
pred: BasicBlock,
251-
exit_state: &'mir mut D,
252-
block: BasicBlock,
253-
propagate: &'mir mut F,
254-
effects_applied: bool,
255-
}
256-
257-
impl<D, F> super::SwitchIntEdgeEffects<D> for BackwardSwitchIntEdgeEffectsApplier<'_, '_, D, F>
258-
where
259-
D: Clone,
260-
F: FnMut(BasicBlock, &D),
261-
{
262-
fn apply(&mut self, mut apply_edge_effect: impl FnMut(&mut D, SwitchIntTarget)) {
263-
assert!(!self.effects_applied);
264-
265-
let values = &self.body.basic_blocks.switch_sources()[&(self.block, self.pred)];
266-
let targets = values.iter().map(|&value| SwitchIntTarget { value, target: self.block });
267-
268-
let mut tmp = None;
269-
for target in targets {
270-
let tmp = opt_clone_from_or_clone(&mut tmp, self.exit_state);
271-
apply_edge_effect(tmp, target);
272-
(self.propagate)(self.pred, tmp);
273-
}
274-
275-
self.effects_applied = true;
276-
}
277-
}
278-
279246
/// Dataflow that runs from the entry of a block (the first statement), to its exit (terminator).
280247
pub struct Forward;
281248

@@ -324,23 +291,27 @@ impl Direction for Forward {
324291
}
325292
}
326293
TerminatorEdges::SwitchInt { targets, discr } => {
327-
let mut applier = ForwardSwitchIntEdgeEffectsApplier {
328-
exit_state,
329-
targets,
330-
propagate,
331-
effects_applied: false,
332-
};
333-
334-
analysis.apply_switch_int_edge_effects(block, discr, &mut applier);
335-
336-
let ForwardSwitchIntEdgeEffectsApplier {
337-
exit_state,
338-
mut propagate,
339-
effects_applied,
340-
..
341-
} = applier;
342-
343-
if !effects_applied {
294+
if let Some(mut data) = analysis.get_switch_int_data(block, discr) {
295+
let mut tmp = None;
296+
for (value, target) in targets.iter() {
297+
let tmp = opt_clone_from_or_clone(&mut tmp, exit_state);
298+
analysis.apply_switch_int_edge_effect(&mut data, tmp, SwitchIntTarget {
299+
value: Some(value),
300+
target,
301+
});
302+
propagate(target, tmp);
303+
}
304+
305+
// Once we get to the final, "otherwise" branch, there is no need to preserve
306+
// `exit_state`, so pass it directly to `apply_switch_int_edge_effect` to save
307+
// a clone of the dataflow state.
308+
let otherwise = targets.otherwise();
309+
analysis.apply_switch_int_edge_effect(&mut data, exit_state, SwitchIntTarget {
310+
value: None,
311+
target: otherwise,
312+
});
313+
propagate(otherwise, exit_state);
314+
} else {
344315
for target in targets.all_targets() {
345316
propagate(*target, exit_state);
346317
}
@@ -455,39 +426,6 @@ impl Direction for Forward {
455426
}
456427
}
457428

458-
struct ForwardSwitchIntEdgeEffectsApplier<'mir, D, F> {
459-
exit_state: &'mir mut D,
460-
targets: &'mir SwitchTargets,
461-
propagate: F,
462-
463-
effects_applied: bool,
464-
}
465-
466-
impl<D, F> super::SwitchIntEdgeEffects<D> for ForwardSwitchIntEdgeEffectsApplier<'_, D, F>
467-
where
468-
D: Clone,
469-
F: FnMut(BasicBlock, &D),
470-
{
471-
fn apply(&mut self, mut apply_edge_effect: impl FnMut(&mut D, SwitchIntTarget)) {
472-
assert!(!self.effects_applied);
473-
474-
let mut tmp = None;
475-
for (value, target) in self.targets.iter() {
476-
let tmp = opt_clone_from_or_clone(&mut tmp, self.exit_state);
477-
apply_edge_effect(tmp, SwitchIntTarget { value: Some(value), target });
478-
(self.propagate)(target, tmp);
479-
}
480-
481-
// Once we get to the final, "otherwise" branch, there is no need to preserve `exit_state`,
482-
// so pass it directly to `apply_edge_effect` to save a clone of the dataflow state.
483-
let otherwise = self.targets.otherwise();
484-
apply_edge_effect(self.exit_state, SwitchIntTarget { value: None, target: otherwise });
485-
(self.propagate)(otherwise, self.exit_state);
486-
487-
self.effects_applied = true;
488-
}
489-
}
490-
491429
/// An analogue of `Option::get_or_insert_with` that stores a clone of `val` into `opt`, but uses
492430
/// the more efficient `clone_from` if `opt` was `Some`.
493431
///

compiler/rustc_mir_dataflow/src/framework/mod.rs

+23-16
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,9 @@ pub trait Analysis<'tcx> {
121121
/// The direction of this analysis. Either `Forward` or `Backward`.
122122
type Direction: Direction = Forward;
123123

124+
/// Auxiliary data used for analyzing `SwitchInt` terminators, if necessary.
125+
type SwitchIntData = !;
126+
124127
/// A descriptive name for this analysis. Used only for debugging.
125128
///
126129
/// This name should be brief and contain no spaces, periods or other characters that are not
@@ -206,25 +209,36 @@ pub trait Analysis<'tcx> {
206209
) {
207210
}
208211

209-
/// Updates the current dataflow state with the effect of taking a particular branch in a
210-
/// `SwitchInt` terminator.
212+
/// Used to update the current dataflow state with the effect of taking a particular branch in
213+
/// a `SwitchInt` terminator.
211214
///
212215
/// Unlike the other edge-specific effects, which are allowed to mutate `Self::Domain`
213-
/// directly, overriders of this method must pass a callback to
214-
/// `SwitchIntEdgeEffects::apply`. The callback will be run once for each outgoing edge and
215-
/// will have access to the dataflow state that will be propagated along that edge.
216+
/// directly, overriders of this method must return a `Self::SwitchIntData` value (wrapped in
217+
/// `Some`). The `apply_switch_int_edge_effect` method will then be called once for each
218+
/// outgoing edge and will have access to the dataflow state that will be propagated along that
219+
/// edge, and also the `Self::SwitchIntData` value.
216220
///
217221
/// This interface is somewhat more complex than the other visitor-like "effect" methods.
218222
/// However, it is both more ergonomic—callers don't need to recompute or cache information
219223
/// about a given `SwitchInt` terminator for each one of its edges—and more efficient—the
220224
/// engine doesn't need to clone the exit state for a block unless
221-
/// `SwitchIntEdgeEffects::apply` is actually called.
222-
fn apply_switch_int_edge_effects(
225+
/// `get_switch_int_data` is actually called.
226+
fn get_switch_int_data(
223227
&mut self,
224-
_block: BasicBlock,
228+
_block: mir::BasicBlock,
225229
_discr: &mir::Operand<'tcx>,
226-
_apply_edge_effects: &mut impl SwitchIntEdgeEffects<Self::Domain>,
230+
) -> Option<Self::SwitchIntData> {
231+
None
232+
}
233+
234+
/// See comments on `get_switch_int_data`.
235+
fn apply_switch_int_edge_effect(
236+
&mut self,
237+
_data: &mut Self::SwitchIntData,
238+
_state: &mut Self::Domain,
239+
_edge: SwitchIntTarget,
227240
) {
241+
unreachable!();
228242
}
229243

230244
/* Extension methods */
@@ -437,12 +451,5 @@ pub struct SwitchIntTarget {
437451
pub target: BasicBlock,
438452
}
439453

440-
/// A type that records the edge-specific effects for a `SwitchInt` terminator.
441-
pub trait SwitchIntEdgeEffects<D> {
442-
/// Calls `apply_edge_effect` for each outgoing edge from a `SwitchInt` terminator and
443-
/// records the results.
444-
fn apply(&mut self, apply_edge_effect: impl FnMut(&mut D, SwitchIntTarget));
445-
}
446-
447454
#[cfg(test)]
448455
mod tests;

0 commit comments

Comments
 (0)