Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit e7b6b61

Browse files
committedNov 14, 2020
Limit storage duration of inlined always live locals
1 parent 30e49a9 commit e7b6b61

File tree

6 files changed

+69
-5
lines changed

6 files changed

+69
-5
lines changed
 

‎compiler/rustc_middle/src/mir/mod.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -420,7 +420,9 @@ impl<'tcx> Body<'tcx> {
420420
/// Returns an iterator over all user-defined variables and compiler-generated temporaries (all
421421
/// locals that are neither arguments nor the return place).
422422
#[inline]
423-
pub fn vars_and_temps_iter(&self) -> impl Iterator<Item = Local> + ExactSizeIterator {
423+
pub fn vars_and_temps_iter(
424+
&self,
425+
) -> impl DoubleEndedIterator<Item = Local> + ExactSizeIterator {
424426
let arg_count = self.arg_count;
425427
let local_count = self.local_decls.len();
426428
(arg_count + 1..local_count).map(Local::new)

‎compiler/rustc_mir/src/transform/inline.rs

+39
Original file line numberDiff line numberDiff line change
@@ -461,6 +461,7 @@ impl Inliner<'tcx> {
461461
tcx: self.tcx,
462462
callsite_span: callsite.source_info.span,
463463
body_span: callee_body.span,
464+
always_live_locals: BitSet::new_filled(callee_body.local_decls.len()),
464465
};
465466

466467
// Map all `Local`s, `SourceScope`s and `BasicBlock`s to new ones
@@ -492,6 +493,34 @@ impl Inliner<'tcx> {
492493
}
493494
}
494495

496+
// If there are any locals without storage markers, give them storage only for the
497+
// duration of the call.
498+
for local in callee_body.vars_and_temps_iter() {
499+
if integrator.always_live_locals.contains(local) {
500+
let new_local = integrator.map_local(local);
501+
caller_body[callsite.block].statements.push(Statement {
502+
source_info: callsite.source_info,
503+
kind: StatementKind::StorageLive(new_local),
504+
});
505+
}
506+
}
507+
if let Some(block) = callsite.target {
508+
// To avoid repeated O(n) insert, push any new statements to the end and rotate
509+
// the slice once.
510+
let mut n = 0;
511+
for local in callee_body.vars_and_temps_iter().rev() {
512+
if integrator.always_live_locals.contains(local) {
513+
let new_local = integrator.map_local(local);
514+
caller_body[block].statements.push(Statement {
515+
source_info: callsite.source_info,
516+
kind: StatementKind::StorageDead(new_local),
517+
});
518+
n += 1;
519+
}
520+
}
521+
caller_body[block].statements.rotate_right(n);
522+
}
523+
495524
// Insert all of the (mapped) parts of the callee body into the caller.
496525
caller_body.local_decls.extend(
497526
// FIXME(eddyb) make `Range<Local>` iterable so that we can use
@@ -673,6 +702,7 @@ struct Integrator<'a, 'tcx> {
673702
tcx: TyCtxt<'tcx>,
674703
callsite_span: Span,
675704
body_span: Span,
705+
always_live_locals: BitSet<Local>,
676706
}
677707

678708
impl<'a, 'tcx> Integrator<'a, 'tcx> {
@@ -762,6 +792,15 @@ impl<'a, 'tcx> MutVisitor<'tcx> for Integrator<'a, 'tcx> {
762792
}
763793
}
764794

