2015-10-21 16:42:25 -05:00
|
|
|
|
// Copyright 2012-2014 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 <LICENSE-APACHE or
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
2016-03-31 00:50:07 -05:00
|
|
|
|
use llvm::{self, ValueRef};
|
2016-03-22 10:30:57 -05:00
|
|
|
|
use rustc::ty::{self, Ty};
|
|
|
|
|
use rustc::ty::cast::{CastTy, IntTy};
|
2015-11-19 09:37:34 -06:00
|
|
|
|
use rustc::mir::repr as mir;
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
2016-03-22 12:23:36 -05:00
|
|
|
|
use asm;
|
|
|
|
|
use base;
|
|
|
|
|
use callee::Callee;
|
2016-05-26 12:02:56 -05:00
|
|
|
|
use common::{self, val_ty, C_bool, C_null, C_uint, BlockAndBuilder, Result};
|
2016-03-22 12:23:36 -05:00
|
|
|
|
use datum::{Datum, Lvalue};
|
|
|
|
|
use debuginfo::DebugLoc;
|
|
|
|
|
use adt;
|
|
|
|
|
use machine;
|
|
|
|
|
use type_of;
|
|
|
|
|
use tvec;
|
|
|
|
|
use value::Value;
|
|
|
|
|
use Disr;
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
|
|
|
|
use super::MirContext;
|
2016-06-05 06:38:29 -05:00
|
|
|
|
use super::constant::const_scalar_checked_binop;
|
2015-11-10 14:05:11 -06:00
|
|
|
|
use super::operand::{OperandRef, OperandValue};
|
2016-03-11 04:54:59 -06:00
|
|
|
|
use super::lvalue::{LvalueRef, get_dataptr};
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
|
|
|
|
impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
|
|
|
|
|
pub fn trans_rvalue(&mut self,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx: BlockAndBuilder<'bcx, 'tcx>,
|
2015-12-24 17:02:34 -06:00
|
|
|
|
dest: LvalueRef<'tcx>,
|
2016-04-07 14:35:11 -05:00
|
|
|
|
rvalue: &mir::Rvalue<'tcx>,
|
|
|
|
|
debug_loc: DebugLoc)
|
2016-02-01 04:04:46 -06:00
|
|
|
|
-> BlockAndBuilder<'bcx, 'tcx>
|
2015-10-21 16:42:25 -05:00
|
|
|
|
{
|
2016-02-18 11:49:45 -06:00
|
|
|
|
debug!("trans_rvalue(dest.llval={:?}, rvalue={:?})",
|
|
|
|
|
Value(dest.llval), rvalue);
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
|
|
|
|
match *rvalue {
|
2016-02-04 11:40:28 -06:00
|
|
|
|
mir::Rvalue::Use(ref operand) => {
|
2016-02-26 06:12:28 -06:00
|
|
|
|
let tr_operand = self.trans_operand(&bcx, operand);
|
2016-02-04 11:40:28 -06:00
|
|
|
|
// FIXME: consider not copying constants through stack. (fixable by translating
|
|
|
|
|
// constants into OperandValue::Ref, why don’t we do that yet if we don’t?)
|
|
|
|
|
self.store_operand(&bcx, dest.llval, tr_operand);
|
|
|
|
|
bcx
|
|
|
|
|
}
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
2016-03-08 07:03:02 -06:00
|
|
|
|
mir::Rvalue::Cast(mir::CastKind::Unsize, ref source, cast_ty) => {
|
2016-05-07 12:00:42 -05:00
|
|
|
|
let cast_ty = bcx.monomorphize(&cast_ty);
|
|
|
|
|
|
2015-11-11 14:02:51 -06:00
|
|
|
|
if common::type_is_fat_ptr(bcx.tcx(), cast_ty) {
|
2015-11-12 16:12:50 -06:00
|
|
|
|
// into-coerce of a thin pointer to a fat pointer - just
|
|
|
|
|
// use the operand path.
|
2016-04-07 14:35:11 -05:00
|
|
|
|
let (bcx, temp) = self.trans_rvalue_operand(bcx, rvalue, debug_loc);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
self.store_operand(&bcx, dest.llval, temp);
|
2015-11-08 18:16:19 -06:00
|
|
|
|
return bcx;
|
|
|
|
|
}
|
|
|
|
|
|
2015-11-11 14:02:51 -06:00
|
|
|
|
// Unsize of a nontrivial struct. I would prefer for
|
|
|
|
|
// this to be eliminated by MIR translation, but
|
|
|
|
|
// `CoerceUnsized` can be passed by a where-clause,
|
|
|
|
|
// so the (generic) MIR may not be able to expand it.
|
2016-03-08 07:03:02 -06:00
|
|
|
|
let operand = self.trans_operand(&bcx, source);
|
2016-05-25 03:55:44 -05:00
|
|
|
|
let operand = operand.pack_if_pair(&bcx);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
match operand.val {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(..) => bug!(),
|
2016-02-01 04:04:46 -06:00
|
|
|
|
OperandValue::Immediate(llval) => {
|
|
|
|
|
// unsize from an immediate structure. We don't
|
|
|
|
|
// really need a temporary alloca here, but
|
|
|
|
|
// avoiding it would require us to have
|
|
|
|
|
// `coerce_unsized_into` use extractvalue to
|
|
|
|
|
// index into the struct, and this case isn't
|
|
|
|
|
// important enough for it.
|
|
|
|
|
debug!("trans_rvalue: creating ugly alloca");
|
|
|
|
|
let lltemp = base::alloc_ty(bcx, operand.ty, "__unsize_temp");
|
|
|
|
|
base::store_ty(bcx, llval, lltemp, operand.ty);
|
|
|
|
|
base::coerce_unsized_into(bcx,
|
|
|
|
|
lltemp, operand.ty,
|
|
|
|
|
dest.llval, cast_ty);
|
|
|
|
|
}
|
|
|
|
|
OperandValue::Ref(llref) => {
|
|
|
|
|
base::coerce_unsized_into(bcx,
|
|
|
|
|
llref, operand.ty,
|
|
|
|
|
dest.llval, cast_ty);
|
|
|
|
|
}
|
2015-11-11 14:02:51 -06:00
|
|
|
|
}
|
2016-02-01 04:04:46 -06:00
|
|
|
|
});
|
2015-11-11 14:02:51 -06:00
|
|
|
|
bcx
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
2015-11-05 08:50:36 -06:00
|
|
|
|
mir::Rvalue::Repeat(ref elem, ref count) => {
|
2016-02-04 11:40:28 -06:00
|
|
|
|
let tr_elem = self.trans_operand(&bcx, elem);
|
2016-04-21 08:15:56 -05:00
|
|
|
|
let size = count.value.as_u64(bcx.tcx().sess.target.uint_type);
|
|
|
|
|
let size = C_uint(bcx.ccx(), size);
|
2016-03-09 06:20:22 -06:00
|
|
|
|
let base = get_dataptr(&bcx, dest.llval);
|
2016-02-04 11:40:28 -06:00
|
|
|
|
let bcx = bcx.map_block(|block| {
|
|
|
|
|
tvec::iter_vec_raw(block, base, tr_elem.ty, size, |block, llslot, _| {
|
|
|
|
|
self.store_operand_direct(block, llslot, tr_elem);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
block
|
|
|
|
|
})
|
2016-02-04 11:40:28 -06:00
|
|
|
|
});
|
|
|
|
|
bcx
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
2015-12-24 17:02:34 -06:00
|
|
|
|
mir::Rvalue::Aggregate(ref kind, ref operands) => {
|
|
|
|
|
match *kind {
|
2016-01-13 18:55:46 -06:00
|
|
|
|
mir::AggregateKind::Adt(adt_def, index, _) => {
|
2015-12-24 17:02:34 -06:00
|
|
|
|
let repr = adt::represent_type(bcx.ccx(), dest.ty.to_ty(bcx.tcx()));
|
2016-01-16 09:03:09 -06:00
|
|
|
|
let disr = Disr::from(adt_def.variants[index].disr_val);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.with_block(|bcx| {
|
2016-02-09 14:24:11 -06:00
|
|
|
|
adt::trans_set_discr(bcx, &repr, dest.llval, Disr::from(disr));
|
2016-02-01 04:04:46 -06:00
|
|
|
|
});
|
2016-01-13 18:55:46 -06:00
|
|
|
|
for (i, operand) in operands.iter().enumerate() {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let op = self.trans_operand(&bcx, operand);
|
2016-01-13 18:55:46 -06:00
|
|
|
|
// Do not generate stores and GEPis for zero-sized fields.
|
|
|
|
|
if !common::type_is_zero_size(bcx.ccx(), op.ty) {
|
|
|
|
|
let val = adt::MaybeSizedValue::sized(dest.llval);
|
2016-03-09 06:20:22 -06:00
|
|
|
|
let lldest_i = adt::trans_field_ptr_builder(&bcx, &repr,
|
|
|
|
|
val, disr, i);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
self.store_operand(&bcx, lldest_i, op);
|
2016-01-13 18:55:46 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-12-24 17:02:34 -06:00
|
|
|
|
},
|
|
|
|
|
_ => {
|
2016-03-08 06:38:44 -06:00
|
|
|
|
// FIXME Shouldn't need to manually trigger closure instantiations.
|
|
|
|
|
if let mir::AggregateKind::Closure(def_id, substs) = *kind {
|
2016-03-22 12:23:36 -05:00
|
|
|
|
use closure;
|
2016-03-08 06:38:44 -06:00
|
|
|
|
|
2016-07-13 16:03:02 -05:00
|
|
|
|
closure::trans_closure_body_via_mir(bcx.ccx(),
|
|
|
|
|
def_id,
|
|
|
|
|
bcx.monomorphize(&substs));
|
2016-03-08 06:38:44 -06:00
|
|
|
|
}
|
|
|
|
|
|
2015-12-24 17:02:34 -06:00
|
|
|
|
for (i, operand) in operands.iter().enumerate() {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let op = self.trans_operand(&bcx, operand);
|
2016-01-12 08:36:47 -06:00
|
|
|
|
// Do not generate stores and GEPis for zero-sized fields.
|
|
|
|
|
if !common::type_is_zero_size(bcx.ccx(), op.ty) {
|
|
|
|
|
// Note: perhaps this should be StructGep, but
|
|
|
|
|
// note that in some cases the values here will
|
|
|
|
|
// not be structs but arrays.
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let dest = bcx.gepi(dest.llval, &[0, i]);
|
|
|
|
|
self.store_operand(&bcx, dest, op);
|
2016-01-12 08:36:47 -06:00
|
|
|
|
}
|
2015-12-24 17:02:34 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-11-02 08:39:59 -06:00
|
|
|
|
}
|
|
|
|
|
bcx
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-09 14:17:02 -06:00
|
|
|
|
mir::Rvalue::InlineAsm { ref asm, ref outputs, ref inputs } => {
|
|
|
|
|
let outputs = outputs.iter().map(|output| {
|
|
|
|
|
let lvalue = self.trans_lvalue(&bcx, output);
|
|
|
|
|
Datum::new(lvalue.llval, lvalue.ty.to_ty(bcx.tcx()),
|
|
|
|
|
Lvalue::new("out"))
|
|
|
|
|
}).collect();
|
|
|
|
|
|
|
|
|
|
let input_vals = inputs.iter().map(|input| {
|
|
|
|
|
self.trans_operand(&bcx, input).immediate()
|
|
|
|
|
}).collect();
|
|
|
|
|
|
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
asm::trans_inline_asm(bcx, asm, outputs, input_vals);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
bcx
|
2015-11-02 08:39:59 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_ => {
|
2016-04-16 00:38:18 -05:00
|
|
|
|
assert!(rvalue_creates_operand(&self.mir, &bcx, rvalue));
|
2016-04-07 14:35:11 -05:00
|
|
|
|
let (bcx, temp) = self.trans_rvalue_operand(bcx, rvalue, debug_loc);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
self.store_operand(&bcx, dest.llval, temp);
|
2015-11-02 08:39:59 -06:00
|
|
|
|
bcx
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn trans_rvalue_operand(&mut self,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx: BlockAndBuilder<'bcx, 'tcx>,
|
2016-04-07 14:35:11 -05:00
|
|
|
|
rvalue: &mir::Rvalue<'tcx>,
|
|
|
|
|
debug_loc: DebugLoc)
|
2016-02-01 04:04:46 -06:00
|
|
|
|
-> (BlockAndBuilder<'bcx, 'tcx>, OperandRef<'tcx>)
|
2015-11-02 08:39:59 -06:00
|
|
|
|
{
|
2016-04-16 00:38:18 -05:00
|
|
|
|
assert!(rvalue_creates_operand(&self.mir, &bcx, rvalue),
|
|
|
|
|
"cannot trans {:?} to operand", rvalue);
|
2015-11-02 08:39:59 -06:00
|
|
|
|
|
|
|
|
|
match *rvalue {
|
2016-03-08 07:03:02 -06:00
|
|
|
|
mir::Rvalue::Cast(ref kind, ref source, cast_ty) => {
|
|
|
|
|
let operand = self.trans_operand(&bcx, source);
|
2016-02-18 11:49:45 -06:00
|
|
|
|
debug!("cast operand is {:?}", operand);
|
2015-11-11 14:02:51 -06:00
|
|
|
|
let cast_ty = bcx.monomorphize(&cast_ty);
|
|
|
|
|
|
|
|
|
|
let val = match *kind {
|
2016-03-06 09:32:47 -06:00
|
|
|
|
mir::CastKind::ReifyFnPointer => {
|
|
|
|
|
match operand.ty.sty {
|
|
|
|
|
ty::TyFnDef(def_id, substs, _) => {
|
|
|
|
|
OperandValue::Immediate(
|
2016-02-23 13:21:50 -06:00
|
|
|
|
Callee::def(bcx.ccx(), def_id, substs)
|
2016-03-06 09:32:47 -06:00
|
|
|
|
.reify(bcx.ccx()).val)
|
|
|
|
|
}
|
|
|
|
|
_ => {
|
2016-03-28 18:46:02 -05:00
|
|
|
|
bug!("{} cannot be reified to a fn ptr", operand.ty)
|
2016-03-06 09:32:47 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-11-11 14:02:51 -06:00
|
|
|
|
mir::CastKind::UnsafeFnPointer => {
|
2016-03-06 09:32:47 -06:00
|
|
|
|
// this is a no-op at the LLVM level
|
2015-11-11 14:02:51 -06:00
|
|
|
|
operand.val
|
|
|
|
|
}
|
|
|
|
|
mir::CastKind::Unsize => {
|
|
|
|
|
// unsize targets other than to a fat pointer currently
|
|
|
|
|
// can't be operands.
|
|
|
|
|
assert!(common::type_is_fat_ptr(bcx.tcx(), cast_ty));
|
|
|
|
|
|
|
|
|
|
match operand.val {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(lldata, llextra) => {
|
2015-11-11 14:02:51 -06:00
|
|
|
|
// unsize from a fat pointer - this is a
|
|
|
|
|
// "trait-object-to-supertrait" coercion, for
|
|
|
|
|
// example,
|
|
|
|
|
// &'a fmt::Debug+Send => &'a fmt::Debug,
|
2016-05-18 21:25:03 -05:00
|
|
|
|
// So we need to pointercast the base to ensure
|
|
|
|
|
// the types match up.
|
|
|
|
|
let llcast_ty = type_of::fat_ptr_base_ty(bcx.ccx(), cast_ty);
|
|
|
|
|
let lldata = bcx.pointercast(lldata, llcast_ty);
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(lldata, llextra)
|
2015-11-11 14:02:51 -06:00
|
|
|
|
}
|
2015-11-12 16:12:50 -06:00
|
|
|
|
OperandValue::Immediate(lldata) => {
|
2015-11-11 14:02:51 -06:00
|
|
|
|
// "standard" unsize
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let (lldata, llextra) = bcx.with_block(|bcx| {
|
2015-11-11 14:02:51 -06:00
|
|
|
|
base::unsize_thin_ptr(bcx, lldata,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
operand.ty, cast_ty)
|
|
|
|
|
});
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(lldata, llextra)
|
2015-11-11 14:02:51 -06:00
|
|
|
|
}
|
|
|
|
|
OperandValue::Ref(_) => {
|
2016-03-28 18:46:02 -05:00
|
|
|
|
bug!("by-ref operand {:?} in trans_rvalue_operand",
|
|
|
|
|
operand);
|
2015-11-11 14:02:51 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-11-05 05:33:06 -06:00
|
|
|
|
mir::CastKind::Misc if common::type_is_immediate(bcx.ccx(), operand.ty) => {
|
|
|
|
|
debug_assert!(common::type_is_immediate(bcx.ccx(), cast_ty));
|
|
|
|
|
let r_t_in = CastTy::from_ty(operand.ty).expect("bad input type for cast");
|
|
|
|
|
let r_t_out = CastTy::from_ty(cast_ty).expect("bad output type for cast");
|
2016-03-06 08:30:21 -06:00
|
|
|
|
let ll_t_in = type_of::immediate_type_of(bcx.ccx(), operand.ty);
|
|
|
|
|
let ll_t_out = type_of::immediate_type_of(bcx.ccx(), cast_ty);
|
2016-03-08 06:21:11 -06:00
|
|
|
|
let llval = operand.immediate();
|
|
|
|
|
let signed = if let CastTy::Int(IntTy::CEnum) = r_t_in {
|
2015-11-05 05:33:06 -06:00
|
|
|
|
let repr = adt::represent_type(bcx.ccx(), operand.ty);
|
2016-03-08 06:21:11 -06:00
|
|
|
|
adt::is_discr_signed(&repr)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else {
|
2016-03-08 06:21:11 -06:00
|
|
|
|
operand.ty.is_signed()
|
2015-11-05 05:33:06 -06:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let newval = match (r_t_in, r_t_out) {
|
|
|
|
|
(CastTy::Int(_), CastTy::Int(_)) => {
|
|
|
|
|
let srcsz = ll_t_in.int_width();
|
|
|
|
|
let dstsz = ll_t_out.int_width();
|
|
|
|
|
if srcsz == dstsz {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.bitcast(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else if srcsz > dstsz {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.trunc(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else if signed {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.sext(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.zext(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
(CastTy::Float, CastTy::Float) => {
|
|
|
|
|
let srcsz = ll_t_in.float_width();
|
|
|
|
|
let dstsz = ll_t_out.float_width();
|
|
|
|
|
if dstsz > srcsz {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fpext(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else if srcsz > dstsz {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fptrunc(llval, ll_t_out)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else {
|
|
|
|
|
llval
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
(CastTy::Ptr(_), CastTy::Ptr(_)) |
|
|
|
|
|
(CastTy::FnPtr, CastTy::Ptr(_)) |
|
|
|
|
|
(CastTy::RPtr(_), CastTy::Ptr(_)) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.pointercast(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Ptr(_), CastTy::Int(_)) |
|
|
|
|
|
(CastTy::FnPtr, CastTy::Int(_)) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.ptrtoint(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Int(_), CastTy::Ptr(_)) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.inttoptr(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Int(_), CastTy::Float) if signed =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.sitofp(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Int(_), CastTy::Float) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.uitofp(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Float, CastTy::Int(IntTy::I)) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fptosi(llval, ll_t_out),
|
2015-11-05 05:33:06 -06:00
|
|
|
|
(CastTy::Float, CastTy::Int(_)) =>
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fptoui(llval, ll_t_out),
|
2016-03-28 18:46:02 -05:00
|
|
|
|
_ => bug!("unsupported cast: {:?} to {:?}", operand.ty, cast_ty)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
};
|
|
|
|
|
OperandValue::Immediate(newval)
|
|
|
|
|
}
|
|
|
|
|
mir::CastKind::Misc => { // Casts from a fat-ptr.
|
2016-03-06 08:30:21 -06:00
|
|
|
|
let ll_cast_ty = type_of::immediate_type_of(bcx.ccx(), cast_ty);
|
|
|
|
|
let ll_from_ty = type_of::immediate_type_of(bcx.ccx(), operand.ty);
|
2016-05-25 03:55:44 -05:00
|
|
|
|
if let OperandValue::Pair(data_ptr, meta_ptr) = operand.val {
|
2015-11-05 05:33:06 -06:00
|
|
|
|
if common::type_is_fat_ptr(bcx.tcx(), cast_ty) {
|
|
|
|
|
let ll_cft = ll_cast_ty.field_types();
|
|
|
|
|
let ll_fft = ll_from_ty.field_types();
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let data_cast = bcx.pointercast(data_ptr, ll_cft[0]);
|
2015-11-05 05:33:06 -06:00
|
|
|
|
assert_eq!(ll_cft[1].kind(), ll_fft[1].kind());
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(data_cast, meta_ptr)
|
2015-11-05 05:33:06 -06:00
|
|
|
|
} else { // cast to thin-ptr
|
|
|
|
|
// Cast of fat-ptr to thin-ptr is an extraction of data-ptr and
|
|
|
|
|
// pointer-cast of that pointer to desired pointer type.
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let llval = bcx.pointercast(data_ptr, ll_cast_ty);
|
2015-11-05 05:33:06 -06:00
|
|
|
|
OperandValue::Immediate(llval)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
bug!("Unexpected non-Pair operand")
|
2015-11-05 05:33:06 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-11-11 14:02:51 -06:00
|
|
|
|
};
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let operand = OperandRef {
|
2015-11-11 14:02:51 -06:00
|
|
|
|
val: val,
|
|
|
|
|
ty: cast_ty
|
2016-02-01 04:04:46 -06:00
|
|
|
|
};
|
|
|
|
|
(bcx, operand)
|
2015-11-02 08:39:59 -06:00
|
|
|
|
}
|
|
|
|
|
|
2015-11-11 14:02:51 -06:00
|
|
|
|
mir::Rvalue::Ref(_, bk, ref lvalue) => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let tr_lvalue = self.trans_lvalue(&bcx, lvalue);
|
2015-11-02 08:39:59 -06:00
|
|
|
|
|
2015-11-10 14:05:11 -06:00
|
|
|
|
let ty = tr_lvalue.ty.to_ty(bcx.tcx());
|
2015-11-11 14:02:51 -06:00
|
|
|
|
let ref_ty = bcx.tcx().mk_ref(
|
2016-06-01 07:10:44 -05:00
|
|
|
|
bcx.tcx().mk_region(ty::ReErased),
|
2015-11-11 14:02:51 -06:00
|
|
|
|
ty::TypeAndMut { ty: ty, mutbl: bk.to_mutbl_lossy() }
|
|
|
|
|
);
|
|
|
|
|
|
2015-10-21 16:42:25 -05:00
|
|
|
|
// Note: lvalues are indirect, so storing the `llval` into the
|
|
|
|
|
// destination effectively creates a reference.
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let operand = if common::type_is_sized(bcx.tcx(), ty) {
|
|
|
|
|
OperandRef {
|
2015-11-12 16:12:50 -06:00
|
|
|
|
val: OperandValue::Immediate(tr_lvalue.llval),
|
2015-11-11 14:02:51 -06:00
|
|
|
|
ty: ref_ty,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
}
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
OperandRef {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
val: OperandValue::Pair(tr_lvalue.llval,
|
|
|
|
|
tr_lvalue.llextra),
|
2015-11-11 14:02:51 -06:00
|
|
|
|
ty: ref_ty,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
(bcx, operand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mir::Rvalue::Len(ref lvalue) => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let tr_lvalue = self.trans_lvalue(&bcx, lvalue);
|
|
|
|
|
let operand = OperandRef {
|
2016-04-21 08:15:56 -05:00
|
|
|
|
val: OperandValue::Immediate(tr_lvalue.len(bcx.ccx())),
|
2015-11-02 08:39:59 -06:00
|
|
|
|
ty: bcx.tcx().types.usize,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
};
|
|
|
|
|
(bcx, operand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mir::Rvalue::BinaryOp(op, ref lhs, ref rhs) => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let lhs = self.trans_operand(&bcx, lhs);
|
|
|
|
|
let rhs = self.trans_operand(&bcx, rhs);
|
2015-11-10 14:05:11 -06:00
|
|
|
|
let llresult = if common::type_is_fat_ptr(bcx.tcx(), lhs.ty) {
|
|
|
|
|
match (lhs.val, rhs.val) {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
(OperandValue::Pair(lhs_addr, lhs_extra),
|
|
|
|
|
OperandValue::Pair(rhs_addr, rhs_extra)) => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
base::compare_fat_ptrs(bcx,
|
|
|
|
|
lhs_addr, lhs_extra,
|
|
|
|
|
rhs_addr, rhs_extra,
|
|
|
|
|
lhs.ty, op.to_hir_binop(),
|
2016-04-07 14:35:11 -05:00
|
|
|
|
debug_loc)
|
2016-02-01 04:04:46 -06:00
|
|
|
|
})
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
2016-03-28 18:46:02 -05:00
|
|
|
|
_ => bug!()
|
2015-11-10 14:05:11 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
self.trans_scalar_binop(&bcx, op,
|
2015-11-10 14:05:11 -06:00
|
|
|
|
lhs.immediate(), rhs.immediate(),
|
2016-02-01 04:04:46 -06:00
|
|
|
|
lhs.ty)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
};
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let operand = OperandRef {
|
2015-11-12 16:12:50 -06:00
|
|
|
|
val: OperandValue::Immediate(llresult),
|
|
|
|
|
ty: self.mir.binop_ty(bcx.tcx(), op, lhs.ty, rhs.ty),
|
2016-02-01 04:04:46 -06:00
|
|
|
|
};
|
|
|
|
|
(bcx, operand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
2016-03-31 00:50:07 -05:00
|
|
|
|
mir::Rvalue::CheckedBinaryOp(op, ref lhs, ref rhs) => {
|
|
|
|
|
let lhs = self.trans_operand(&bcx, lhs);
|
|
|
|
|
let rhs = self.trans_operand(&bcx, rhs);
|
|
|
|
|
let result = self.trans_scalar_checked_binop(&bcx, op,
|
|
|
|
|
lhs.immediate(), rhs.immediate(),
|
|
|
|
|
lhs.ty);
|
|
|
|
|
let val_ty = self.mir.binop_ty(bcx.tcx(), op, lhs.ty, rhs.ty);
|
|
|
|
|
let operand_ty = bcx.tcx().mk_tup(vec![val_ty, bcx.tcx().types.bool]);
|
|
|
|
|
let operand = OperandRef {
|
2016-05-25 03:55:44 -05:00
|
|
|
|
val: result,
|
2016-03-31 00:50:07 -05:00
|
|
|
|
ty: operand_ty
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
(bcx, operand)
|
|
|
|
|
}
|
2015-10-21 16:42:25 -05:00
|
|
|
|
|
|
|
|
|
mir::Rvalue::UnaryOp(op, ref operand) => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let operand = self.trans_operand(&bcx, operand);
|
2015-11-10 14:05:11 -06:00
|
|
|
|
let lloperand = operand.immediate();
|
2015-10-21 16:42:25 -05:00
|
|
|
|
let is_float = operand.ty.is_fp();
|
|
|
|
|
let llval = match op {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
mir::UnOp::Not => bcx.not(lloperand),
|
2015-10-21 16:42:25 -05:00
|
|
|
|
mir::UnOp::Neg => if is_float {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fneg(lloperand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.neg(lloperand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
};
|
2015-11-02 08:39:59 -06:00
|
|
|
|
(bcx, OperandRef {
|
2015-11-12 16:12:50 -06:00
|
|
|
|
val: OperandValue::Immediate(llval),
|
2015-11-02 08:39:59 -06:00
|
|
|
|
ty: operand.ty,
|
|
|
|
|
})
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mir::Rvalue::Box(content_ty) => {
|
2015-11-02 08:39:59 -06:00
|
|
|
|
let content_ty: Ty<'tcx> = bcx.monomorphize(&content_ty);
|
2015-10-21 16:42:25 -05:00
|
|
|
|
let llty = type_of::type_of(bcx.ccx(), content_ty);
|
|
|
|
|
let llsize = machine::llsize_of(bcx.ccx(), llty);
|
|
|
|
|
let align = type_of::align_of(bcx.ccx(), content_ty);
|
2016-03-09 13:49:23 -06:00
|
|
|
|
let llalign = C_uint(bcx.ccx(), align);
|
2015-10-21 16:42:25 -05:00
|
|
|
|
let llty_ptr = llty.ptr_to();
|
|
|
|
|
let box_ty = bcx.tcx().mk_box(content_ty);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
let mut llval = None;
|
|
|
|
|
let bcx = bcx.map_block(|bcx| {
|
|
|
|
|
let Result { bcx, val } = base::malloc_raw_dyn(bcx,
|
|
|
|
|
llty_ptr,
|
|
|
|
|
box_ty,
|
|
|
|
|
llsize,
|
|
|
|
|
llalign,
|
2016-04-07 14:35:11 -05:00
|
|
|
|
debug_loc);
|
2016-02-01 04:04:46 -06:00
|
|
|
|
llval = Some(val);
|
|
|
|
|
bcx
|
|
|
|
|
});
|
|
|
|
|
let operand = OperandRef {
|
|
|
|
|
val: OperandValue::Immediate(llval.unwrap()),
|
2015-11-02 08:39:59 -06:00
|
|
|
|
ty: box_ty,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
};
|
|
|
|
|
(bcx, operand)
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
|
Various improvements to MIR and LLVM IR Construction
Primarily affects the MIR construction, which indirectly improves LLVM
IR generation, but some LLVM IR changes have been made too.
* Handle "statement expressions" more intelligently. These are
expressions that always evaluate to `()`. Previously a temporary would
be generated as a destination to translate into, which is unnecessary.
This affects assignment, augmented assignment, `return`, `break` and
`continue`.
* Avoid inserting drops for non-drop types in more places. Scheduled
drops were already skipped for types that we knew wouldn't need
dropping at construction time. However manually-inserted drops like
those for `x` in `x = y;` were still generated. `build_drop` now takes
a type parameter like its `schedule_drop` counterpart and checks to
see if the type needs dropping.
* Avoid generating an extra temporary for an assignment where the types
involved don't need dropping. Previously an expression like
`a = b + 1;` would result in a temporary for `b + 1`. This is so the
RHS can be evaluated, then the LHS evaluated and dropped and have
everything work correctly. However, this isn't necessary if the `LHS`
doesn't need a drop, as we can just overwrite the existing value.
* Improves lvalue analysis to allow treating an `Rvalue::Use` as an
operand in certain conditions. The reason for it never being an
operand is so it can be zeroed/drop-filled, but this is only true for
types that need dropping.
The first two changes result in significantly fewer MIR blocks being
generated, as previously almost every statement would end up generating
a new block due to the drop of the `()` temporary being generated.
2016-04-14 19:36:16 -05:00
|
|
|
|
mir::Rvalue::Use(ref operand) => {
|
|
|
|
|
let operand = self.trans_operand(&bcx, operand);
|
|
|
|
|
(bcx, operand)
|
|
|
|
|
}
|
2015-11-02 08:39:59 -06:00
|
|
|
|
mir::Rvalue::Repeat(..) |
|
|
|
|
|
mir::Rvalue::Aggregate(..) |
|
2016-03-09 14:17:02 -06:00
|
|
|
|
mir::Rvalue::InlineAsm { .. } => {
|
2016-03-28 18:46:02 -05:00
|
|
|
|
bug!("cannot generate operand from rvalue {:?}", rvalue);
|
|
|
|
|
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-11-10 14:05:11 -06:00
|
|
|
|
|
|
|
|
|
pub fn trans_scalar_binop(&mut self,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx: &BlockAndBuilder<'bcx, 'tcx>,
|
2015-11-10 14:05:11 -06:00
|
|
|
|
op: mir::BinOp,
|
|
|
|
|
lhs: ValueRef,
|
|
|
|
|
rhs: ValueRef,
|
2016-02-01 04:04:46 -06:00
|
|
|
|
input_ty: Ty<'tcx>) -> ValueRef {
|
2015-11-10 14:05:11 -06:00
|
|
|
|
let is_float = input_ty.is_fp();
|
|
|
|
|
let is_signed = input_ty.is_signed();
|
|
|
|
|
match op {
|
|
|
|
|
mir::BinOp::Add => if is_float {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fadd(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.add(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
},
|
|
|
|
|
mir::BinOp::Sub => if is_float {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fsub(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.sub(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
},
|
|
|
|
|
mir::BinOp::Mul => if is_float {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fmul(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.mul(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
},
|
|
|
|
|
mir::BinOp::Div => if is_float {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.fdiv(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else if is_signed {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.sdiv(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.udiv(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
},
|
|
|
|
|
mir::BinOp::Rem => if is_float {
|
2016-05-08 23:36:44 -05:00
|
|
|
|
bcx.frem(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else if is_signed {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.srem(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
} else {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.urem(lhs, rhs)
|
2015-11-10 14:05:11 -06:00
|
|
|
|
},
|
2016-02-01 04:04:46 -06:00
|
|
|
|
mir::BinOp::BitOr => bcx.or(lhs, rhs),
|
|
|
|
|
mir::BinOp::BitAnd => bcx.and(lhs, rhs),
|
|
|
|
|
mir::BinOp::BitXor => bcx.xor(lhs, rhs),
|
|
|
|
|
mir::BinOp::Shl => {
|
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
common::build_unchecked_lshift(bcx,
|
|
|
|
|
lhs,
|
|
|
|
|
rhs,
|
|
|
|
|
DebugLoc::None)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
mir::BinOp::Shr => {
|
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
common::build_unchecked_rshift(bcx,
|
|
|
|
|
input_ty,
|
|
|
|
|
lhs,
|
|
|
|
|
rhs,
|
|
|
|
|
DebugLoc::None)
|
|
|
|
|
})
|
|
|
|
|
}
|
2015-11-10 14:05:11 -06:00
|
|
|
|
mir::BinOp::Eq | mir::BinOp::Lt | mir::BinOp::Gt |
|
|
|
|
|
mir::BinOp::Ne | mir::BinOp::Le | mir::BinOp::Ge => {
|
2016-02-01 04:04:46 -06:00
|
|
|
|
bcx.with_block(|bcx| {
|
|
|
|
|
base::compare_scalar_types(bcx, lhs, rhs, input_ty,
|
|
|
|
|
op.to_hir_binop(), DebugLoc::None)
|
|
|
|
|
})
|
2015-11-10 14:05:11 -06:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-03-31 00:50:07 -05:00
|
|
|
|
|
|
|
|
|
pub fn trans_scalar_checked_binop(&mut self,
|
|
|
|
|
bcx: &BlockAndBuilder<'bcx, 'tcx>,
|
|
|
|
|
op: mir::BinOp,
|
|
|
|
|
lhs: ValueRef,
|
|
|
|
|
rhs: ValueRef,
|
2016-05-25 03:55:44 -05:00
|
|
|
|
input_ty: Ty<'tcx>) -> OperandValue {
|
2016-05-26 12:02:56 -05:00
|
|
|
|
// This case can currently arise only from functions marked
|
|
|
|
|
// with #[rustc_inherit_overflow_checks] and inlined from
|
|
|
|
|
// another crate (mostly core::num generic/#[inline] fns),
|
|
|
|
|
// while the current crate doesn't use overflow checks.
|
|
|
|
|
if !bcx.ccx().check_overflow() {
|
|
|
|
|
let val = self.trans_scalar_binop(bcx, op, lhs, rhs, input_ty);
|
|
|
|
|
return OperandValue::Pair(val, C_bool(bcx.ccx(), false));
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-05 06:38:29 -05:00
|
|
|
|
// First try performing the operation on constants, which
|
|
|
|
|
// will only succeed if both operands are constant.
|
|
|
|
|
// This is necessary to determine when an overflow Assert
|
|
|
|
|
// will always panic at runtime, and produce a warning.
|
2016-07-03 16:38:37 -05:00
|
|
|
|
if let Some((val, of)) = const_scalar_checked_binop(bcx.tcx(), op, lhs, rhs, input_ty) {
|
|
|
|
|
return OperandValue::Pair(val, C_bool(bcx.ccx(), of));
|
2016-06-05 06:38:29 -05:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-31 00:50:07 -05:00
|
|
|
|
let (val, of) = match op {
|
|
|
|
|
// These are checked using intrinsics
|
|
|
|
|
mir::BinOp::Add | mir::BinOp::Sub | mir::BinOp::Mul => {
|
|
|
|
|
let oop = match op {
|
|
|
|
|
mir::BinOp::Add => OverflowOp::Add,
|
|
|
|
|
mir::BinOp::Sub => OverflowOp::Sub,
|
|
|
|
|
mir::BinOp::Mul => OverflowOp::Mul,
|
|
|
|
|
_ => unreachable!()
|
|
|
|
|
};
|
|
|
|
|
let intrinsic = get_overflow_intrinsic(oop, bcx, input_ty);
|
|
|
|
|
let res = bcx.call(intrinsic, &[lhs, rhs], None);
|
|
|
|
|
|
2016-05-25 03:55:44 -05:00
|
|
|
|
(bcx.extract_value(res, 0),
|
|
|
|
|
bcx.extract_value(res, 1))
|
2016-03-31 00:50:07 -05:00
|
|
|
|
}
|
|
|
|
|
mir::BinOp::Shl | mir::BinOp::Shr => {
|
|
|
|
|
let lhs_llty = val_ty(lhs);
|
|
|
|
|
let rhs_llty = val_ty(rhs);
|
|
|
|
|
let invert_mask = bcx.with_block(|bcx| {
|
|
|
|
|
common::shift_mask_val(bcx, lhs_llty, rhs_llty, true)
|
|
|
|
|
});
|
|
|
|
|
let outer_bits = bcx.and(rhs, invert_mask);
|
|
|
|
|
|
|
|
|
|
let of = bcx.icmp(llvm::IntNE, outer_bits, C_null(rhs_llty));
|
|
|
|
|
let val = self.trans_scalar_binop(bcx, op, lhs, rhs, input_ty);
|
|
|
|
|
|
2016-05-25 03:55:44 -05:00
|
|
|
|
(val, of)
|
2016-03-31 00:50:07 -05:00
|
|
|
|
}
|
|
|
|
|
_ => {
|
2016-05-25 00:39:32 -05:00
|
|
|
|
bug!("Operator `{:?}` is not a checkable operator", op)
|
2016-03-31 00:50:07 -05:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-05-25 03:55:44 -05:00
|
|
|
|
OperandValue::Pair(val, of)
|
2016-03-31 00:50:07 -05:00
|
|
|
|
}
|
2015-10-21 16:42:25 -05:00
|
|
|
|
}
|
2015-11-03 05:35:09 -06:00
|
|
|
|
|
2016-05-27 17:10:16 -05:00
|
|
|
|
pub fn rvalue_creates_operand<'bcx, 'tcx>(_mir: &mir::Mir<'tcx>,
|
|
|
|
|
_bcx: &BlockAndBuilder<'bcx, 'tcx>,
|
Various improvements to MIR and LLVM IR Construction
Primarily affects the MIR construction, which indirectly improves LLVM
IR generation, but some LLVM IR changes have been made too.
* Handle "statement expressions" more intelligently. These are
expressions that always evaluate to `()`. Previously a temporary would
be generated as a destination to translate into, which is unnecessary.
This affects assignment, augmented assignment, `return`, `break` and
`continue`.
* Avoid inserting drops for non-drop types in more places. Scheduled
drops were already skipped for types that we knew wouldn't need
dropping at construction time. However manually-inserted drops like
those for `x` in `x = y;` were still generated. `build_drop` now takes
a type parameter like its `schedule_drop` counterpart and checks to
see if the type needs dropping.
* Avoid generating an extra temporary for an assignment where the types
involved don't need dropping. Previously an expression like
`a = b + 1;` would result in a temporary for `b + 1`. This is so the
RHS can be evaluated, then the LHS evaluated and dropped and have
everything work correctly. However, this isn't necessary if the `LHS`
doesn't need a drop, as we can just overwrite the existing value.
* Improves lvalue analysis to allow treating an `Rvalue::Use` as an
operand in certain conditions. The reason for it never being an
operand is so it can be zeroed/drop-filled, but this is only true for
types that need dropping.
The first two changes result in significantly fewer MIR blocks being
generated, as previously almost every statement would end up generating
a new block due to the drop of the `()` temporary being generated.
2016-04-14 19:36:16 -05:00
|
|
|
|
rvalue: &mir::Rvalue<'tcx>) -> bool {
|
2015-11-03 05:35:09 -06:00
|
|
|
|
match *rvalue {
|
|
|
|
|
mir::Rvalue::Ref(..) |
|
|
|
|
|
mir::Rvalue::Len(..) |
|
|
|
|
|
mir::Rvalue::Cast(..) | // (*)
|
|
|
|
|
mir::Rvalue::BinaryOp(..) |
|
2016-03-31 00:50:07 -05:00
|
|
|
|
mir::Rvalue::CheckedBinaryOp(..) |
|
2015-11-03 05:35:09 -06:00
|
|
|
|
mir::Rvalue::UnaryOp(..) |
|
2016-05-27 17:10:16 -05:00
|
|
|
|
mir::Rvalue::Box(..) |
|
|
|
|
|
mir::Rvalue::Use(..) =>
|
2015-11-03 05:35:09 -06:00
|
|
|
|
true,
|
|
|
|
|
mir::Rvalue::Repeat(..) |
|
|
|
|
|
mir::Rvalue::Aggregate(..) |
|
2016-03-09 14:17:02 -06:00
|
|
|
|
mir::Rvalue::InlineAsm { .. } =>
|
2015-11-03 05:35:09 -06:00
|
|
|
|
false,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// (*) this is only true if the type is suitable
|
|
|
|
|
}
|
2016-03-31 00:50:07 -05:00
|
|
|
|
|
|
|
|
|
#[derive(Copy, Clone)]
|
|
|
|
|
enum OverflowOp {
|
|
|
|
|
Add, Sub, Mul
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn get_overflow_intrinsic(oop: OverflowOp, bcx: &BlockAndBuilder, ty: Ty) -> ValueRef {
|
|
|
|
|
use syntax::ast::IntTy::*;
|
|
|
|
|
use syntax::ast::UintTy::*;
|
|
|
|
|
use rustc::ty::{TyInt, TyUint};
|
|
|
|
|
|
|
|
|
|
let tcx = bcx.tcx();
|
|
|
|
|
|
|
|
|
|
let new_sty = match ty.sty {
|
|
|
|
|
TyInt(Is) => match &tcx.sess.target.target.target_pointer_width[..] {
|
|
|
|
|
"32" => TyInt(I32),
|
|
|
|
|
"64" => TyInt(I64),
|
|
|
|
|
_ => panic!("unsupported target word size")
|
|
|
|
|
},
|
|
|
|
|
TyUint(Us) => match &tcx.sess.target.target.target_pointer_width[..] {
|
|
|
|
|
"32" => TyUint(U32),
|
|
|
|
|
"64" => TyUint(U64),
|
|
|
|
|
_ => panic!("unsupported target word size")
|
|
|
|
|
},
|
|
|
|
|
ref t @ TyUint(_) | ref t @ TyInt(_) => t.clone(),
|
|
|
|
|
_ => panic!("tried to get overflow intrinsic for op applied to non-int type")
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let name = match oop {
|
|
|
|
|
OverflowOp::Add => match new_sty {
|
|
|
|
|
TyInt(I8) => "llvm.sadd.with.overflow.i8",
|
|
|
|
|
TyInt(I16) => "llvm.sadd.with.overflow.i16",
|
|
|
|
|
TyInt(I32) => "llvm.sadd.with.overflow.i32",
|
|
|
|
|
TyInt(I64) => "llvm.sadd.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
TyUint(U8) => "llvm.uadd.with.overflow.i8",
|
|
|
|
|
TyUint(U16) => "llvm.uadd.with.overflow.i16",
|
|
|
|
|
TyUint(U32) => "llvm.uadd.with.overflow.i32",
|
|
|
|
|
TyUint(U64) => "llvm.uadd.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
_ => unreachable!(),
|
|
|
|
|
},
|
|
|
|
|
OverflowOp::Sub => match new_sty {
|
|
|
|
|
TyInt(I8) => "llvm.ssub.with.overflow.i8",
|
|
|
|
|
TyInt(I16) => "llvm.ssub.with.overflow.i16",
|
|
|
|
|
TyInt(I32) => "llvm.ssub.with.overflow.i32",
|
|
|
|
|
TyInt(I64) => "llvm.ssub.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
TyUint(U8) => "llvm.usub.with.overflow.i8",
|
|
|
|
|
TyUint(U16) => "llvm.usub.with.overflow.i16",
|
|
|
|
|
TyUint(U32) => "llvm.usub.with.overflow.i32",
|
|
|
|
|
TyUint(U64) => "llvm.usub.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
_ => unreachable!(),
|
|
|
|
|
},
|
|
|
|
|
OverflowOp::Mul => match new_sty {
|
|
|
|
|
TyInt(I8) => "llvm.smul.with.overflow.i8",
|
|
|
|
|
TyInt(I16) => "llvm.smul.with.overflow.i16",
|
|
|
|
|
TyInt(I32) => "llvm.smul.with.overflow.i32",
|
|
|
|
|
TyInt(I64) => "llvm.smul.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
TyUint(U8) => "llvm.umul.with.overflow.i8",
|
|
|
|
|
TyUint(U16) => "llvm.umul.with.overflow.i16",
|
|
|
|
|
TyUint(U32) => "llvm.umul.with.overflow.i32",
|
|
|
|
|
TyUint(U64) => "llvm.umul.with.overflow.i64",
|
|
|
|
|
|
|
|
|
|
_ => unreachable!(),
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
bcx.ccx().get_intrinsic(&name)
|
|
|
|
|
}
|