2012-12-03 18:48:01 -06:00
|
|
|
// Copyright 2012 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.
|
|
|
|
|
2013-05-17 17:28:44 -05:00
|
|
|
|
2013-03-07 14:34:58 -06:00
|
|
|
use back::abi;
|
2014-01-25 11:48:57 -06:00
|
|
|
use lib::llvm::{llvm, ConstFCmp, ConstICmp, SetLinkage, PrivateLinkage, ValueRef, Bool, True,
|
|
|
|
False};
|
2013-05-19 15:03:52 -05:00
|
|
|
use lib::llvm::{IntEQ, IntNE, IntUGT, IntUGE, IntULT, IntULE, IntSGT, IntSGE, IntSLT, IntSLE,
|
|
|
|
RealOEQ, RealOGT, RealOGE, RealOLT, RealOLE, RealONE};
|
|
|
|
|
2013-03-07 00:30:20 -06:00
|
|
|
use metadata::csearch;
|
2012-12-23 16:41:37 -06:00
|
|
|
use middle::const_eval;
|
2014-05-14 14:31:30 -05:00
|
|
|
use middle::def;
|
2013-02-24 16:38:59 -06:00
|
|
|
use middle::trans::adt;
|
2013-02-25 13:11:21 -06:00
|
|
|
use middle::trans::base;
|
2013-06-16 23:23:24 -05:00
|
|
|
use middle::trans::base::push_ctxt;
|
2014-01-27 06:18:36 -06:00
|
|
|
use middle::trans::closure;
|
2012-12-13 15:05:22 -06:00
|
|
|
use middle::trans::common::*;
|
2012-12-23 16:41:37 -06:00
|
|
|
use middle::trans::consts;
|
|
|
|
use middle::trans::expr;
|
2013-03-07 00:30:20 -06:00
|
|
|
use middle::trans::inline;
|
2013-01-30 13:46:19 -06:00
|
|
|
use middle::trans::machine;
|
2014-03-08 14:36:22 -06:00
|
|
|
use middle::trans::type_::Type;
|
2013-02-25 13:11:21 -06:00
|
|
|
use middle::trans::type_of;
|
2014-02-20 19:44:29 -06:00
|
|
|
use middle::trans::debuginfo;
|
2012-12-23 16:41:37 -06:00
|
|
|
use middle::ty;
|
Cleanup substitutions and treatment of generics around traits in a number of ways.
- In a TraitRef, use the self type consistently to refer to the Self type:
- trait ref in `impl Trait<A,B,C> for S` has a self type of `S`.
- trait ref in `A:Trait` has the self type `A`
- trait ref associated with a trait decl has self type `Self`
- trait ref associated with a supertype has self type `Self`
- trait ref in an object type `@Trait` has no self type
- Rewrite `each_bound_traits_and_supertraits` to perform
substitutions as it goes, and thus yield a series of trait refs
that are always in the same 'namespace' as the type parameter
bound given as input. Before, we left this to the caller, but
this doesn't work because the caller lacks adequare information
to perform the type substitutions correctly.
- For provided methods, substitute the generics involved in the provided
method correctly.
- Introduce TypeParameterDef, which tracks the bounds declared on a type
parameter and brings them together with the def_id and (in the future)
other information (maybe even the parameter's name!).
- Introduce Subst trait, which helps to cleanup a lot of the
repetitive code involved with doing type substitution.
- Introduce Repr trait, which makes debug printouts far more convenient.
Fixes #4183. Needed for #5656.
2013-04-09 00:54:49 -05:00
|
|
|
use util::ppaux::{Repr, ty_to_str};
|
2012-12-13 15:05:22 -06:00
|
|
|
|
2013-08-03 19:13:14 -05:00
|
|
|
use std::c_str::ToCStr;
|
2014-05-16 12:15:33 -05:00
|
|
|
use std::gc::Gc;
|
2014-05-04 01:43:38 -05:00
|
|
|
use std::vec;
|
2014-02-26 11:58:41 -06:00
|
|
|
use libc::c_uint;
|
2014-02-13 23:07:09 -06:00
|
|
|
use syntax::{ast, ast_util};
|
2012-07-31 20:34:36 -05:00
|
|
|
|
2014-01-09 07:05:33 -06:00
|
|
|
pub fn const_lit(cx: &CrateContext, e: &ast::Expr, lit: ast::Lit)
|
2012-07-31 20:34:36 -05:00
|
|
|
-> ValueRef {
|
2013-06-16 23:23:24 -05:00
|
|
|
let _icx = push_ctxt("trans_lit");
|
2012-08-06 14:34:08 -05:00
|
|
|
match lit.node {
|
2014-06-06 10:04:04 -05:00
|
|
|
ast::LitByte(b) => C_integral(Type::uint_from_ty(cx, ast::TyU8), b as u64, false),
|
2014-03-15 15:29:34 -05:00
|
|
|
ast::LitChar(i) => C_integral(Type::char(cx), i as u64, false),
|
2014-01-09 07:05:33 -06:00
|
|
|
ast::LitInt(i, t) => C_integral(Type::int_from_ty(cx, t), i as u64, true),
|
|
|
|
ast::LitUint(u, t) => C_integral(Type::uint_from_ty(cx, t), u, false),
|
|
|
|
ast::LitIntUnsuffixed(i) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let lit_int_ty = ty::node_id_to_type(cx.tcx(), e.id);
|
2014-01-09 07:05:33 -06:00
|
|
|
match ty::get(lit_int_ty).sty {
|
|
|
|
ty::ty_int(t) => {
|
|
|
|
C_integral(Type::int_from_ty(cx, t), i as u64, true)
|
|
|
|
}
|
|
|
|
ty::ty_uint(t) => {
|
|
|
|
C_integral(Type::uint_from_ty(cx, t), i as u64, false)
|
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(lit.span,
|
2014-05-16 12:45:16 -05:00
|
|
|
format!("integer literal has type {} (expected int \
|
|
|
|
or uint)",
|
|
|
|
ty_to_str(cx.tcx(), lit_int_ty)).as_slice())
|
2014-01-09 07:05:33 -06:00
|
|
|
}
|
2012-07-31 20:34:36 -05:00
|
|
|
}
|
2014-01-15 19:15:39 -06:00
|
|
|
ast::LitFloat(ref fs, t) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
C_floating(fs.get(), Type::float_from_ty(cx, t))
|
2014-01-15 19:15:39 -06:00
|
|
|
}
|
|
|
|
ast::LitFloatUnsuffixed(ref fs) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let lit_float_ty = ty::node_id_to_type(cx.tcx(), e.id);
|
2014-01-09 07:05:33 -06:00
|
|
|
match ty::get(lit_float_ty).sty {
|
|
|
|
ty::ty_float(t) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
C_floating(fs.get(), Type::float_from_ty(cx, t))
|
2014-01-09 07:05:33 -06:00
|
|
|
}
|
|
|
|
_ => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().span_bug(lit.span,
|
2014-01-09 07:05:33 -06:00
|
|
|
"floating point literal doesn't have the right type");
|
|
|
|
}
|
|
|
|
}
|
2012-11-07 20:40:34 -06:00
|
|
|
}
|
2014-03-15 15:29:34 -05:00
|
|
|
ast::LitBool(b) => C_bool(cx, b),
|
|
|
|
ast::LitNil => C_nil(cx),
|
2014-01-10 16:02:36 -06:00
|
|
|
ast::LitStr(ref s, _) => C_str_slice(cx, (*s).clone()),
|
2014-03-21 00:10:44 -05:00
|
|
|
ast::LitBinary(ref data) => C_binary_slice(cx, data.as_slice()),
|
2012-07-31 20:34:36 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-19 18:47:15 -06:00
|
|
|
pub fn const_ptrcast(cx: &CrateContext, a: ValueRef, t: Type) -> ValueRef {
|
2013-01-10 23:23:07 -06:00
|
|
|
unsafe {
|
2013-06-16 05:52:44 -05:00
|
|
|
let b = llvm::LLVMConstPointerCast(a, t.ptr_to().to_ref());
|
2014-03-20 21:49:20 -05:00
|
|
|
assert!(cx.const_globals.borrow_mut().insert(b as int, a));
|
2013-01-10 23:23:07 -06:00
|
|
|
b
|
|
|
|
}
|
2012-08-09 18:05:34 -05:00
|
|
|
}
|
|
|
|
|
2014-03-06 10:47:24 -06:00
|
|
|
fn const_vec(cx: &CrateContext, e: &ast::Expr,
|
2014-05-16 12:15:33 -05:00
|
|
|
es: &[Gc<ast::Expr>], is_local: bool) -> (ValueRef, Type, bool) {
|
2014-03-15 15:29:34 -05:00
|
|
|
let vec_ty = ty::expr_ty(cx.tcx(), e);
|
|
|
|
let unit_ty = ty::sequence_element_type(cx.tcx(), vec_ty);
|
2013-10-14 23:37:32 -05:00
|
|
|
let llunitty = type_of::type_of(cx, unit_ty);
|
2014-05-16 12:15:33 -05:00
|
|
|
let (vs, inlineable) = vec::unzip(es.iter().map(|e| const_expr(cx, &**e, is_local)));
|
2013-10-14 23:37:32 -05:00
|
|
|
// If the vector contains enums, an LLVM array won't work.
|
|
|
|
let v = if vs.iter().any(|vi| val_ty(*vi) != llunitty) {
|
2014-05-04 01:43:38 -05:00
|
|
|
C_struct(cx, vs.as_slice(), false)
|
2013-10-14 23:37:32 -05:00
|
|
|
} else {
|
2014-05-04 01:43:38 -05:00
|
|
|
C_array(llunitty, vs.as_slice())
|
2013-10-14 23:37:32 -05:00
|
|
|
};
|
|
|
|
(v, llunitty, inlineable.iter().fold(true, |a, &b| a && b))
|
2012-08-03 23:44:42 -05:00
|
|
|
}
|
|
|
|
|
2013-12-19 18:47:15 -06:00
|
|
|
fn const_addr_of(cx: &CrateContext, cv: ValueRef) -> ValueRef {
|
2013-03-07 14:34:39 -06:00
|
|
|
unsafe {
|
2013-11-21 17:42:55 -06:00
|
|
|
let gv = "const".with_c_str(|name| {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMAddGlobal(cx.llmod, val_ty(cv).to_ref(), name)
|
2013-11-21 17:42:55 -06:00
|
|
|
});
|
2013-03-07 14:34:39 -06:00
|
|
|
llvm::LLVMSetInitializer(gv, cv);
|
|
|
|
llvm::LLVMSetGlobalConstant(gv, True);
|
2013-03-28 20:15:35 -05:00
|
|
|
SetLinkage(gv, PrivateLinkage);
|
2013-03-07 14:34:39 -06:00
|
|
|
gv
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-19 18:47:15 -06:00
|
|
|
fn const_deref_ptr(cx: &CrateContext, v: ValueRef) -> ValueRef {
|
2014-03-20 21:49:20 -05:00
|
|
|
let v = match cx.const_globals.borrow().find(&(v as int)) {
|
2013-03-22 21:26:41 -05:00
|
|
|
Some(&v) => v,
|
2013-03-08 12:22:48 -06:00
|
|
|
None => v
|
|
|
|
};
|
2013-01-10 23:23:07 -06:00
|
|
|
unsafe {
|
2013-05-18 21:02:45 -05:00
|
|
|
assert_eq!(llvm::LLVMIsGlobalConstant(v), True);
|
2013-03-08 12:22:48 -06:00
|
|
|
llvm::LLVMGetInitializer(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-19 18:47:15 -06:00
|
|
|
fn const_deref_newtype(cx: &CrateContext, v: ValueRef, t: ty::t)
|
2013-03-08 12:22:48 -06:00
|
|
|
-> ValueRef {
|
|
|
|
let repr = adt::represent_type(cx, t);
|
2014-04-21 19:03:02 -05:00
|
|
|
adt::const_get_field(cx, &*repr, v, 0, 0)
|
2013-03-08 12:22:48 -06:00
|
|
|
}
|
|
|
|
|
2013-12-19 18:47:15 -06:00
|
|
|
fn const_deref(cx: &CrateContext, v: ValueRef, t: ty::t, explicit: bool)
|
2013-03-08 12:22:48 -06:00
|
|
|
-> (ValueRef, ty::t) {
|
2013-11-01 20:06:31 -05:00
|
|
|
match ty::deref(t, explicit) {
|
2013-03-08 12:22:48 -06:00
|
|
|
Some(ref mt) => {
|
2013-09-01 20:45:37 -05:00
|
|
|
assert!(mt.mutbl != ast::MutMutable);
|
2013-03-08 12:22:48 -06:00
|
|
|
let dv = match ty::get(t).sty {
|
2014-04-09 02:15:31 -05:00
|
|
|
ty::ty_ptr(mt) | ty::ty_rptr(_, mt) => {
|
|
|
|
match ty::get(mt.ty).sty {
|
2014-06-11 00:18:57 -05:00
|
|
|
ty::ty_vec(_, None) | ty::ty_str | ty::ty_trait(..) => {
|
|
|
|
cx.sess().bug("unexpected unsized type")
|
|
|
|
}
|
2014-04-09 02:15:31 -05:00
|
|
|
_ => const_deref_ptr(cx, v),
|
|
|
|
}
|
2013-03-08 12:22:48 -06:00
|
|
|
}
|
2013-11-28 14:22:53 -06:00
|
|
|
ty::ty_enum(..) | ty::ty_struct(..) => {
|
2013-03-08 12:22:48 -06:00
|
|
|
const_deref_newtype(cx, v, t)
|
|
|
|
}
|
|
|
|
_ => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().bug(format!("unexpected dereferenceable type {}",
|
2014-05-16 12:45:16 -05:00
|
|
|
ty_to_str(cx.tcx(), t)).as_slice())
|
2013-03-08 12:22:48 -06:00
|
|
|
}
|
|
|
|
};
|
|
|
|
(dv, mt.ty)
|
|
|
|
}
|
|
|
|
None => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().bug(format!("can't dereference const of type {}",
|
2014-05-16 12:45:16 -05:00
|
|
|
ty_to_str(cx.tcx(), t)).as_slice())
|
2013-03-08 12:22:48 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2012-08-08 21:41:50 -05:00
|
|
|
}
|
|
|
|
|
2014-03-06 10:47:24 -06:00
|
|
|
pub fn get_const_val(cx: &CrateContext,
|
2013-09-11 12:06:16 -05:00
|
|
|
mut def_id: ast::DefId) -> (ValueRef, bool) {
|
2014-03-20 21:49:20 -05:00
|
|
|
let contains_key = cx.const_values.borrow().contains_key(&def_id.node);
|
2013-06-13 02:19:50 -05:00
|
|
|
if !ast_util::is_local(def_id) || !contains_key {
|
2013-03-07 00:30:20 -06:00
|
|
|
if !ast_util::is_local(def_id) {
|
2013-07-11 16:34:10 -05:00
|
|
|
def_id = inline::maybe_instantiate_inline(cx, def_id);
|
2013-03-07 00:30:20 -06:00
|
|
|
}
|
2013-12-27 18:09:29 -06:00
|
|
|
|
2014-02-13 23:07:09 -06:00
|
|
|
match cx.tcx.map.expect_item(def_id.node).node {
|
|
|
|
ast::ItemStatic(_, ast::MutImmutable, _) => {
|
|
|
|
trans_const(cx, ast::MutImmutable, def_id.node);
|
2012-11-13 00:10:15 -06:00
|
|
|
}
|
2014-02-13 23:07:09 -06:00
|
|
|
_ => {}
|
2012-11-13 00:10:15 -06:00
|
|
|
}
|
|
|
|
}
|
2013-12-18 19:08:56 -06:00
|
|
|
|
2014-03-20 21:49:20 -05:00
|
|
|
(cx.const_values.borrow().get_copy(&def_id.node),
|
|
|
|
!cx.non_inlineable_statics.borrow().contains(&def_id.node))
|
2012-11-13 00:10:15 -06:00
|
|
|
}
|
2012-08-08 21:41:50 -05:00
|
|
|
|
2014-03-06 10:47:24 -06:00
|
|
|
pub fn const_expr(cx: &CrateContext, e: &ast::Expr, is_local: bool) -> (ValueRef, bool) {
|
2014-01-27 06:18:36 -06:00
|
|
|
let (llconst, inlineable) = const_expr_unadjusted(cx, e, is_local);
|
2013-09-11 12:06:16 -05:00
|
|
|
let mut llconst = llconst;
|
|
|
|
let mut inlineable = inlineable;
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2014-04-09 10:18:40 -05:00
|
|
|
let ety_adjusted = ty::expr_ty_adjusted(cx.tcx(), e);
|
2014-03-20 21:49:20 -05:00
|
|
|
let opt_adj = cx.tcx.adjustments.borrow().find_copy(&e.id);
|
|
|
|
match opt_adj {
|
2013-03-07 14:34:58 -06:00
|
|
|
None => { }
|
2014-01-03 17:08:48 -06:00
|
|
|
Some(adj) => {
|
2014-04-10 08:26:26 -05:00
|
|
|
match adj {
|
2014-04-11 10:03:10 -05:00
|
|
|
ty::AutoAddEnv(ty::RegionTraitStore(ty::ReStatic, _)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let def = ty::resolve_expr(cx.tcx(), e);
|
2014-01-27 06:18:36 -06:00
|
|
|
let wrapper = closure::get_wrapper_for_bare_fn(cx,
|
|
|
|
ety_adjusted,
|
|
|
|
def,
|
|
|
|
llconst,
|
|
|
|
is_local);
|
2014-03-15 15:29:34 -05:00
|
|
|
llconst = C_struct(cx, [wrapper, C_null(Type::i8p(cx))], false)
|
2014-01-03 17:08:48 -06:00
|
|
|
}
|
2014-04-11 10:03:10 -05:00
|
|
|
ty::AutoAddEnv(store) => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess()
|
2014-01-03 17:08:48 -06:00
|
|
|
.span_bug(e.span,
|
2014-04-11 10:03:10 -05:00
|
|
|
format!("unexpected static function: {:?}",
|
2014-05-16 12:45:16 -05:00
|
|
|
store).as_slice())
|
2014-01-03 17:08:48 -06:00
|
|
|
}
|
|
|
|
ty::AutoObject(..) => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess()
|
2014-01-03 17:08:48 -06:00
|
|
|
.span_unimpl(e.span,
|
|
|
|
"unimplemented const coercion to trait \
|
|
|
|
object");
|
|
|
|
}
|
|
|
|
ty::AutoDerefRef(ref adj) => {
|
|
|
|
let mut ty = ety;
|
|
|
|
let mut maybe_ptr = None;
|
2014-01-29 18:20:34 -06:00
|
|
|
for _ in range(0, adj.autoderefs) {
|
2014-01-03 17:08:48 -06:00
|
|
|
let (dv, dt) = const_deref(cx, llconst, ty, false);
|
|
|
|
maybe_ptr = Some(llconst);
|
|
|
|
llconst = dv;
|
|
|
|
ty = dt;
|
2014-01-29 18:20:34 -06:00
|
|
|
}
|
2013-07-27 03:38:55 -05:00
|
|
|
|
2014-01-03 17:08:48 -06:00
|
|
|
match adj.autoref {
|
|
|
|
None => { }
|
|
|
|
Some(ref autoref) => {
|
|
|
|
// Don't copy data to do a deref+ref.
|
|
|
|
let llptr = match maybe_ptr {
|
|
|
|
Some(ptr) => ptr,
|
|
|
|
None => {
|
|
|
|
inlineable = false;
|
|
|
|
const_addr_of(cx, llconst)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
match *autoref {
|
|
|
|
ty::AutoUnsafe(m) |
|
|
|
|
ty::AutoPtr(ty::ReStatic, m) => {
|
|
|
|
assert!(m != ast::MutMutable);
|
|
|
|
llconst = llptr;
|
|
|
|
}
|
|
|
|
ty::AutoBorrowVec(ty::ReStatic, m) => {
|
|
|
|
assert!(m != ast::MutMutable);
|
|
|
|
assert_eq!(abi::slice_elt_base, 0);
|
|
|
|
assert_eq!(abi::slice_elt_len, 1);
|
|
|
|
match ty::get(ty).sty {
|
2014-04-09 02:15:31 -05:00
|
|
|
ty::ty_vec(_, Some(len)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
llconst = C_struct(cx, [
|
2014-01-03 17:08:48 -06:00
|
|
|
llptr,
|
|
|
|
C_uint(cx, len)
|
|
|
|
], false);
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
2014-05-16 12:45:16 -05:00
|
|
|
cx.sess()
|
|
|
|
.span_bug(e.span,
|
|
|
|
format!("unimplemented const \
|
|
|
|
autoref {:?}",
|
|
|
|
autoref).as_slice())
|
2013-07-27 03:38:55 -05:00
|
|
|
}
|
|
|
|
}
|
2013-03-07 14:34:58 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let llty = type_of::sizing_type_of(cx, ety_adjusted);
|
2013-03-03 16:30:06 -06:00
|
|
|
let csize = machine::llsize_of_alloc(cx, val_ty(llconst));
|
|
|
|
let tsize = machine::llsize_of_alloc(cx, llty);
|
|
|
|
if csize != tsize {
|
|
|
|
unsafe {
|
2014-01-26 02:43:42 -06:00
|
|
|
// FIXME these values could use some context
|
2013-03-03 16:30:06 -06:00
|
|
|
llvm::LLVMDumpValue(llconst);
|
2013-03-07 14:34:58 -06:00
|
|
|
llvm::LLVMDumpValue(C_undef(llty));
|
2013-03-03 16:30:06 -06:00
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().bug(format!("const {} of type {} has size {} instead of {}",
|
2014-03-15 15:29:34 -05:00
|
|
|
e.repr(cx.tcx()), ty_to_str(cx.tcx(), ety),
|
2014-05-16 12:45:16 -05:00
|
|
|
csize, tsize).as_slice());
|
2013-03-03 16:30:06 -06:00
|
|
|
}
|
2013-09-11 12:06:16 -05:00
|
|
|
(llconst, inlineable)
|
2013-03-03 16:30:06 -06:00
|
|
|
}
|
|
|
|
|
2013-09-11 12:06:16 -05:00
|
|
|
// the bool returned is whether this expression can be inlined into other crates
|
|
|
|
// if it's assigned to a static.
|
2014-03-06 10:47:24 -06:00
|
|
|
fn const_expr_unadjusted(cx: &CrateContext, e: &ast::Expr,
|
2014-01-27 06:18:36 -06:00
|
|
|
is_local: bool) -> (ValueRef, bool) {
|
2014-05-16 12:15:33 -05:00
|
|
|
let map_list = |exprs: &[Gc<ast::Expr>]| {
|
|
|
|
exprs.iter().map(|e| const_expr(cx, &**e, is_local))
|
2014-03-04 12:02:49 -06:00
|
|
|
.fold((Vec::new(), true),
|
2014-03-08 14:36:22 -06:00
|
|
|
|(l, all_inlineable), (val, inlineable)| {
|
2014-03-30 22:53:26 -05:00
|
|
|
(l.append_one(val), all_inlineable && inlineable)
|
2013-09-11 12:06:16 -05:00
|
|
|
})
|
2014-01-27 06:18:36 -06:00
|
|
|
};
|
2013-01-10 23:23:07 -06:00
|
|
|
unsafe {
|
2013-06-16 23:23:24 -05:00
|
|
|
let _icx = push_ctxt("const_expr");
|
2013-03-20 00:17:42 -05:00
|
|
|
return match e.node {
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprLit(ref lit) => {
|
|
|
|
(consts::const_lit(cx, e, (**lit).clone()), true)
|
2014-01-10 16:02:36 -06:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprBinary(b, ref e1, ref e2) => {
|
|
|
|
let (te1, _) = const_expr(cx, &**e1, is_local);
|
|
|
|
let (te2, _) = const_expr(cx, &**e2, is_local);
|
2012-07-31 20:34:36 -05:00
|
|
|
|
2013-01-10 23:23:07 -06:00
|
|
|
let te2 = base::cast_shift_const_rhs(b, te1, te2);
|
2012-07-31 20:34:36 -05:00
|
|
|
|
2013-01-10 23:23:07 -06:00
|
|
|
/* Neither type is bottom, and we expect them to be unified
|
|
|
|
* already, so the following is safe. */
|
2014-05-16 12:15:33 -05:00
|
|
|
let ty = ty::expr_ty(cx.tcx(), &**e1);
|
2013-01-10 23:23:07 -06:00
|
|
|
let is_float = ty::type_is_fp(ty);
|
|
|
|
let signed = ty::type_is_signed(ty);
|
2013-09-11 12:06:16 -05:00
|
|
|
return (match b {
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiAdd => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFAdd(te1, te2) }
|
|
|
|
else { llvm::LLVMConstAdd(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiSub => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFSub(te1, te2) }
|
|
|
|
else { llvm::LLVMConstSub(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiMul => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFMul(te1, te2) }
|
|
|
|
else { llvm::LLVMConstMul(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiDiv => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFDiv(te1, te2) }
|
|
|
|
else if signed { llvm::LLVMConstSDiv(te1, te2) }
|
|
|
|
else { llvm::LLVMConstUDiv(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiRem => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFRem(te1, te2) }
|
|
|
|
else if signed { llvm::LLVMConstSRem(te1, te2) }
|
|
|
|
else { llvm::LLVMConstURem(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiAnd => llvm::LLVMConstAnd(te1, te2),
|
|
|
|
ast::BiOr => llvm::LLVMConstOr(te1, te2),
|
|
|
|
ast::BiBitXor => llvm::LLVMConstXor(te1, te2),
|
|
|
|
ast::BiBitAnd => llvm::LLVMConstAnd(te1, te2),
|
|
|
|
ast::BiBitOr => llvm::LLVMConstOr(te1, te2),
|
|
|
|
ast::BiShl => llvm::LLVMConstShl(te1, te2),
|
|
|
|
ast::BiShr => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if signed { llvm::LLVMConstAShr(te1, te2) }
|
|
|
|
else { llvm::LLVMConstLShr(te1, te2) }
|
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiEq => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealOEQ, te1, te2) }
|
|
|
|
else { ConstICmp(IntEQ, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
},
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiLt => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealOLT, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
else {
|
2013-05-19 19:18:56 -05:00
|
|
|
if signed { ConstICmp(IntSLT, te1, te2) }
|
|
|
|
else { ConstICmp(IntULT, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
}
|
|
|
|
},
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiLe => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealOLE, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
else {
|
2013-05-19 19:18:56 -05:00
|
|
|
if signed { ConstICmp(IntSLE, te1, te2) }
|
|
|
|
else { ConstICmp(IntULE, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
}
|
|
|
|
},
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiNe => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealONE, te1, te2) }
|
|
|
|
else { ConstICmp(IntNE, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
},
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiGe => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealOGE, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
else {
|
2013-05-19 19:18:56 -05:00
|
|
|
if signed { ConstICmp(IntSGE, te1, te2) }
|
|
|
|
else { ConstICmp(IntUGE, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
}
|
|
|
|
},
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::BiGt => {
|
2013-05-19 19:18:56 -05:00
|
|
|
if is_float { ConstFCmp(RealOGT, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
else {
|
2013-05-19 19:18:56 -05:00
|
|
|
if signed { ConstICmp(IntSGT, te1, te2) }
|
|
|
|
else { ConstICmp(IntUGT, te1, te2) }
|
2013-05-19 15:03:52 -05:00
|
|
|
}
|
|
|
|
},
|
2013-09-11 12:06:16 -05:00
|
|
|
}, true)
|
2013-05-19 15:03:52 -05:00
|
|
|
},
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprUnary(u, ref e) => {
|
|
|
|
let (te, _) = const_expr(cx, &**e, is_local);
|
|
|
|
let ty = ty::expr_ty(cx.tcx(), &**e);
|
2013-01-10 23:23:07 -06:00
|
|
|
let is_float = ty::type_is_fp(ty);
|
2013-09-11 12:06:16 -05:00
|
|
|
return (match u {
|
2013-12-31 14:55:39 -06:00
|
|
|
ast::UnBox | ast::UnUniq | ast::UnDeref => {
|
2013-03-08 12:22:48 -06:00
|
|
|
let (dv, _dt) = const_deref(cx, te, ty, true);
|
|
|
|
dv
|
|
|
|
}
|
2014-03-16 03:29:05 -05:00
|
|
|
ast::UnNot => llvm::LLVMConstNot(te),
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::UnNeg => {
|
2013-01-10 23:23:07 -06:00
|
|
|
if is_float { llvm::LLVMConstFNeg(te) }
|
|
|
|
else { llvm::LLVMConstNeg(te) }
|
|
|
|
}
|
2013-09-11 12:06:16 -05:00
|
|
|
}, true)
|
2012-07-31 20:34:36 -05:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprField(ref base, field, _) => {
|
|
|
|
let bt = ty::expr_ty_adjusted(cx.tcx(), &**base);
|
2013-02-24 17:03:47 -06:00
|
|
|
let brepr = adt::represent_type(cx, bt);
|
2014-05-16 12:15:33 -05:00
|
|
|
let (bv, inlineable) = const_expr(cx, &**base, is_local);
|
2014-03-15 15:29:34 -05:00
|
|
|
expr::with_field_tys(cx.tcx(), bt, None, |discr, field_tys| {
|
2014-06-13 16:56:42 -05:00
|
|
|
let ix = ty::field_idx_strict(cx.tcx(), field.node.name, field_tys);
|
2014-04-21 19:03:02 -05:00
|
|
|
(adt::const_get_field(cx, &*brepr, bv, discr, ix), inlineable)
|
2013-11-21 17:42:55 -06:00
|
|
|
})
|
2012-08-28 17:54:45 -05:00
|
|
|
}
|
2012-08-08 21:41:50 -05:00
|
|
|
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprIndex(ref base, ref index) => {
|
|
|
|
let bt = ty::expr_ty_adjusted(cx.tcx(), &**base);
|
|
|
|
let (bv, inlineable) = const_expr(cx, &**base, is_local);
|
|
|
|
let iv = match const_eval::eval_const_expr(cx.tcx(), &**index) {
|
2013-01-10 23:23:07 -06:00
|
|
|
const_eval::const_int(i) => i as u64,
|
|
|
|
const_eval::const_uint(u) => u,
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(index.span,
|
|
|
|
"index is not an integer-constant expression")
|
2013-01-10 23:23:07 -06:00
|
|
|
};
|
2013-02-08 03:52:55 -06:00
|
|
|
let (arr, len) = match ty::get(bt).sty {
|
2014-04-09 02:15:31 -05:00
|
|
|
ty::ty_vec(_, Some(u)) => (bv, C_uint(cx, u)),
|
|
|
|
ty::ty_rptr(_, mt) => match ty::get(mt.ty).sty {
|
2014-04-28 20:10:23 -05:00
|
|
|
ty::ty_vec(_, None) | ty::ty_str => {
|
2014-04-09 02:15:31 -05:00
|
|
|
let e1 = const_get_elt(cx, bv, [0]);
|
|
|
|
(const_deref_ptr(cx, e1), const_get_elt(cx, bv, [1]))
|
|
|
|
},
|
|
|
|
_ => cx.sess().span_bug(base.span,
|
|
|
|
"index-expr base must be a vector or string type")
|
|
|
|
},
|
2014-04-11 01:01:31 -05:00
|
|
|
_ => cx.sess().span_bug(base.span,
|
2014-04-09 02:15:31 -05:00
|
|
|
"index-expr base must be a vector or string type")
|
2013-01-10 23:23:07 -06:00
|
|
|
};
|
2012-08-09 18:05:34 -05:00
|
|
|
|
2013-02-08 03:52:55 -06:00
|
|
|
let len = llvm::LLVMConstIntGetZExtValue(len) as u64;
|
2013-01-10 23:23:07 -06:00
|
|
|
let len = match ty::get(bt).sty {
|
2014-04-26 17:19:15 -05:00
|
|
|
ty::ty_uniq(ty) | ty::ty_rptr(_, ty::mt{ty, ..}) => match ty::get(ty).sty {
|
2014-04-28 20:10:23 -05:00
|
|
|
ty::ty_str => {
|
2014-04-26 17:19:15 -05:00
|
|
|
assert!(len > 0);
|
|
|
|
len - 1
|
|
|
|
}
|
|
|
|
_ => len
|
|
|
|
},
|
2013-01-10 23:23:07 -06:00
|
|
|
_ => len
|
|
|
|
};
|
|
|
|
if iv >= len {
|
|
|
|
// FIXME #3170: report this earlier on in the const-eval
|
|
|
|
// pass. Reporting here is a bit late.
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().span_err(e.span,
|
|
|
|
"const index-expr is out of bounds");
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2013-09-11 12:06:16 -05:00
|
|
|
(const_get_elt(cx, arr, [iv as c_uint]), inlineable)
|
2012-08-08 21:41:50 -05:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprCast(ref base, _) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-01-10 23:23:07 -06:00
|
|
|
let llty = type_of::type_of(cx, ety);
|
2014-05-16 12:15:33 -05:00
|
|
|
let basety = ty::expr_ty(cx.tcx(), &**base);
|
|
|
|
let (v, inlineable) = const_expr(cx, &**base, is_local);
|
2013-09-11 12:06:16 -05:00
|
|
|
return (match (expr::cast_type_kind(basety),
|
|
|
|
expr::cast_type_kind(ety)) {
|
2012-08-06 14:34:08 -05:00
|
|
|
|
2013-01-10 23:23:07 -06:00
|
|
|
(expr::cast_integral, expr::cast_integral) => {
|
2013-03-01 03:40:23 -06:00
|
|
|
let s = ty::type_is_signed(basety) as Bool;
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstIntCast(v, llty.to_ref(), s)
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
(expr::cast_integral, expr::cast_float) => {
|
|
|
|
if ty::type_is_signed(basety) {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstSIToFP(v, llty.to_ref())
|
2013-01-10 23:23:07 -06:00
|
|
|
} else {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstUIToFP(v, llty.to_ref())
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
(expr::cast_float, expr::cast_float) => {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstFPCast(v, llty.to_ref())
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
(expr::cast_float, expr::cast_integral) => {
|
2013-06-16 05:52:44 -05:00
|
|
|
if ty::type_is_signed(ety) { llvm::LLVMConstFPToSI(v, llty.to_ref()) }
|
|
|
|
else { llvm::LLVMConstFPToUI(v, llty.to_ref()) }
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2013-03-01 03:40:23 -06:00
|
|
|
(expr::cast_enum, expr::cast_integral) |
|
|
|
|
(expr::cast_enum, expr::cast_float) => {
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, basety);
|
2014-04-21 19:03:02 -05:00
|
|
|
let discr = adt::const_get_discrim(cx, &*repr, v);
|
2013-08-02 14:24:21 -05:00
|
|
|
let iv = C_integral(cx.int_type, discr, false);
|
2013-03-01 03:40:23 -06:00
|
|
|
let ety_cast = expr::cast_type_kind(ety);
|
|
|
|
match ety_cast {
|
|
|
|
expr::cast_integral => {
|
|
|
|
let s = ty::type_is_signed(ety) as Bool;
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstIntCast(iv, llty.to_ref(), s)
|
2013-03-01 03:40:23 -06:00
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().bug("enum cast destination is not \
|
2014-06-13 13:14:04 -05:00
|
|
|
integral")
|
2013-03-01 03:40:23 -06:00
|
|
|
}
|
|
|
|
}
|
2013-03-04 17:22:03 -06:00
|
|
|
(expr::cast_pointer, expr::cast_pointer) => {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstPointerCast(v, llty.to_ref())
|
2013-03-04 17:22:03 -06:00
|
|
|
}
|
2013-03-07 14:35:43 -06:00
|
|
|
(expr::cast_integral, expr::cast_pointer) => {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMConstIntToPtr(v, llty.to_ref())
|
2013-03-07 14:35:43 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
_ => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().impossible_case(e.span,
|
|
|
|
"bad combination of types for cast")
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2013-09-11 12:06:16 -05:00
|
|
|
}, inlineable)
|
2012-08-03 23:44:42 -05:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprAddrOf(ast::MutImmutable, ref sub) => {
|
|
|
|
let (e, _) = const_expr(cx, &**sub, is_local);
|
2013-09-11 12:06:16 -05:00
|
|
|
(const_addr_of(cx, e), false)
|
2012-08-03 23:44:42 -05:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::ExprTup(ref es) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, ety);
|
2014-02-28 17:25:15 -06:00
|
|
|
let (vals, inlineable) = map_list(es.as_slice());
|
2014-04-21 19:03:02 -05:00
|
|
|
(adt::trans_const(cx, &*repr, 0, vals.as_slice()), inlineable)
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::ExprStruct(_, ref fs, ref base_opt) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, ety);
|
2014-03-15 15:29:34 -05:00
|
|
|
let tcx = cx.tcx();
|
2013-07-26 22:49:42 -05:00
|
|
|
|
|
|
|
let base_val = match *base_opt {
|
2014-05-16 12:15:33 -05:00
|
|
|
Some(ref base) => Some(const_expr(cx, &**base, is_local)),
|
2013-07-26 22:49:42 -05:00
|
|
|
None => None
|
|
|
|
};
|
|
|
|
|
2013-11-21 17:42:55 -06:00
|
|
|
expr::with_field_tys(tcx, ety, Some(e.id), |discr, field_tys| {
|
2014-05-03 20:10:04 -05:00
|
|
|
let (cs, inlineable) = vec::unzip(field_tys.iter().enumerate()
|
2013-08-09 22:09:47 -05:00
|
|
|
.map(|(ix, &field_ty)| {
|
2013-10-28 21:22:42 -05:00
|
|
|
match fs.iter().find(|f| field_ty.ident.name == f.ident.node.name) {
|
2014-05-16 12:15:33 -05:00
|
|
|
Some(ref f) => const_expr(cx, &*f.expr, is_local),
|
2013-01-10 23:23:07 -06:00
|
|
|
None => {
|
2013-07-26 22:49:42 -05:00
|
|
|
match base_val {
|
2013-09-11 12:06:16 -05:00
|
|
|
Some((bv, inlineable)) => {
|
2014-04-21 19:03:02 -05:00
|
|
|
(adt::const_get_field(cx, &*repr, bv, discr, ix),
|
2013-09-11 12:06:16 -05:00
|
|
|
inlineable)
|
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
None => cx.sess().span_bug(e.span, "missing struct field")
|
2013-07-26 22:49:42 -05:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
}
|
2014-03-22 18:18:37 -05:00
|
|
|
}));
|
2014-05-04 01:43:38 -05:00
|
|
|
(adt::trans_const(cx, &*repr, discr, cs.as_slice()),
|
2013-09-11 12:06:16 -05:00
|
|
|
inlineable.iter().fold(true, |a, &b| a && b))
|
2013-11-21 17:42:55 -06:00
|
|
|
})
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-04-04 05:12:18 -05:00
|
|
|
ast::ExprVec(ref es) => {
|
2014-02-28 17:25:15 -06:00
|
|
|
let (v, _, inlineable) = const_vec(cx,
|
|
|
|
e,
|
|
|
|
es.as_slice(),
|
|
|
|
is_local);
|
2013-09-11 12:06:16 -05:00
|
|
|
(v, inlineable)
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprVstore(ref sub, store @ ast::ExprVstoreSlice) |
|
|
|
|
ast::ExprVstore(ref sub, store @ ast::ExprVstoreMutSlice) => {
|
2013-03-20 00:17:42 -05:00
|
|
|
match sub.node {
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::ExprLit(ref lit) => {
|
2013-01-10 23:23:07 -06:00
|
|
|
match lit.node {
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::LitStr(..) => { const_expr(cx, &**sub, is_local) }
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => { cx.sess().span_bug(e.span, "bad const-slice lit") }
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
}
|
2014-04-04 05:12:18 -05:00
|
|
|
ast::ExprVec(ref es) => {
|
2014-02-28 17:25:15 -06:00
|
|
|
let (cv, llunitty, _) = const_vec(cx,
|
|
|
|
e,
|
|
|
|
es.as_slice(),
|
|
|
|
is_local);
|
2013-01-10 23:23:07 -06:00
|
|
|
let llty = val_ty(cv);
|
2013-11-21 17:42:55 -06:00
|
|
|
let gv = "const".with_c_str(|name| {
|
2013-06-16 05:52:44 -05:00
|
|
|
llvm::LLVMAddGlobal(cx.llmod, llty.to_ref(), name)
|
2013-11-21 17:42:55 -06:00
|
|
|
});
|
2013-01-10 23:23:07 -06:00
|
|
|
llvm::LLVMSetInitializer(gv, cv);
|
2014-01-25 11:48:57 -06:00
|
|
|
llvm::LLVMSetGlobalConstant(gv,
|
|
|
|
if store == ast::ExprVstoreMutSlice { False } else { True });
|
2013-03-28 20:15:35 -05:00
|
|
|
SetLinkage(gv, PrivateLinkage);
|
2013-01-10 23:23:07 -06:00
|
|
|
let p = const_ptrcast(cx, gv, llunitty);
|
2014-03-15 15:29:34 -05:00
|
|
|
(C_struct(cx, [p, C_uint(cx, es.len())], false), false)
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(e.span, "bad const-slice expr")
|
2012-11-30 15:25:43 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprRepeat(ref elem, ref count) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let vec_ty = ty::expr_ty(cx.tcx(), e);
|
|
|
|
let unit_ty = ty::sequence_element_type(cx.tcx(), vec_ty);
|
2013-08-12 20:10:29 -05:00
|
|
|
let llunitty = type_of::type_of(cx, unit_ty);
|
2014-05-16 12:15:33 -05:00
|
|
|
let n = match const_eval::eval_const_expr(cx.tcx(), &**count) {
|
2013-08-12 20:10:29 -05:00
|
|
|
const_eval::const_int(i) => i as uint,
|
|
|
|
const_eval::const_uint(i) => i as uint,
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(count.span, "count must be integral const expression.")
|
2013-08-12 20:10:29 -05:00
|
|
|
};
|
2014-05-16 12:15:33 -05:00
|
|
|
let vs = Vec::from_elem(n, const_expr(cx, &**elem, is_local).val0());
|
2013-08-12 20:10:29 -05:00
|
|
|
let v = if vs.iter().any(|vi| val_ty(*vi) != llunitty) {
|
2014-04-17 17:59:07 -05:00
|
|
|
C_struct(cx, vs.as_slice(), false)
|
2013-08-12 20:10:29 -05:00
|
|
|
} else {
|
2014-04-17 17:59:07 -05:00
|
|
|
C_array(llunitty, vs.as_slice())
|
2013-08-12 20:10:29 -05:00
|
|
|
};
|
2013-09-11 12:06:16 -05:00
|
|
|
(v, true)
|
2013-08-12 20:10:29 -05:00
|
|
|
}
|
2013-09-01 20:45:37 -05:00
|
|
|
ast::ExprPath(ref pth) => {
|
2013-08-07 11:47:28 -05:00
|
|
|
// Assert that there are no type parameters in this path.
|
|
|
|
assert!(pth.segments.iter().all(|seg| seg.types.is_empty()));
|
|
|
|
|
2014-03-20 21:49:20 -05:00
|
|
|
let opt_def = cx.tcx().def_map.borrow().find_copy(&e.id);
|
2013-12-23 13:15:16 -06:00
|
|
|
match opt_def {
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefFn(def_id, _fn_style)) => {
|
2013-03-07 14:34:58 -06:00
|
|
|
if !ast_util::is_local(def_id) {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ty = csearch::get_type(cx.tcx(), def_id).ty;
|
2013-09-11 12:06:16 -05:00
|
|
|
(base::trans_external_path(cx, def_id, ty), true)
|
2013-03-07 00:30:20 -06:00
|
|
|
} else {
|
2013-03-28 20:39:09 -05:00
|
|
|
assert!(ast_util::is_local(def_id));
|
2013-09-11 12:06:16 -05:00
|
|
|
(base::get_item_val(cx, def_id.node), true)
|
2013-01-18 09:12:07 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefStatic(def_id, false)) => {
|
2013-01-10 23:23:07 -06:00
|
|
|
get_const_val(cx, def_id)
|
2012-11-30 15:25:43 -06:00
|
|
|
}
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefVariant(enum_did, variant_did, _)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, ety);
|
2014-03-15 15:29:34 -05:00
|
|
|
let vinfo = ty::enum_variant_with_id(cx.tcx(),
|
2013-02-24 16:38:59 -06:00
|
|
|
enum_did,
|
|
|
|
variant_did);
|
2014-04-21 19:03:02 -05:00
|
|
|
(adt::trans_const(cx, &*repr, vinfo.disr_val, []), true)
|
2013-01-07 05:42:49 -06:00
|
|
|
}
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefStruct(_)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-01-10 23:23:07 -06:00
|
|
|
let llty = type_of::type_of(cx, ety);
|
2013-09-11 12:06:16 -05:00
|
|
|
(C_null(llty), true)
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
|
|
|
_ => {
|
2014-03-05 08:36:01 -06:00
|
|
|
cx.sess().span_bug(e.span, "expected a const, fn, struct, or variant def")
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2012-11-30 15:25:43 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-02-14 02:28:32 -06:00
|
|
|
ast::ExprCall(callee, ref args) => {
|
2014-03-20 21:49:20 -05:00
|
|
|
let opt_def = cx.tcx().def_map.borrow().find_copy(&callee.id);
|
2013-12-23 13:15:16 -06:00
|
|
|
match opt_def {
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefStruct(_)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, ety);
|
2014-02-28 17:25:15 -06:00
|
|
|
let (arg_vals, inlineable) = map_list(args.as_slice());
|
2014-04-21 19:03:02 -05:00
|
|
|
(adt::trans_const(cx, &*repr, 0, arg_vals.as_slice()),
|
2014-03-08 14:36:22 -06:00
|
|
|
inlineable)
|
2013-02-24 16:38:59 -06:00
|
|
|
}
|
2014-05-14 14:31:30 -05:00
|
|
|
Some(def::DefVariant(enum_did, variant_did, _)) => {
|
2014-03-15 15:29:34 -05:00
|
|
|
let ety = ty::expr_ty(cx.tcx(), e);
|
2013-02-24 16:38:59 -06:00
|
|
|
let repr = adt::represent_type(cx, ety);
|
2014-03-15 15:29:34 -05:00
|
|
|
let vinfo = ty::enum_variant_with_id(cx.tcx(),
|
2013-02-24 16:38:59 -06:00
|
|
|
enum_did,
|
|
|
|
variant_did);
|
2014-02-28 17:25:15 -06:00
|
|
|
let (arg_vals, inlineable) = map_list(args.as_slice());
|
2014-03-08 14:36:22 -06:00
|
|
|
(adt::trans_const(cx,
|
2014-04-21 19:03:02 -05:00
|
|
|
&*repr,
|
2014-03-08 14:36:22 -06:00
|
|
|
vinfo.disr_val,
|
|
|
|
arg_vals.as_slice()), inlineable)
|
2013-02-24 16:38:59 -06:00
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(e.span, "expected a struct or variant def")
|
2013-02-24 16:38:59 -06:00
|
|
|
}
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2014-05-16 12:15:33 -05:00
|
|
|
ast::ExprParen(ref e) => { const_expr(cx, &**e, is_local) }
|
2014-05-04 03:39:11 -05:00
|
|
|
ast::ExprBlock(ref block) => {
|
|
|
|
match block.expr {
|
|
|
|
Some(ref expr) => const_expr(cx, &**expr, is_local),
|
|
|
|
None => (C_nil(cx), true)
|
|
|
|
}
|
|
|
|
}
|
2014-03-05 08:36:01 -06:00
|
|
|
_ => cx.sess().span_bug(e.span,
|
2013-05-02 11:28:53 -05:00
|
|
|
"bad constant expression type in consts::const_expr")
|
2013-01-10 23:23:07 -06:00
|
|
|
};
|
|
|
|
}
|
2012-07-31 20:34:36 -05:00
|
|
|
}
|
|
|
|
|
2014-03-06 10:47:24 -06:00
|
|
|
pub fn trans_const(ccx: &CrateContext, m: ast::Mutability, id: ast::NodeId) {
|
2013-01-10 23:23:07 -06:00
|
|
|
unsafe {
|
2013-06-16 23:23:24 -05:00
|
|
|
let _icx = push_ctxt("trans_const");
|
2013-01-10 23:23:07 -06:00
|
|
|
let g = base::get_item_val(ccx, id);
|
2012-12-30 22:30:23 -06:00
|
|
|
// At this point, get_item_val has already translated the
|
|
|
|
// constant's initializer to determine its LLVM type.
|
2014-03-20 21:49:20 -05:00
|
|
|
let v = ccx.const_values.borrow().get_copy(&id);
|
2013-01-10 23:23:07 -06:00
|
|
|
llvm::LLVMSetInitializer(g, v);
|
2013-09-01 20:45:37 -05:00
|
|
|
if m != ast::MutMutable {
|
2013-06-21 20:46:34 -05:00
|
|
|
llvm::LLVMSetGlobalConstant(g, True);
|
|
|
|
}
|
2014-02-20 19:44:29 -06:00
|
|
|
debuginfo::create_global_var_metadata(ccx, id, g);
|
2013-01-10 23:23:07 -06:00
|
|
|
}
|
2012-07-31 20:34:36 -05:00
|
|
|
}
|