[mir-opt] Handle const-prop for the return place
This commit is contained in:
parent
0d5264a03c
commit
8316701d37
2 changed files with 87 additions and 5 deletions
|
@ -9,7 +9,7 @@ use rustc::hir::def_id::DefId;
|
||||||
use rustc::mir::{
|
use rustc::mir::{
|
||||||
AggregateKind, Constant, Location, Place, PlaceBase, Body, Operand, Rvalue, Local, UnOp,
|
AggregateKind, Constant, Location, Place, PlaceBase, Body, Operand, Rvalue, Local, UnOp,
|
||||||
StatementKind, Statement, LocalKind, TerminatorKind, Terminator, ClearCrossCrate, SourceInfo,
|
StatementKind, Statement, LocalKind, TerminatorKind, Terminator, ClearCrossCrate, SourceInfo,
|
||||||
BinOp, SourceScope, SourceScopeLocalData, LocalDecl, BasicBlock,
|
BinOp, SourceScope, SourceScopeLocalData, LocalDecl, BasicBlock, RETURN_PLACE,
|
||||||
};
|
};
|
||||||
use rustc::mir::visit::{
|
use rustc::mir::visit::{
|
||||||
Visitor, PlaceContext, MutatingUseContext, MutVisitor, NonMutatingUseContext,
|
Visitor, PlaceContext, MutatingUseContext, MutVisitor, NonMutatingUseContext,
|
||||||
|
@ -25,6 +25,7 @@ use rustc::ty::layout::{
|
||||||
LayoutOf, TyLayout, LayoutError, HasTyCtxt, TargetDataLayout, HasDataLayout,
|
LayoutOf, TyLayout, LayoutError, HasTyCtxt, TargetDataLayout, HasDataLayout,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::rustc::ty::subst::Subst;
|
||||||
use crate::interpret::{
|
use crate::interpret::{
|
||||||
self, InterpCx, ScalarMaybeUndef, Immediate, OpTy,
|
self, InterpCx, ScalarMaybeUndef, Immediate, OpTy,
|
||||||
StackPopCleanup, LocalValue, LocalState, AllocId, Frame,
|
StackPopCleanup, LocalValue, LocalState, AllocId, Frame,
|
||||||
|
@ -269,6 +270,7 @@ struct ConstPropagator<'mir, 'tcx> {
|
||||||
param_env: ParamEnv<'tcx>,
|
param_env: ParamEnv<'tcx>,
|
||||||
source_scope_local_data: ClearCrossCrate<IndexVec<SourceScope, SourceScopeLocalData>>,
|
source_scope_local_data: ClearCrossCrate<IndexVec<SourceScope, SourceScopeLocalData>>,
|
||||||
local_decls: IndexVec<Local, LocalDecl<'tcx>>,
|
local_decls: IndexVec<Local, LocalDecl<'tcx>>,
|
||||||
|
ret: Option<OpTy<'tcx, ()>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'mir, 'tcx> LayoutOf for ConstPropagator<'mir, 'tcx> {
|
impl<'mir, 'tcx> LayoutOf for ConstPropagator<'mir, 'tcx> {
|
||||||
|
@ -308,11 +310,21 @@ impl<'mir, 'tcx> ConstPropagator<'mir, 'tcx> {
|
||||||
let mut ecx = InterpCx::new(tcx.at(span), param_env, ConstPropMachine, ());
|
let mut ecx = InterpCx::new(tcx.at(span), param_env, ConstPropMachine, ());
|
||||||
let can_const_prop = CanConstProp::check(body);
|
let can_const_prop = CanConstProp::check(body);
|
||||||
|
|
||||||
|
let substs = &InternalSubsts::identity_for_item(tcx, def_id);
|
||||||
|
|
||||||
|
let ret =
|
||||||
|
ecx
|
||||||
|
.layout_of(body.return_ty().subst(tcx, substs))
|
||||||
|
.ok()
|
||||||
|
// Don't bother allocating memory for ZST types which have no values.
|
||||||
|
.filter(|ret_layout| !ret_layout.is_zst())
|
||||||
|
.map(|ret_layout| ecx.allocate(ret_layout, MemoryKind::Stack));
|
||||||
|
|
||||||
ecx.push_stack_frame(
|
ecx.push_stack_frame(
|
||||||
Instance::new(def_id, &InternalSubsts::identity_for_item(tcx, def_id)),
|
Instance::new(def_id, substs),
|
||||||
span,
|
span,
|
||||||
dummy_body,
|
dummy_body,
|
||||||
None,
|
ret.map(Into::into),
|
||||||
StackPopCleanup::None {
|
StackPopCleanup::None {
|
||||||
cleanup: false,
|
cleanup: false,
|
||||||
},
|
},
|
||||||
|
@ -327,6 +339,7 @@ impl<'mir, 'tcx> ConstPropagator<'mir, 'tcx> {
|
||||||
source_scope_local_data,
|
source_scope_local_data,
|
||||||
//FIXME(wesleywiser) we can't steal this because `Visitor::super_visit_body()` needs it
|
//FIXME(wesleywiser) we can't steal this because `Visitor::super_visit_body()` needs it
|
||||||
local_decls: body.local_decls.clone(),
|
local_decls: body.local_decls.clone(),
|
||||||
|
ret: ret.map(Into::into),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -335,6 +348,15 @@ impl<'mir, 'tcx> ConstPropagator<'mir, 'tcx> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn get_const(&self, local: Local) -> Option<Const<'tcx>> {
|
fn get_const(&self, local: Local) -> Option<Const<'tcx>> {
|
||||||
|
if local == RETURN_PLACE {
|
||||||
|
// Try to read the return place as an immediate so that if it is representable as a
|
||||||
|
// scalar, we can handle it as such, but otherwise, just return the value as is.
|
||||||
|
return match self.ret.map(|ret| self.ecx.try_read_immediate(ret)) {
|
||||||
|
Some(Ok(Ok(imm))) => Some(imm.into()),
|
||||||
|
_ => self.ret,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
self.ecx.access_local(self.ecx.frame(), local, None).ok()
|
self.ecx.access_local(self.ecx.frame(), local, None).ok()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -643,7 +665,8 @@ impl CanConstProp {
|
||||||
// lint for x != y
|
// lint for x != y
|
||||||
// FIXME(oli-obk): lint variables until they are used in a condition
|
// FIXME(oli-obk): lint variables until they are used in a condition
|
||||||
// FIXME(oli-obk): lint if return value is constant
|
// FIXME(oli-obk): lint if return value is constant
|
||||||
*val = body.local_kind(local) == LocalKind::Temp;
|
let local_kind = body.local_kind(local);
|
||||||
|
*val = local_kind == LocalKind::Temp || local_kind == LocalKind::ReturnPointer;
|
||||||
|
|
||||||
if !*val {
|
if !*val {
|
||||||
trace!("local {:?} can't be propagated because it's not a temporary", local);
|
trace!("local {:?} can't be propagated because it's not a temporary", local);
|
||||||
|
@ -731,7 +754,9 @@ impl<'mir, 'tcx> MutVisitor<'tcx> for ConstPropagator<'mir, 'tcx> {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
trace!("can't propagate into {:?}", local);
|
trace!("can't propagate into {:?}", local);
|
||||||
self.remove_const(local);
|
if local != RETURN_PLACE {
|
||||||
|
self.remove_const(local);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
57
src/test/mir-opt/const_prop/return_place.rs
Normal file
57
src/test/mir-opt/const_prop/return_place.rs
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
// compile-flags: -C overflow-checks=on
|
||||||
|
|
||||||
|
fn add() -> u32 {
|
||||||
|
2 + 2
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
add();
|
||||||
|
}
|
||||||
|
|
||||||
|
// END RUST SOURCE
|
||||||
|
// START rustc.add.ConstProp.before.mir
|
||||||
|
// fn add() -> u32 {
|
||||||
|
// let mut _0: u32;
|
||||||
|
// let mut _1: (u32, bool);
|
||||||
|
// bb0: {
|
||||||
|
// _1 = CheckedAdd(const 2u32, const 2u32);
|
||||||
|
// assert(!move (_1.1: bool), "attempt to add with overflow") -> bb1;
|
||||||
|
// }
|
||||||
|
// bb1: {
|
||||||
|
// _0 = move (_1.0: u32);
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
// bb2 (cleanup): {
|
||||||
|
// resume;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// END rustc.add.ConstProp.before.mir
|
||||||
|
// START rustc.add.ConstProp.after.mir
|
||||||
|
// fn add() -> u32 {
|
||||||
|
// let mut _0: u32;
|
||||||
|
// let mut _1: (u32, bool);
|
||||||
|
// bb0: {
|
||||||
|
// _1 = (const 4u32, const false);
|
||||||
|
// assert(!const false, "attempt to add with overflow") -> bb1;
|
||||||
|
// }
|
||||||
|
// bb1: {
|
||||||
|
// _0 = const 4u32;
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
// bb2 (cleanup): {
|
||||||
|
// resume;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// END rustc.add.ConstProp.after.mir
|
||||||
|
// START rustc.add.PreCodegen.before.mir
|
||||||
|
// fn add() -> u32 {
|
||||||
|
// let mut _0: u32;
|
||||||
|
// let mut _1: (u32, bool);
|
||||||
|
// bb0: {
|
||||||
|
// (_1.0: u32) = const 4u32;
|
||||||
|
// (_1.1: bool) = const false;
|
||||||
|
// _0 = const 4u32;
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// END rustc.add.PreCodegen.before.mir
|
Loading…
Add table
Add a link
Reference in a new issue