// Copyright 2017 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! Propagates constants for early reporting of statically known //! assertion failures use rustc::mir::{Constant, Literal, Location, Place, Mir, Operand, Rvalue, Local}; use rustc::mir::{NullOp, StatementKind, Statement, BasicBlock, LocalKind}; use rustc::mir::{TerminatorKind, ClearCrossCrate, SourceInfo, BinOp}; use rustc::mir::visit::Visitor; use rustc::ty::layout::LayoutOf; use rustc::middle::const_val::ConstVal; use rustc::ty::{TyCtxt, self, Instance}; use rustc::mir::interpret::{Value, PrimVal, GlobalId}; use interpret::{eval_body_with_mir, eval_body, mk_borrowck_eval_cx, unary_op, ValTy}; use transform::{MirPass, MirSource}; use syntax::codemap::Span; use rustc::ty::subst::Substs; use rustc_data_structures::indexed_vec::IndexVec; pub struct ConstProp; impl MirPass for ConstProp { fn run_pass<'a, 'tcx>(&self, tcx: TyCtxt<'a, 'tcx, 'tcx>, source: MirSource, mir: &mut Mir<'tcx>) { trace!("ConstProp starting for {:?}", source.def_id); // FIXME(oli-obk, eddyb) Optimize locals (or even local paths) to hold // constants, instead of just checking for const-folding succeeding. // That would require an uniform one-def no-mutation analysis // and RPO (or recursing when needing the value of a local). let mut optimization_finder = OptimizationFinder::new(mir, tcx, source); optimization_finder.visit_mir(mir); trace!("ConstProp done for {:?}", source.def_id); } } type Const<'tcx> = (Value, ty::Ty<'tcx>, Span); /// Finds optimization opportunities on the MIR. struct OptimizationFinder<'b, 'a, 'tcx:'a+'b> { mir: &'b Mir<'tcx>, tcx: TyCtxt<'a, 'tcx, 'tcx>, source: MirSource, places: IndexVec>>, } impl<'b, 'a, 'tcx:'b> OptimizationFinder<'b, 'a, 'tcx> { fn new( mir: &'b Mir<'tcx>, tcx: TyCtxt<'a, 'tcx, 'tcx>, source: MirSource, ) -> OptimizationFinder<'b, 'a, 'tcx> { OptimizationFinder { mir, tcx, source, places: IndexVec::from_elem(None, &mir.local_decls), } } fn eval_constant(&mut self, c: &Constant<'tcx>) -> Option> { match c.literal { Literal::Value { value } => match value.val { ConstVal::Value(v) => Some((v, value.ty, c.span)), ConstVal::Unevaluated(did, substs) => { let param_env = self.tcx.param_env(self.source.def_id); let instance = Instance::resolve( self.tcx, param_env, did, substs, )?; let cid = GlobalId { instance, promoted: None, }; let (value, _, ty) = eval_body(self.tcx, cid, param_env)?; let val = (value, ty, c.span); trace!("evaluated {:?} to {:?}", c, val); Some(val) }, }, // evaluate the promoted and replace the constant with the evaluated result Literal::Promoted { index } => { let generics = self.tcx.generics_of(self.source.def_id); if generics.parent_types as usize + generics.types.len() > 0 { // FIXME: can't handle code with generics return None; } let substs = Substs::identity_for_item(self.tcx, self.source.def_id); let instance = Instance::new(self.source.def_id, substs); let cid = GlobalId { instance, promoted: Some(index), }; let param_env = self.tcx.param_env(self.source.def_id); let (value, _, ty) = eval_body_with_mir(self.tcx, cid, self.mir, param_env)?; let val = (value, ty, c.span); trace!("evaluated {:?} to {:?}", c, val); Some(val) } } } fn eval_operand(&mut self, op: &Operand<'tcx>) -> Option> { match *op { Operand::Constant(ref c) => self.eval_constant(c), Operand::Move(ref place) | Operand::Copy(ref place) => match *place { Place::Local(loc) => self.places[loc].clone(), // FIXME(oli-obk): field and index projections Place::Projection(_) => None, _ => None, }, } } fn const_prop( &mut self, rvalue: &Rvalue<'tcx>, place_ty: ty::Ty<'tcx>, source_info: SourceInfo, ) -> Option> { let span = source_info.span; match *rvalue { // No need to overwrite an already evaluated constant Rvalue::Use(Operand::Constant(box Constant { literal: Literal::Value { value: &ty::Const { val: ConstVal::Value(_), .. }, }, .. })) => None, // This branch exists for the sanity type check Rvalue::Use(Operand::Constant(ref c)) => { assert_eq!(c.ty, place_ty); self.eval_constant(c) }, Rvalue::Use(ref op) => { self.eval_operand(op) }, Rvalue::Repeat(..) | Rvalue::Ref(..) | Rvalue::Cast(..) | Rvalue::Aggregate(..) | Rvalue::NullaryOp(NullOp::Box, _) | Rvalue::Discriminant(..) => None, // FIXME(oli-obk): evaluate static/constant slice lengths Rvalue::Len(_) => None, Rvalue::NullaryOp(NullOp::SizeOf, ty) => { let param_env = self.tcx.param_env(self.source.def_id); type_size_of(self.tcx, param_env, ty).map(|n| ( Value::ByVal(PrimVal::Bytes(n as u128)), self.tcx.types.usize, span, )) } Rvalue::UnaryOp(op, ref arg) => { let def_id = if self.tcx.is_closure(self.source.def_id) { self.tcx.closure_base_def_id(self.source.def_id) } else { self.source.def_id }; let generics = self.tcx.generics_of(def_id); if generics.parent_types as usize + generics.types.len() > 0 { // FIXME: can't handle code with generics return None; } let substs = Substs::identity_for_item(self.tcx, self.source.def_id); let instance = Instance::new(self.source.def_id, substs); let ecx = mk_borrowck_eval_cx(self.tcx, instance, self.mir, span).unwrap(); let val = self.eval_operand(arg)?; let prim = ecx.value_to_primval(ValTy { value: val.0, ty: val.1 }).ok()?; let kind = ecx.ty_to_primval_kind(val.1).ok()?; match unary_op(op, prim, kind) { Ok(val) => Some((Value::ByVal(val), place_ty, span)), Err(mut err) => { ecx.report(&mut err, false, Some(span)); None }, } } Rvalue::CheckedBinaryOp(op, ref left, ref right) | Rvalue::BinaryOp(op, ref left, ref right) => { trace!("rvalue binop {:?} for {:?} and {:?}", op, left, right); let right = self.eval_operand(right)?; let def_id = if self.tcx.is_closure(self.source.def_id) { self.tcx.closure_base_def_id(self.source.def_id) } else { self.source.def_id }; let generics = self.tcx.generics_of(def_id); let has_generics = generics.parent_types as usize + generics.types.len() > 0; if has_generics { // FIXME: can't handle code with generics return None; } let substs = Substs::identity_for_item(self.tcx, self.source.def_id); let instance = Instance::new(self.source.def_id, substs); let ecx = mk_borrowck_eval_cx(self.tcx, instance, self.mir, span).unwrap(); let r = ecx.value_to_primval(ValTy { value: right.0, ty: right.1 }).ok()?; if op == BinOp::Shr || op == BinOp::Shl { let param_env = self.tcx.param_env(self.source.def_id); let bits = (self.tcx, param_env).layout_of(place_ty).unwrap().size.bits(); if r.to_bytes().ok().map_or(false, |b| b >= bits as u128) { let scope_info = match self.mir.visibility_scope_info { ClearCrossCrate::Set(ref data) => data, ClearCrossCrate::Clear => return None, }; let node_id = scope_info[source_info.scope].lint_root; self.tcx.lint_node( ::rustc::lint::builtin::EXCEEDING_BITSHIFTS, node_id, span, "bitshift exceeds the type's number of bits"); return None; } } let left = self.eval_operand(left)?; let l = ecx.value_to_primval(ValTy { value: left.0, ty: left.1 }).ok()?; trace!("const evaluating {:?} for {:?} and {:?}", op, left, right); match ecx.binary_op(op, l, left.1, r, right.1) { Ok((val, overflow)) => { let val = if let Rvalue::CheckedBinaryOp(..) = *rvalue { Value::ByValPair( val, PrimVal::from_bool(overflow), ) } else { if overflow { use rustc::mir::interpret::EvalErrorKind; let mut err = EvalErrorKind::OverflowingMath.into(); ecx.report(&mut err, false, Some(span)); return None; } Value::ByVal(val) }; Some((val, place_ty, span)) }, Err(mut err) => { ecx.report(&mut err, false, Some(span)); None }, } }, } } } fn type_size_of<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>, param_env: ty::ParamEnv<'tcx>, ty: ty::Ty<'tcx>) -> Option { use rustc::ty::layout::LayoutOf; (tcx, param_env).layout_of(ty).ok().map(|layout| layout.size.bytes()) } struct CanConstProp { local: Local, can_const_prop: bool, // false at the beginning, once set, there are not allowed to be any more assignments found_assignment: bool, } impl CanConstProp { /// returns true if `local` can be propagated fn check<'tcx>(local: Local, mir: &Mir<'tcx>) -> bool { let mut cpv = CanConstProp { local, can_const_prop: true, found_assignment: false, }; cpv.visit_mir(mir); cpv.can_const_prop } fn is_our_local(&mut self, mut place: &Place) -> bool { while let Place::Projection(ref proj) = place { place = &proj.base; } if let Place::Local(local) = *place { local == self.local } else { false } } } impl<'tcx> Visitor<'tcx> for CanConstProp { fn visit_statement( &mut self, block: BasicBlock, statement: &Statement<'tcx>, location: Location, ) { self.super_statement(block, statement, location); match statement.kind { StatementKind::SetDiscriminant { ref place, .. } | StatementKind::Assign(ref place, _) => { if self.is_our_local(place) { if self.found_assignment { self.can_const_prop = false; } else { self.found_assignment = true } } }, StatementKind::InlineAsm { ref outputs, .. } => { for place in outputs { if self.is_our_local(place) { if self.found_assignment { self.can_const_prop = false; } else { self.found_assignment = true } return; } } } _ => {} } } fn visit_rvalue(&mut self, rvalue: &Rvalue<'tcx>, location: Location) { self.super_rvalue(rvalue, location); if let Rvalue::Ref(_, _, ref place) = *rvalue { if self.is_our_local(place) { self.can_const_prop = false; } } } } impl<'b, 'a, 'tcx> Visitor<'tcx> for OptimizationFinder<'b, 'a, 'tcx> { fn visit_constant( &mut self, constant: &Constant<'tcx>, location: Location, ) { trace!("visit_constant: {:?}", constant); self.super_constant(constant, location); self.eval_constant(constant); } fn visit_statement( &mut self, block: BasicBlock, statement: &Statement<'tcx>, location: Location, ) { trace!("visit_statement: {:?}", statement); if let StatementKind::Assign(ref place, ref rval) = statement.kind { let place_ty = place .ty(&self.mir.local_decls, self.tcx) .to_ty(self.tcx); if let Some(value) = self.const_prop(rval, place_ty, statement.source_info) { if let Place::Local(local) = *place { if self.mir.local_kind(local) == LocalKind::Temp && CanConstProp::check(local, self.mir) { trace!("storing {:?} to {:?}", value, local); assert!(self.places[local].is_none()); self.places[local] = Some(value); } } } } self.super_statement(block, statement, location); } fn visit_terminator_kind( &mut self, block: BasicBlock, kind: &TerminatorKind<'tcx>, _location: Location, ) { match kind { TerminatorKind::SwitchInt { discr: value, .. } | TerminatorKind::Yield { value, .. } | TerminatorKind::Assert { cond: value, .. } => { match value { Operand::Constant(box Constant { literal: Literal::Value { value: &ty::Const { val: ConstVal::Value(_), .. }, }, .. }) => return, _ => {}, } if let Some(value) = self.eval_operand(value) { if let TerminatorKind::Assert { expected, msg, .. } = kind { if Value::ByVal(PrimVal::from_bool(*expected)) != value.0 { let span = self.mir[block] .terminator .as_ref() .unwrap() .source_info .span; let node_id = self .tcx .hir .as_local_node_id(self.source.def_id) .expect("some part of a failing const eval must be local"); let mut lint = self.tcx.struct_span_lint_node( ::rustc::lint::builtin::CONST_ERR, node_id, span, "constant evaluation error", ); use rustc::mir::AssertMessage::*; match msg { GeneratorResumedAfterReturn => lint.span_label(span, "generator resumed after completion"), GeneratorResumedAfterPanic => lint.span_label(span, "generator resumed after panicking"), Math(ref err) => lint.span_label(span, err.description()), BoundsCheck { ref len, ref index } => { let len = self.eval_operand(len).expect("len must be const"); let len = match len.0 { Value::ByVal(PrimVal::Bytes(n)) => n, _ => bug!("const len not primitive: {:?}", len), }; let index = self .eval_operand(index) .expect("index must be const"); let index = match index.0 { Value::ByVal(PrimVal::Bytes(n)) => n, _ => bug!("const index not primitive: {:?}", index), }; lint.span_label( span, format!( "index out of bounds: \ the len is {} but the index is {}", len, index, ), ) }, }.emit(); } } } } // FIXME: do this optimization for function calls _ => {}, } } }