Auto merge of #38989 - arielb1:constant-mir-overflow2, r=eddyb

fix function arguments in constant promotion

we can't create the target block until *after* we promote the arguments - otherwise the arguments will be promoted into the target block. oops.

Fixes #38985.

This is a regression introduced in the beta-nominated #38833, so beta-nominating this one too (sorry @brson).

r? @eddyb
This commit is contained in:
bors 2017-01-11 15:45:28 +00:00
commit 513d942a7e
2 changed files with 22 additions and 13 deletions

View File

@ -237,7 +237,7 @@ impl<'a, 'tcx> Promoter<'a, 'tcx> {
self.visit_rvalue(&mut rvalue, loc);
self.assign(new_temp, rvalue, source_info.span);
} else {
let mut terminator = if self.keep_original {
let terminator = if self.keep_original {
self.source[loc.block].terminator().clone()
} else {
let terminator = self.source[loc.block].terminator_mut();
@ -255,28 +255,30 @@ impl<'a, 'tcx> Promoter<'a, 'tcx> {
}
};
let last = self.promoted.basic_blocks().last().unwrap();
let new_target = self.new_block();
terminator.kind = match terminator.kind {
match terminator.kind {
TerminatorKind::Call { mut func, mut args, .. } => {
self.visit_operand(&mut func, loc);
for arg in &mut args {
self.visit_operand(arg, loc);
}
TerminatorKind::Call {
func: func,
args: args,
cleanup: None,
destination: Some((Lvalue::Local(new_temp), new_target))
}
let last = self.promoted.basic_blocks().last().unwrap();
let new_target = self.new_block();
*self.promoted[last].terminator_mut() = Terminator {
kind: TerminatorKind::Call {
func: func,
args: args,
cleanup: None,
destination: Some((Lvalue::Local(new_temp), new_target))
},
..terminator
};
}
ref kind => {
span_bug!(terminator.source_info.span, "{:?} not promotable", kind);
}
};
*self.promoted[last].terminator_mut() = terminator;
};
self.keep_original = old_keep_original;

View File

@ -14,7 +14,14 @@ const fn foo() -> i64 {
3
}
const fn bar(x: i64) -> i64 {
x*2
}
fn main() {
let val = &(foo() % 2);
assert_eq!(*val, 1);
let val2 = &(bar(1+1) % 3);
assert_eq!(*val2, 1);
}