795+
fn visit_statement(&mut self, statement: &mut Statement<'tcx>, location: Location) {
796+
if let StatementKind::StorageLive(local) | StatementKind::StorageDead(local) =
797+
statement.kind
798+
{
799+
self.always_live_locals.remove(local);
800+
}
801+
self.super_statement(statement, location);
802+
}
803+
765804
fn visit_terminator(&mut self, terminator: &mut Terminator<'tcx>, loc: Location) {
766805
// Don't try to modify the implicit `_0` access on return (`return` terminators are
767806
// replaced down below anyways).

‎src/test/mir-opt/inline/inline_diverging.h.Inline.diff

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@
4040
- // mir::Constant
4141
// + span: $DIR/inline-diverging.rs:22:16: 22:21
4242
// + literal: Const { ty: fn() -> ! {sleep}, val: Value(Scalar(<ZST>)) }
43+
+ StorageLive(_6); // scope 0 at $DIR/inline-diverging.rs:22:5: 22:22
4344
+ StorageLive(_3); // scope 1 at $DIR/inline-diverging.rs:22:5: 22:22
4445
+ StorageLive(_4); // scope 1 at $DIR/inline-diverging.rs:22:5: 22:22
4546
+ _4 = &_2; // scope 1 at $DIR/inline-diverging.rs:22:5: 22:22

‎src/test/mir-opt/inline/inline_shims.drop.Inline.diff

+4
Original file line numberDiff line numberDiff line change
@@ -36,11 +36,15 @@
3636
- // mir::Constant
3737
- // + span: $DIR/inline-shims.rs:12:14: 12:37
3838
- // + literal: Const { ty: unsafe fn(*mut std::option::Option<B>) {std::intrinsics::drop_in_place::<std::option::Option<B>>}, val: Value(Scalar(<ZST>)) }
39+
+ StorageLive(_6); // scope 2 at $DIR/inline-shims.rs:12:14: 12:40
40+
+ StorageLive(_7); // scope 2 at $DIR/inline-shims.rs:12:14: 12:40
3941
+ _6 = discriminant((*_5)); // scope 3 at $DIR/inline-shims.rs:12:14: 12:40
4042
+ switchInt(move _6) -> [0_isize: bb2, otherwise: bb3]; // scope 3 at $DIR/inline-shims.rs:12:14: 12:40
4143
}
4244

4345
bb2: {
46+
+ StorageDead(_7); // scope 2 at $DIR/inline-shims.rs:12:14: 12:40
47+
+ StorageDead(_6); // scope 2 at $DIR/inline-shims.rs:12:14: 12:40
4448
StorageDead(_5); // scope 2 at $DIR/inline-shims.rs:12:39: 12:40
4549
return; // scope 0 at $DIR/inline-shims.rs:13:2: 13:2
4650
+ }
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
11
// edition:2018
2-
// compile-flags: -Z mir-opt-level=2 -Z unsound-mir-opts
2+
// compile-flags: -Z mir-opt-level=2
33

44
#[inline(always)]
5-
pub fn f(s: bool) -> String {
5+
pub fn copy_prop(s: bool) -> String {
66
let a = "Hello world!".to_string();
77
let b = a;
88
let c = b;
@@ -12,3 +12,9 @@ pub fn f(s: bool) -> String {
1212
String::new()
1313
}
1414
}
15+
16+
#[inline(always)]
17+
pub fn dest_prop(x: &[u8]) -> &[u8] {
18+
let y = &x[..x.len()];
19+
y
20+
}

‎src/test/ui/mir/issue-76375.rs

+14-2
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,27 @@
1+
// Regression test for issue #76375.
2+
//
13
// edition:2018
24
// build-pass
3-
// compile-flags: -Z mir-opt-level=2 -L.
5+
// compile-flags: -Z mir-opt-level=2
46
// aux-build:issue_76375_aux.rs
57

68
#![crate_type = "lib"]
79

810
extern crate issue_76375_aux;
911

1012
pub async fn g() {
11-
issue_76375_aux::f(true);
13+
issue_76375_aux::copy_prop(true);
1214
h().await;
1315
}
1416

17+
pub async fn u() {
18+
let b = [0u8; 32];
19+
let mut i = 0;
20+
while i != 10 {
21+
issue_76375_aux::dest_prop(&b);
22+
h().await;
23+
i += 1;
24+
}
25+
}
26+
1527
pub async fn h() {}

0 commit comments

Comments
 (0)
Please sign in to comment.