Skip to content

Commit e8f9a6f

Browse files
committed
Auto merge of #127024 - cjgillot:jump-prof, r=<try>
Avoid cloning jump threading state when possible The current implementation of jump threading passes most of its time cloning its state. This PR attempts to avoid such clones by special-casing the last predecessor when recursing through a terminator. This is not optimal, but a first step while I refactor the state data structure to be sparse. The two other commits are drive-by. Fixes #116721 r? `@oli-obk`
2 parents 536235f + 271b9c2 commit e8f9a6f

File tree

2 files changed

+52
-45
lines changed

2 files changed

+52
-45
lines changed

compiler/rustc_mir_dataflow/src/value_analysis.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -846,9 +846,10 @@ impl Map {
846846

847847
if let ty::Ref(_, ref_ty, _) | ty::RawPtr(ref_ty, _) = ty.kind()
848848
&& let ty::Slice(..) = ref_ty.kind()
849+
// The user may have written a predicate like `[T]: Sized` in their where claures,
850+
// which makes slices scalars.
851+
&& self.places[place].value_index.is_none()
849852
{
850-
assert!(self.places[place].value_index.is_none(), "slices are not scalars");
851-
852853
// Prepend new child to the linked list.
853854
let len = self.places.push(PlaceInfo::new(Some(TrackElem::DerefLen)));
854855
self.places[len].next_sibling = self.places[place].first_child;

compiler/rustc_mir_transform/src/jump_threading.rs

+49-43
Original file line numberDiff line numberDiff line change
@@ -91,43 +91,8 @@ impl<'tcx> MirPass<'tcx> for JumpThreading {
9191
opportunities: Vec::new(),
9292
};
9393

94-
for (bb, bbdata) in body.basic_blocks.iter_enumerated() {
95-
debug!(?bb, term = ?bbdata.terminator());
96-
if bbdata.is_cleanup || loop_headers.contains(bb) {
97-
continue;
98-
}
99-
let Some((discr, targets)) = bbdata.terminator().kind.as_switch() else { continue };
100-
let Some(discr) = discr.place() else { continue };
101-
debug!(?discr, ?bb);
102-
103-
let discr_ty = discr.ty(body, tcx).ty;
104-
let Ok(discr_layout) = finder.ecx.layout_of(discr_ty) else { continue };
105-
106-
let Some(discr) = finder.map.find(discr.as_ref()) else { continue };
107-
debug!(?discr);
108-
109-
let cost = CostChecker::new(tcx, param_env, None, body);
110-
111-
let mut state = State::new(ConditionSet::default(), finder.map);
112-
113-
let conds = if let Some((value, then, else_)) = targets.as_static_if() {
114-
let Some(value) = ScalarInt::try_from_uint(value, discr_layout.size) else {
115-
continue;
116-
};
117-
arena.alloc_from_iter([
118-
Condition { value, polarity: Polarity::Eq, target: then },
119-
Condition { value, polarity: Polarity::Ne, target: else_ },
120-
])
121-
} else {
122-
arena.alloc_from_iter(targets.iter().filter_map(|(value, target)| {
123-
let value = ScalarInt::try_from_uint(value, discr_layout.size)?;
124-
Some(Condition { value, polarity: Polarity::Eq, target })
125-
}))
126-
};
127-
let conds = ConditionSet(conds);
128-
state.insert_value_idx(discr, conds, finder.map);
129-
130-
finder.find_opportunity(bb, state, cost, 0);
94+
for bb in body.basic_blocks.indices() {
95+
finder.start_from_switch(bb);
13196
}
13297

13398
let opportunities = finder.opportunities;
@@ -215,6 +180,45 @@ impl<'tcx, 'a> TOFinder<'tcx, 'a> {
215180
state.all(|cs| cs.0.is_empty())
216181
}
217182

183+
/// Recursion entry point to find threading opportunities.
184+
#[instrument(level = "trace", skip(self))]
185+
fn start_from_switch(&mut self, bb: BasicBlock) -> Option<!> {
186+
let bbdata = &self.body[bb];
187+
if bbdata.is_cleanup || self.loop_headers.contains(bb) {
188+
return None;
189+
}
190+
let (discr, targets) = bbdata.terminator().kind.as_switch()?;
191+
let discr = discr.place()?;
192+
debug!(?discr, ?bb);
193+
194+
let discr_ty = discr.ty(self.body, self.tcx).ty;
195+
let discr_layout = self.ecx.layout_of(discr_ty).ok()?;
196+
197+
let discr = self.map.find(discr.as_ref())?;
198+
debug!(?discr);
199+
200+
let cost = CostChecker::new(self.tcx, self.param_env, None, self.body);
201+
let mut state = State::new(ConditionSet::default(), self.map);
202+
203+
let conds = if let Some((value, then, else_)) = targets.as_static_if() {
204+
let value = ScalarInt::try_from_uint(value, discr_layout.size)?;
205+
self.arena.alloc_from_iter([
206+
Condition { value, polarity: Polarity::Eq, target: then },
207+
Condition { value, polarity: Polarity::Ne, target: else_ },
208+
])
209+
} else {
210+
self.arena.alloc_from_iter(targets.iter().filter_map(|(value, target)| {
211+
let value = ScalarInt::try_from_uint(value, discr_layout.size)?;
212+
Some(Condition { value, polarity: Polarity::Eq, target })
213+
}))
214+
};
215+
let conds = ConditionSet(conds);
216+
state.insert_value_idx(discr, conds, self.map);
217+
218+
self.find_opportunity(bb, state, cost, 0);
219+
None
220+
}
221+
218222
/// Recursion entry point to find threading opportunities.
219223
#[instrument(level = "trace", skip(self, cost), ret)]
220224
fn find_opportunity(
@@ -270,12 +274,13 @@ impl<'tcx, 'a> TOFinder<'tcx, 'a> {
270274
self.process_switch_int(discr, targets, bb, &mut state);
271275
self.find_opportunity(pred, state, cost, depth + 1);
272276
}
273-
_ => self.recurse_through_terminator(pred, &state, &cost, depth),
277+
_ => self.recurse_through_terminator(pred, || state, &cost, depth),
274278
}
275-
} else {
279+
} else if let &[ref predecessors @ .., last_pred] = &predecessors[..] {
276280
for &pred in predecessors {
277-
self.recurse_through_terminator(pred, &state, &cost, depth);
281+
self.recurse_through_terminator(pred, || state.clone(), &cost, depth);
278282
}
283+
self.recurse_through_terminator(last_pred, || state, &cost, depth);
279284
}
280285

281286
let new_tos = &mut self.opportunities[last_non_rec..];
@@ -566,11 +571,12 @@ impl<'tcx, 'a> TOFinder<'tcx, 'a> {
566571
None
567572
}
568573

569-
#[instrument(level = "trace", skip(self, cost))]
574+
#[instrument(level = "trace", skip(self, state, cost))]
570575
fn recurse_through_terminator(
571576
&mut self,
572577
bb: BasicBlock,
573-
state: &State<ConditionSet<'a>>,
578+
// Pass a closure that may clone the state, as we don't want to do it each time.
579+
state: impl FnOnce() -> State<ConditionSet<'a>>,
574580
cost: &CostChecker<'_, 'tcx>,
575581
depth: usize,
576582
) {
@@ -600,7 +606,7 @@ impl<'tcx, 'a> TOFinder<'tcx, 'a> {
600606
};
601607

602608
// We can recurse through this terminator.
603-
let mut state = state.clone();
609+
let mut state = state();
604610
if let Some(place_to_flood) = place_to_flood {
605611
state.flood_with(place_to_flood.as_ref(), self.map, ConditionSet::default());
606612
}

0 commit comments

Comments
 (0)