Emit intrinsic lazily
This commit is contained in:
parent
8801d891c4
commit
a00be50e00
@ -977,7 +977,7 @@ pub fn call_memcpy(cx: &Block, dst: ValueRef, src: ValueRef, n_bytes: ValueRef,
|
|||||||
X86 | Arm | Mips => "llvm.memcpy.p0i8.p0i8.i32",
|
X86 | Arm | Mips => "llvm.memcpy.p0i8.p0i8.i32",
|
||||||
X86_64 => "llvm.memcpy.p0i8.p0i8.i64"
|
X86_64 => "llvm.memcpy.p0i8.p0i8.i64"
|
||||||
};
|
};
|
||||||
let memcpy = ccx.intrinsics.get_copy(&key);
|
let memcpy = ccx.get_intrinsic(&key);
|
||||||
let src_ptr = PointerCast(cx, src, Type::i8p(ccx));
|
let src_ptr = PointerCast(cx, src, Type::i8p(ccx));
|
||||||
let dst_ptr = PointerCast(cx, dst, Type::i8p(ccx));
|
let dst_ptr = PointerCast(cx, dst, Type::i8p(ccx));
|
||||||
let size = IntCast(cx, n_bytes, ccx.int_type);
|
let size = IntCast(cx, n_bytes, ccx.int_type);
|
||||||
@ -1022,7 +1022,7 @@ fn memzero(b: &Builder, llptr: ValueRef, ty: Type) {
|
|||||||
X86_64 => "llvm.memset.p0i8.i64"
|
X86_64 => "llvm.memset.p0i8.i64"
|
||||||
};
|
};
|
||||||
|
|
||||||
let llintrinsicfn = ccx.intrinsics.get_copy(&intrinsic_key);
|
let llintrinsicfn = ccx.get_intrinsic(&intrinsic_key);
|
||||||
let llptr = b.pointercast(llptr, Type::i8(ccx).ptr_to());
|
let llptr = b.pointercast(llptr, Type::i8(ccx).ptr_to());
|
||||||
let llzeroval = C_u8(ccx, 0);
|
let llzeroval = C_u8(ccx, 0);
|
||||||
let size = machine::llsize_of(ccx, ty);
|
let size = machine::llsize_of(ccx, ty);
|
||||||
@ -2043,168 +2043,6 @@ pub fn p2i(ccx: &CrateContext, v: ValueRef) -> ValueRef {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
pub fn declare_intrinsics(ccx: &mut CrateContext) {
|
|
||||||
macro_rules! ifn (
|
|
||||||
($name:expr fn() -> $ret:expr) => ({
|
|
||||||
let name = $name;
|
|
||||||
// HACK(eddyb) dummy output type, shouln't affect anything.
|
|
||||||
let f = decl_cdecl_fn(ccx.llmod, name, Type::func([], &$ret), ty::mk_nil());
|
|
||||||
ccx.intrinsics.insert(name, f);
|
|
||||||
});
|
|
||||||
($name:expr fn($($arg:expr),*) -> $ret:expr) => ({
|
|
||||||
let name = $name;
|
|
||||||
// HACK(eddyb) dummy output type, shouln't affect anything.
|
|
||||||
let f = decl_cdecl_fn(ccx.llmod, name,
|
|
||||||
Type::func([$($arg),*], &$ret), ty::mk_nil());
|
|
||||||
ccx.intrinsics.insert(name, f);
|
|
||||||
})
|
|
||||||
)
|
|
||||||
macro_rules! mk_struct (
|
|
||||||
($($field_ty:expr),*) => (Type::struct_(ccx, [$($field_ty),*], false))
|
|
||||||
)
|
|
||||||
|
|
||||||
let i8p = Type::i8p(ccx);
|
|
||||||
let void = Type::void(ccx);
|
|
||||||
let i1 = Type::i1(ccx);
|
|
||||||
let t_i8 = Type::i8(ccx);
|
|
||||||
let t_i16 = Type::i16(ccx);
|
|
||||||
let t_i32 = Type::i32(ccx);
|
|
||||||
let t_i64 = Type::i64(ccx);
|
|
||||||
let t_f32 = Type::f32(ccx);
|
|
||||||
let t_f64 = Type::f64(ccx);
|
|
||||||
|
|
||||||
ifn!("llvm.memcpy.p0i8.p0i8.i32" fn(i8p, i8p, t_i32, t_i32, i1) -> void);
|
|
||||||
ifn!("llvm.memcpy.p0i8.p0i8.i64" fn(i8p, i8p, t_i64, t_i32, i1) -> void);
|
|
||||||
ifn!("llvm.memmove.p0i8.p0i8.i32" fn(i8p, i8p, t_i32, t_i32, i1) -> void);
|
|
||||||
ifn!("llvm.memmove.p0i8.p0i8.i64" fn(i8p, i8p, t_i64, t_i32, i1) -> void);
|
|
||||||
ifn!("llvm.memset.p0i8.i32" fn(i8p, t_i8, t_i32, t_i32, i1) -> void);
|
|
||||||
ifn!("llvm.memset.p0i8.i64" fn(i8p, t_i8, t_i64, t_i32, i1) -> void);
|
|
||||||
|
|
||||||
ifn!("llvm.trap" fn() -> void);
|
|
||||||
ifn!("llvm.debugtrap" fn() -> void);
|
|
||||||
ifn!("llvm.frameaddress" fn(t_i32) -> i8p);
|
|
||||||
|
|
||||||
ifn!("llvm.powi.f32" fn(t_f32, t_i32) -> t_f32);
|
|
||||||
ifn!("llvm.powi.f64" fn(t_f64, t_i32) -> t_f64);
|
|
||||||
ifn!("llvm.pow.f32" fn(t_f32, t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.pow.f64" fn(t_f64, t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.sqrt.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.sqrt.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.sin.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.sin.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.cos.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.cos.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.exp.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.exp.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.exp2.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.exp2.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.log.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.log.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.log10.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.log10.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.log2.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.log2.f64" fn(t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.fma.f32" fn(t_f32, t_f32, t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.fma.f64" fn(t_f64, t_f64, t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.fabs.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.fabs.f64" fn(t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.floor.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.floor.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.ceil.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.ceil.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.trunc.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.trunc.f64" fn(t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.rint.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.rint.f64" fn(t_f64) -> t_f64);
|
|
||||||
ifn!("llvm.nearbyint.f32" fn(t_f32) -> t_f32);
|
|
||||||
ifn!("llvm.nearbyint.f64" fn(t_f64) -> t_f64);
|
|
||||||
|
|
||||||
ifn!("llvm.ctpop.i8" fn(t_i8) -> t_i8);
|
|
||||||
ifn!("llvm.ctpop.i16" fn(t_i16) -> t_i16);
|
|
||||||
ifn!("llvm.ctpop.i32" fn(t_i32) -> t_i32);
|
|
||||||
ifn!("llvm.ctpop.i64" fn(t_i64) -> t_i64);
|
|
||||||
|
|
||||||
ifn!("llvm.ctlz.i8" fn(t_i8 , i1) -> t_i8);
|
|
||||||
ifn!("llvm.ctlz.i16" fn(t_i16, i1) -> t_i16);
|
|
||||||
ifn!("llvm.ctlz.i32" fn(t_i32, i1) -> t_i32);
|
|
||||||
ifn!("llvm.ctlz.i64" fn(t_i64, i1) -> t_i64);
|
|
||||||
|
|
||||||
ifn!("llvm.cttz.i8" fn(t_i8 , i1) -> t_i8);
|
|
||||||
ifn!("llvm.cttz.i16" fn(t_i16, i1) -> t_i16);
|
|
||||||
ifn!("llvm.cttz.i32" fn(t_i32, i1) -> t_i32);
|
|
||||||
ifn!("llvm.cttz.i64" fn(t_i64, i1) -> t_i64);
|
|
||||||
|
|
||||||
ifn!("llvm.bswap.i16" fn(t_i16) -> t_i16);
|
|
||||||
ifn!("llvm.bswap.i32" fn(t_i32) -> t_i32);
|
|
||||||
ifn!("llvm.bswap.i64" fn(t_i64) -> t_i64);
|
|
||||||
|
|
||||||
ifn!("llvm.sadd.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.sadd.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.sadd.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.sadd.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.uadd.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.uadd.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.uadd.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.uadd.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.ssub.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.ssub.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.ssub.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.ssub.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.usub.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.usub.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.usub.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.usub.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.smul.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.smul.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.smul.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.smul.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.umul.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
|
||||||
ifn!("llvm.umul.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
|
||||||
ifn!("llvm.umul.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
|
||||||
ifn!("llvm.umul.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
|
||||||
|
|
||||||
ifn!("llvm.expect.i1" fn(i1, i1) -> i1);
|
|
||||||
|
|
||||||
// Some intrinsics were introduced in later versions of LLVM, but they have
|
|
||||||
// fallbacks in libc or libm and such. Currently, all of these intrinsics
|
|
||||||
// were introduced in LLVM 3.4, so we case on that.
|
|
||||||
macro_rules! compatible_ifn (
|
|
||||||
($name:expr, $cname:ident ($($arg:expr),*) -> $ret:expr) => ({
|
|
||||||
let name = $name;
|
|
||||||
if unsafe { llvm::LLVMVersionMinor() >= 4 } {
|
|
||||||
ifn!(name fn($($arg),*) -> $ret);
|
|
||||||
} else {
|
|
||||||
let f = decl_cdecl_fn(ccx.llmod, stringify!($cname),
|
|
||||||
Type::func([$($arg),*], &$ret),
|
|
||||||
ty::mk_nil());
|
|
||||||
ccx.intrinsics.insert(name, f);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
)
|
|
||||||
|
|
||||||
compatible_ifn!("llvm.copysign.f32", copysignf(t_f32, t_f32) -> t_f32);
|
|
||||||
compatible_ifn!("llvm.copysign.f64", copysign(t_f64, t_f64) -> t_f64);
|
|
||||||
compatible_ifn!("llvm.round.f32", roundf(t_f32) -> t_f32);
|
|
||||||
compatible_ifn!("llvm.round.f64", round(t_f64) -> t_f64);
|
|
||||||
|
|
||||||
|
|
||||||
if ccx.sess().opts.debuginfo != NoDebugInfo {
|
|
||||||
ifn!("llvm.dbg.declare" fn(Type::metadata(ccx), Type::metadata(ccx)) -> void);
|
|
||||||
ifn!("llvm.dbg.value" fn(Type::metadata(ccx), t_i64, Type::metadata(ccx)) -> void);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn crate_ctxt_to_encode_parms<'r>(cx: &'r CrateContext, ie: encoder::EncodeInlinedItem<'r>)
|
pub fn crate_ctxt_to_encode_parms<'r>(cx: &'r CrateContext, ie: encoder::EncodeInlinedItem<'r>)
|
||||||
-> encoder::EncodeParams<'r> {
|
-> encoder::EncodeParams<'r> {
|
||||||
|
|
||||||
|
@ -41,7 +41,6 @@ pub struct CrateContext {
|
|||||||
pub td: TargetData,
|
pub td: TargetData,
|
||||||
pub tn: TypeNames,
|
pub tn: TypeNames,
|
||||||
pub externs: RefCell<ExternMap>,
|
pub externs: RefCell<ExternMap>,
|
||||||
pub intrinsics: HashMap<&'static str, ValueRef>,
|
|
||||||
pub item_vals: RefCell<NodeMap<ValueRef>>,
|
pub item_vals: RefCell<NodeMap<ValueRef>>,
|
||||||
pub exp_map2: resolve::ExportMap2,
|
pub exp_map2: resolve::ExportMap2,
|
||||||
pub reachable: NodeSet,
|
pub reachable: NodeSet,
|
||||||
@ -107,6 +106,8 @@ pub struct CrateContext {
|
|||||||
/// is not emitted by LLVM's GC pass when no functions use GC.
|
/// is not emitted by LLVM's GC pass when no functions use GC.
|
||||||
pub uses_gc: bool,
|
pub uses_gc: bool,
|
||||||
pub dbg_cx: Option<debuginfo::CrateDebugContext>,
|
pub dbg_cx: Option<debuginfo::CrateDebugContext>,
|
||||||
|
|
||||||
|
intrinsics: RefCell<HashMap<&'static str, ValueRef>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CrateContext {
|
impl CrateContext {
|
||||||
@ -150,7 +151,6 @@ impl CrateContext {
|
|||||||
td: td,
|
td: td,
|
||||||
tn: TypeNames::new(),
|
tn: TypeNames::new(),
|
||||||
externs: RefCell::new(HashMap::new()),
|
externs: RefCell::new(HashMap::new()),
|
||||||
intrinsics: HashMap::new(),
|
|
||||||
item_vals: RefCell::new(NodeMap::new()),
|
item_vals: RefCell::new(NodeMap::new()),
|
||||||
exp_map2: emap2,
|
exp_map2: emap2,
|
||||||
reachable: reachable,
|
reachable: reachable,
|
||||||
@ -197,6 +197,7 @@ impl CrateContext {
|
|||||||
builder: BuilderRef_res(llvm::LLVMCreateBuilderInContext(llcx)),
|
builder: BuilderRef_res(llvm::LLVMCreateBuilderInContext(llcx)),
|
||||||
uses_gc: false,
|
uses_gc: false,
|
||||||
dbg_cx: dbg_cx,
|
dbg_cx: dbg_cx,
|
||||||
|
intrinsics: RefCell::new(HashMap::new()),
|
||||||
};
|
};
|
||||||
|
|
||||||
ccx.int_type = Type::int(&ccx);
|
ccx.int_type = Type::int(&ccx);
|
||||||
@ -208,8 +209,6 @@ impl CrateContext {
|
|||||||
str_slice_ty.set_struct_body([Type::i8p(&ccx), ccx.int_type], false);
|
str_slice_ty.set_struct_body([Type::i8p(&ccx), ccx.int_type], false);
|
||||||
ccx.tn.associate_type("str_slice", &str_slice_ty);
|
ccx.tn.associate_type("str_slice", &str_slice_ty);
|
||||||
|
|
||||||
base::declare_intrinsics(&mut ccx);
|
|
||||||
|
|
||||||
if ccx.sess().count_llvm_insns() {
|
if ccx.sess().count_llvm_insns() {
|
||||||
base::init_insn_ctxt()
|
base::init_insn_ctxt()
|
||||||
}
|
}
|
||||||
@ -233,4 +232,180 @@ impl CrateContext {
|
|||||||
pub fn tydesc_type(&self) -> Type {
|
pub fn tydesc_type(&self) -> Type {
|
||||||
self.tn.find_type("tydesc").unwrap()
|
self.tn.find_type("tydesc").unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_intrinsic(&self, key: & &'static str) -> ValueRef {
|
||||||
|
match self.intrinsics.borrow().find_copy(key) {
|
||||||
|
Some(v) => return v,
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
match declare_intrinsic(self, key) {
|
||||||
|
Some(v) => return v,
|
||||||
|
None => fail!()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn declare_intrinsic(ccx: &CrateContext, key: & &'static str) -> Option<ValueRef> {
|
||||||
|
macro_rules! ifn (
|
||||||
|
($name:expr fn() -> $ret:expr) => (
|
||||||
|
if *key == $name {
|
||||||
|
let f = base::decl_cdecl_fn(ccx.llmod, $name, Type::func([], &$ret), ty::mk_nil());
|
||||||
|
ccx.intrinsics.borrow_mut().insert($name, f.clone());
|
||||||
|
return Some(f);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
($name:expr fn($($arg:expr),*) -> $ret:expr) => (
|
||||||
|
if *key == $name {
|
||||||
|
let f = base::decl_cdecl_fn(ccx.llmod, $name,
|
||||||
|
Type::func([$($arg),*], &$ret), ty::mk_nil());
|
||||||
|
ccx.intrinsics.borrow_mut().insert($name, f.clone());
|
||||||
|
return Some(f);
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
macro_rules! mk_struct (
|
||||||
|
($($field_ty:expr),*) => (Type::struct_(ccx, [$($field_ty),*], false))
|
||||||
|
)
|
||||||
|
|
||||||
|
let i8p = Type::i8p(ccx);
|
||||||
|
let void = Type::void(ccx);
|
||||||
|
let i1 = Type::i1(ccx);
|
||||||
|
let t_i8 = Type::i8(ccx);
|
||||||
|
let t_i16 = Type::i16(ccx);
|
||||||
|
let t_i32 = Type::i32(ccx);
|
||||||
|
let t_i64 = Type::i64(ccx);
|
||||||
|
let t_f32 = Type::f32(ccx);
|
||||||
|
let t_f64 = Type::f64(ccx);
|
||||||
|
|
||||||
|
ifn!("llvm.memcpy.p0i8.p0i8.i32" fn(i8p, i8p, t_i32, t_i32, i1) -> void);
|
||||||
|
ifn!("llvm.memcpy.p0i8.p0i8.i64" fn(i8p, i8p, t_i64, t_i32, i1) -> void);
|
||||||
|
ifn!("llvm.memmove.p0i8.p0i8.i32" fn(i8p, i8p, t_i32, t_i32, i1) -> void);
|
||||||
|
ifn!("llvm.memmove.p0i8.p0i8.i64" fn(i8p, i8p, t_i64, t_i32, i1) -> void);
|
||||||
|
ifn!("llvm.memset.p0i8.i32" fn(i8p, t_i8, t_i32, t_i32, i1) -> void);
|
||||||
|
ifn!("llvm.memset.p0i8.i64" fn(i8p, t_i8, t_i64, t_i32, i1) -> void);
|
||||||
|
|
||||||
|
ifn!("llvm.trap" fn() -> void);
|
||||||
|
ifn!("llvm.debugtrap" fn() -> void);
|
||||||
|
ifn!("llvm.frameaddress" fn(t_i32) -> i8p);
|
||||||
|
|
||||||
|
ifn!("llvm.powi.f32" fn(t_f32, t_i32) -> t_f32);
|
||||||
|
ifn!("llvm.powi.f64" fn(t_f64, t_i32) -> t_f64);
|
||||||
|
ifn!("llvm.pow.f32" fn(t_f32, t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.pow.f64" fn(t_f64, t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.sqrt.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.sqrt.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.sin.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.sin.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.cos.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.cos.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.exp.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.exp.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.exp2.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.exp2.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.log.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.log.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.log10.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.log10.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.log2.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.log2.f64" fn(t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.fma.f32" fn(t_f32, t_f32, t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.fma.f64" fn(t_f64, t_f64, t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.fabs.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.fabs.f64" fn(t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.floor.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.floor.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.ceil.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.ceil.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.trunc.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.trunc.f64" fn(t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.rint.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.rint.f64" fn(t_f64) -> t_f64);
|
||||||
|
ifn!("llvm.nearbyint.f32" fn(t_f32) -> t_f32);
|
||||||
|
ifn!("llvm.nearbyint.f64" fn(t_f64) -> t_f64);
|
||||||
|
|
||||||
|
ifn!("llvm.ctpop.i8" fn(t_i8) -> t_i8);
|
||||||
|
ifn!("llvm.ctpop.i16" fn(t_i16) -> t_i16);
|
||||||
|
ifn!("llvm.ctpop.i32" fn(t_i32) -> t_i32);
|
||||||
|
ifn!("llvm.ctpop.i64" fn(t_i64) -> t_i64);
|
||||||
|
|
||||||
|
ifn!("llvm.ctlz.i8" fn(t_i8 , i1) -> t_i8);
|
||||||
|
ifn!("llvm.ctlz.i16" fn(t_i16, i1) -> t_i16);
|
||||||
|
ifn!("llvm.ctlz.i32" fn(t_i32, i1) -> t_i32);
|
||||||
|
ifn!("llvm.ctlz.i64" fn(t_i64, i1) -> t_i64);
|
||||||
|
|
||||||
|
ifn!("llvm.cttz.i8" fn(t_i8 , i1) -> t_i8);
|
||||||
|
ifn!("llvm.cttz.i16" fn(t_i16, i1) -> t_i16);
|
||||||
|
ifn!("llvm.cttz.i32" fn(t_i32, i1) -> t_i32);
|
||||||
|
ifn!("llvm.cttz.i64" fn(t_i64, i1) -> t_i64);
|
||||||
|
|
||||||
|
ifn!("llvm.bswap.i16" fn(t_i16) -> t_i16);
|
||||||
|
ifn!("llvm.bswap.i32" fn(t_i32) -> t_i32);
|
||||||
|
ifn!("llvm.bswap.i64" fn(t_i64) -> t_i64);
|
||||||
|
|
||||||
|
ifn!("llvm.sadd.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.sadd.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.sadd.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.sadd.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.uadd.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.uadd.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.uadd.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.uadd.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.ssub.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.ssub.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.ssub.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.ssub.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.usub.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.usub.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.usub.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.usub.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.smul.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.smul.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.smul.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.smul.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.umul.with.overflow.i8" fn(t_i8, t_i8) -> mk_struct!{t_i8, i1});
|
||||||
|
ifn!("llvm.umul.with.overflow.i16" fn(t_i16, t_i16) -> mk_struct!{t_i16, i1});
|
||||||
|
ifn!("llvm.umul.with.overflow.i32" fn(t_i32, t_i32) -> mk_struct!{t_i32, i1});
|
||||||
|
ifn!("llvm.umul.with.overflow.i64" fn(t_i64, t_i64) -> mk_struct!{t_i64, i1});
|
||||||
|
|
||||||
|
ifn!("llvm.expect.i1" fn(i1, i1) -> i1);
|
||||||
|
|
||||||
|
// Some intrinsics were introduced in later versions of LLVM, but they have
|
||||||
|
// fallbacks in libc or libm and such. Currently, all of these intrinsics
|
||||||
|
// were introduced in LLVM 3.4, so we case on that.
|
||||||
|
macro_rules! compatible_ifn (
|
||||||
|
($name:expr, $cname:ident ($($arg:expr),*) -> $ret:expr) => (
|
||||||
|
if unsafe { llvm::LLVMVersionMinor() >= 4 } {
|
||||||
|
// The `if key == $name` is already in ifn!
|
||||||
|
ifn!($name fn($($arg),*) -> $ret);
|
||||||
|
} else if *key == $name {
|
||||||
|
let f = base::decl_cdecl_fn(ccx.llmod, stringify!($cname),
|
||||||
|
Type::func([$($arg),*], &$ret),
|
||||||
|
ty::mk_nil());
|
||||||
|
ccx.intrinsics.borrow_mut().insert($name, f.clone());
|
||||||
|
return Some(f);
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
compatible_ifn!("llvm.copysign.f32", copysignf(t_f32, t_f32) -> t_f32);
|
||||||
|
compatible_ifn!("llvm.copysign.f64", copysign(t_f64, t_f64) -> t_f64);
|
||||||
|
compatible_ifn!("llvm.round.f32", roundf(t_f32) -> t_f32);
|
||||||
|
compatible_ifn!("llvm.round.f64", round(t_f64) -> t_f64);
|
||||||
|
|
||||||
|
|
||||||
|
if ccx.sess().opts.debuginfo != NoDebugInfo {
|
||||||
|
ifn!("llvm.dbg.declare" fn(Type::metadata(ccx), Type::metadata(ccx)) -> void);
|
||||||
|
ifn!("llvm.dbg.value" fn(Type::metadata(ccx), t_i64, Type::metadata(ccx)) -> void);
|
||||||
|
}
|
||||||
|
return None;
|
||||||
}
|
}
|
||||||
|
@ -513,7 +513,7 @@ fn trans_index<'a>(bcx: &'a Block<'a>,
|
|||||||
debug!("trans_index: len {}", bcx.val_to_str(len));
|
debug!("trans_index: len {}", bcx.val_to_str(len));
|
||||||
|
|
||||||
let bounds_check = ICmp(bcx, lib::llvm::IntUGE, ix_val, len);
|
let bounds_check = ICmp(bcx, lib::llvm::IntUGE, ix_val, len);
|
||||||
let expect = ccx.intrinsics.get_copy(&("llvm.expect.i1"));
|
let expect = ccx.get_intrinsic(&("llvm.expect.i1"));
|
||||||
let expected = Call(bcx, expect, [bounds_check, C_i1(ccx, false)], []);
|
let expected = Call(bcx, expect, [bounds_check, C_i1(ccx, false)], []);
|
||||||
let bcx = with_cond(bcx, expected, |bcx| {
|
let bcx = with_cond(bcx, expected, |bcx| {
|
||||||
controlflow::trans_fail_bounds_check(bcx, index_expr.span, ix_val, len)
|
controlflow::trans_fail_bounds_check(bcx, index_expr.span, ix_val, len)
|
||||||
|
@ -79,7 +79,7 @@ pub fn get_simple_intrinsic(ccx: &CrateContext, item: &ast::ForeignItem) -> Opti
|
|||||||
"bswap64" => "llvm.bswap.i64",
|
"bswap64" => "llvm.bswap.i64",
|
||||||
_ => return None
|
_ => return None
|
||||||
};
|
};
|
||||||
Some(ccx.intrinsics.get_copy(&name))
|
Some(ccx.get_intrinsic(&name))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn trans_intrinsic(ccx: &CrateContext,
|
pub fn trans_intrinsic(ccx: &CrateContext,
|
||||||
@ -93,7 +93,7 @@ pub fn trans_intrinsic(ccx: &CrateContext,
|
|||||||
let first_real_arg = bcx.fcx.arg_pos(0u);
|
let first_real_arg = bcx.fcx.arg_pos(0u);
|
||||||
let a = get_param(bcx.fcx.llfn, first_real_arg);
|
let a = get_param(bcx.fcx.llfn, first_real_arg);
|
||||||
let b = get_param(bcx.fcx.llfn, first_real_arg + 1);
|
let b = get_param(bcx.fcx.llfn, first_real_arg + 1);
|
||||||
let llfn = bcx.ccx().intrinsics.get_copy(&name);
|
let llfn = bcx.ccx().get_intrinsic(&name);
|
||||||
|
|
||||||
// convert `i1` to a `bool`, and write to the out parameter
|
// convert `i1` to a `bool`, and write to the out parameter
|
||||||
let val = Call(bcx, llfn, [a, b], []);
|
let val = Call(bcx, llfn, [a, b], []);
|
||||||
@ -155,7 +155,7 @@ pub fn trans_intrinsic(ccx: &CrateContext,
|
|||||||
let src_ptr = PointerCast(bcx, get_param(decl, first_real_arg + 1), Type::i8p(ccx));
|
let src_ptr = PointerCast(bcx, get_param(decl, first_real_arg + 1), Type::i8p(ccx));
|
||||||
let count = get_param(decl, first_real_arg + 2);
|
let count = get_param(decl, first_real_arg + 2);
|
||||||
let volatile = C_i1(ccx, false);
|
let volatile = C_i1(ccx, false);
|
||||||
let llfn = ccx.intrinsics.get_copy(&name);
|
let llfn = ccx.get_intrinsic(&name);
|
||||||
Call(bcx, llfn, [dst_ptr, src_ptr, Mul(bcx, size, count), align, volatile], []);
|
Call(bcx, llfn, [dst_ptr, src_ptr, Mul(bcx, size, count), align, volatile], []);
|
||||||
RetVoid(bcx);
|
RetVoid(bcx);
|
||||||
}
|
}
|
||||||
@ -177,7 +177,7 @@ pub fn trans_intrinsic(ccx: &CrateContext,
|
|||||||
let val = get_param(decl, first_real_arg + 1);
|
let val = get_param(decl, first_real_arg + 1);
|
||||||
let count = get_param(decl, first_real_arg + 2);
|
let count = get_param(decl, first_real_arg + 2);
|
||||||
let volatile = C_i1(ccx, false);
|
let volatile = C_i1(ccx, false);
|
||||||
let llfn = ccx.intrinsics.get_copy(&name);
|
let llfn = ccx.get_intrinsic(&name);
|
||||||
Call(bcx, llfn, [dst_ptr, val, Mul(bcx, size, count), align, volatile], []);
|
Call(bcx, llfn, [dst_ptr, val, Mul(bcx, size, count), align, volatile], []);
|
||||||
RetVoid(bcx);
|
RetVoid(bcx);
|
||||||
}
|
}
|
||||||
@ -185,7 +185,7 @@ pub fn trans_intrinsic(ccx: &CrateContext,
|
|||||||
fn count_zeros_intrinsic(bcx: &Block, name: &'static str) {
|
fn count_zeros_intrinsic(bcx: &Block, name: &'static str) {
|
||||||
let x = get_param(bcx.fcx.llfn, bcx.fcx.arg_pos(0u));
|
let x = get_param(bcx.fcx.llfn, bcx.fcx.arg_pos(0u));
|
||||||
let y = C_i1(bcx.ccx(), false);
|
let y = C_i1(bcx.ccx(), false);
|
||||||
let llfn = bcx.ccx().intrinsics.get_copy(&name);
|
let llfn = bcx.ccx().get_intrinsic(&name);
|
||||||
let llcall = Call(bcx, llfn, [x, y], []);
|
let llcall = Call(bcx, llfn, [x, y], []);
|
||||||
Ret(bcx, llcall);
|
Ret(bcx, llcall);
|
||||||
}
|
}
|
||||||
@ -274,12 +274,12 @@ pub fn trans_intrinsic(ccx: &CrateContext,
|
|||||||
|
|
||||||
match name.get() {
|
match name.get() {
|
||||||
"abort" => {
|
"abort" => {
|
||||||
let llfn = bcx.ccx().intrinsics.get_copy(&("llvm.trap"));
|
let llfn = bcx.ccx().get_intrinsic(&("llvm.trap"));
|
||||||
Call(bcx, llfn, [], []);
|
Call(bcx, llfn, [], []);
|
||||||
Unreachable(bcx);
|
Unreachable(bcx);
|
||||||
}
|
}
|
||||||
"breakpoint" => {
|
"breakpoint" => {
|
||||||
let llfn = bcx.ccx().intrinsics.get_copy(&("llvm.debugtrap"));
|
let llfn = bcx.ccx().get_intrinsic(&("llvm.debugtrap"));
|
||||||
Call(bcx, llfn, [], []);
|
Call(bcx, llfn, [], []);
|
||||||
RetVoid(bcx);
|
RetVoid(bcx);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user