2 use rustc_index::bit_set::BitSet;
3 use rustc_middle::mir::patch::MirPatch;
4 use rustc_middle::mir::*;
5 use rustc_middle::ty::TyCtxt;
6 use rustc_target::spec::PanicStrategy;
8 /// A pass that removes noop landing pads and replaces jumps to them with
9 /// `None`. This is important because otherwise LLVM generates terrible
11 pub struct RemoveNoopLandingPads;
13 impl<'tcx> MirPass<'tcx> for RemoveNoopLandingPads {
14 fn is_enabled(&self, sess: &rustc_session::Session) -> bool {
15 sess.panic_strategy() != PanicStrategy::Abort
18 fn run_pass(&self, _tcx: TyCtxt<'tcx>, body: &mut Body<'tcx>) {
19 debug!("remove_noop_landing_pads({:?})", body);
20 self.remove_nop_landing_pads(body)
24 impl RemoveNoopLandingPads {
25 fn is_nop_landing_pad(
29 nop_landing_pads: &BitSet<BasicBlock>,
31 for stmt in &body[bb].statements {
33 StatementKind::FakeRead(..)
34 | StatementKind::StorageLive(_)
35 | StatementKind::StorageDead(_)
36 | StatementKind::AscribeUserType(..)
37 | StatementKind::Coverage(..)
38 | StatementKind::ConstEvalCounter
39 | StatementKind::Nop => {
40 // These are all noops in a landing pad
43 StatementKind::Assign(box (place, Rvalue::Use(_) | Rvalue::Discriminant(_))) => {
44 if place.as_local().is_some() {
45 // Writing to a local (e.g., a drop flag) does not
46 // turn a landing pad to a non-nop
52 StatementKind::Assign { .. }
53 | StatementKind::SetDiscriminant { .. }
54 | StatementKind::Deinit(..)
55 | StatementKind::Intrinsic(..)
56 | StatementKind::Retag { .. } => {
62 let terminator = body[bb].terminator();
63 match terminator.kind {
64 TerminatorKind::Goto { .. }
65 | TerminatorKind::Resume
66 | TerminatorKind::SwitchInt { .. }
67 | TerminatorKind::FalseEdge { .. }
68 | TerminatorKind::FalseUnwind { .. } => {
69 terminator.successors().all(|succ| nop_landing_pads.contains(succ))
71 TerminatorKind::GeneratorDrop
72 | TerminatorKind::Yield { .. }
73 | TerminatorKind::Return
74 | TerminatorKind::Abort
75 | TerminatorKind::Unreachable
76 | TerminatorKind::Call { .. }
77 | TerminatorKind::Assert { .. }
78 | TerminatorKind::DropAndReplace { .. }
79 | TerminatorKind::Drop { .. }
80 | TerminatorKind::InlineAsm { .. } => false,
84 fn remove_nop_landing_pads(&self, body: &mut Body<'_>) {
85 debug!("body: {:#?}", body);
87 // make sure there's a resume block
89 let mut patch = MirPatch::new(body);
90 let resume_block = patch.resume_block();
94 debug!("remove_noop_landing_pads: resume block is {:?}", resume_block);
96 let mut jumps_folded = 0;
97 let mut landing_pads_removed = 0;
98 let mut nop_landing_pads = BitSet::new_empty(body.basic_blocks.len());
100 // This is a post-order traversal, so that if A post-dominates B
101 // then A will be visited before B.
102 let postorder: Vec<_> = traversal::postorder(body).map(|(bb, _)| bb).collect();
103 for bb in postorder {
104 debug!(" processing {:?}", bb);
105 if let Some(unwind) = body[bb].terminator_mut().unwind_mut() {
106 if let Some(unwind_bb) = *unwind {
107 if nop_landing_pads.contains(unwind_bb) {
108 debug!(" removing noop landing pad");
109 landing_pads_removed += 1;
115 for target in body[bb].terminator_mut().successors_mut() {
116 if *target != resume_block && nop_landing_pads.contains(*target) {
117 debug!(" folding noop jump to {:?} to resume block", target);
118 *target = resume_block;
123 let is_nop_landing_pad = self.is_nop_landing_pad(bb, body, &nop_landing_pads);
124 if is_nop_landing_pad {
125 nop_landing_pads.insert(bb);
127 debug!(" is_nop_landing_pad({:?}) = {}", bb, is_nop_landing_pad);
130 debug!("removed {:?} jumps and {:?} landing pads", jumps_folded, landing_pads_removed);