Skip to content
This repository was archived by the owner on May 28, 2025. It is now read-only.

Commit f54bfac

Browse files
committed
Implement multiple return terminators optimization
1 parent 3bbc443 commit f54bfac

File tree

2 files changed

+41
-0
lines changed

2 files changed

+41
-0
lines changed

compiler/rustc_mir/src/transform/mod.rs

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ pub mod inline;
3333
pub mod instcombine;
3434
pub mod instrument_coverage;
3535
pub mod match_branches;
36+
pub mod multiple_return_terminators;
3637
pub mod no_landing_pads;
3738
pub mod nrvo;
3839
pub mod promote_consts;
@@ -464,6 +465,7 @@ fn run_optimization_passes<'tcx>(
464465
&remove_unneeded_drops::RemoveUnneededDrops,
465466
&match_branches::MatchBranchSimplification,
466467
// inst combine is after MatchBranchSimplification to clean up Ne(_1, false)
468+
&multiple_return_terminators::MultipleReturnTerminators,
467469
&instcombine::InstCombine,
468470
&const_prop::ConstProp,
469471
&simplify_branches::SimplifyBranches::new("after-const-prop"),
@@ -478,6 +480,7 @@ fn run_optimization_passes<'tcx>(
478480
&simplify::SimplifyCfg::new("final"),
479481
&nrvo::RenameReturnPlace,
480482
&simplify::SimplifyLocals,
483+
&multiple_return_terminators::MultipleReturnTerminators,
481484
];
482485

483486
// Optimizations to run even if mir optimizations have been disabled.
Lines changed: 38 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
//! This pass removes jumps to basic blocks containing only a return, and replaces them with a
2+
//! return instead.
3+
4+
use crate::transform::{simplify, MirPass, MirSource};
5+
use rustc_index::bit_set::BitSet;
6+
use rustc_middle::mir::*;
7+
use rustc_middle::ty::TyCtxt;
8+
9+
pub struct MultipleReturnTerminators;
10+
11+
impl<'tcx> MirPass<'tcx> for MultipleReturnTerminators {
12+
fn run_pass(&self, tcx: TyCtxt<'tcx>, _: MirSource<'tcx>, body: &mut Body<'tcx>) {
13+
if tcx.sess.opts.debugging_opts.mir_opt_level < 3 {
14+
return;
15+
}
16+
17+
// find basic blocks with no statement and a return terminator
18+
let mut bbs_simple_returns = BitSet::new_empty(body.basic_blocks().len());
19+
let bbs = body.basic_blocks_mut();
20+
for idx in bbs.indices() {
21+
if bbs[idx].statements.is_empty()
22+
&& bbs[idx].terminator().kind == TerminatorKind::Return
23+
{
24+
bbs_simple_returns.insert(idx);
25+
}
26+
}
27+
28+
for bb in bbs {
29+
if let TerminatorKind::Goto { target } = bb.terminator().kind {
30+
if bbs_simple_returns.contains(target) {
31+
bb.terminator_mut().kind = TerminatorKind::Return;
32+
}
33+
}
34+
}
35+
36+
simplify::remove_dead_blocks(body)
37+
}
38+
}

0 commit comments

Comments
 (0)