import driver::session::{session, arch_x86_64}; import syntax::codemap::span; import libc::c_uint; import front::attr; import lib::llvm::{ llvm, TypeRef, ValueRef, ModuleRef, CallConv, Attribute, StructRetAttribute, ByValAttribute }; import syntax::{ast, ast_util}; import back::link; import common::*; import build::*; import base::*; import type_of::*; import std::map::hashmap; import util::ppaux::ty_to_str; export link_name, trans_native_mod, register_crust_fn, trans_crust_fn, trans_intrinsic; enum x86_64_reg_class { no_class, integer_class, sse_fs_class, sse_fv_class, sse_ds_class, sse_dv_class, sse_int_class, sseup_class, x87_class, x87up_class, complex_x87_class, memory_class } fn is_sse(c: x86_64_reg_class) -> bool { ret alt c { sse_fs_class | sse_fv_class | sse_ds_class | sse_dv_class { true } _ { false } }; } fn is_ymm(cls: [x86_64_reg_class]) -> bool { let len = vec::len(cls); ret (len > 2u && is_sse(cls[0]) && cls[1] == sseup_class && cls[2] == sseup_class) || (len > 3u && is_sse(cls[1]) && cls[2] == sseup_class && cls[3] == sseup_class); } fn classify_ty(ty: TypeRef) -> [x86_64_reg_class] { fn align(off: uint, ty: TypeRef) -> uint { let a = ty_align(ty); ret (off + a - 1u) / a * a; } fn struct_tys(ty: TypeRef) -> [TypeRef] { let n = llvm::LLVMCountStructElementTypes(ty); let elts = vec::from_elem(n as uint, ptr::null()); vec::as_buf(elts) {|buf| llvm::LLVMGetStructElementTypes(ty, buf); } ret elts; } fn ty_align(ty: TypeRef) -> uint { ret alt llvm::LLVMGetTypeKind(ty) as int { 8 /* integer */ { ((llvm::LLVMGetIntTypeWidth(ty) as uint) + 7u) / 8u } 12 /* pointer */ { 8u } 2 /* float */ { 4u } 3 /* double */ { 8u } 10 /* struct */ { vec::foldl(0u, struct_tys(ty)) {|a, t| uint::max(a, ty_align(t)) } } _ { fail "ty_size: unhandled type" } }; } fn ty_size(ty: TypeRef) -> uint { ret alt llvm::LLVMGetTypeKind(ty) as int { 8 /* integer */ { ((llvm::LLVMGetIntTypeWidth(ty) as uint) + 7u) / 8u } 12 /* pointer */ { 8u } 2 /* float */ { 4u } 3 /* double */ { 8u } 10 /* struct */ { vec::foldl(0u, struct_tys(ty)) {|s, t| s + ty_size(t) } } _ { fail "ty_size: unhandled type" } }; } fn all_mem(cls: [mut x86_64_reg_class]) { vec::iteri(cls) {|i, _c| cls[i] = memory_class; } } fn unify(cls: [mut x86_64_reg_class], i: uint, newv: x86_64_reg_class) { if cls[i] == newv { ret; } else if cls[i] == no_class { cls[i] = newv; } else if newv == no_class { ret; } else if cls[i] == memory_class || newv == memory_class { cls[i] = memory_class; } else if cls[i] == integer_class || newv == integer_class { cls[i] = integer_class; } else if cls[i] == x87_class || cls[i] == x87up_class || cls[i] == complex_x87_class || newv == x87_class || newv == x87up_class || newv == complex_x87_class { cls[i] = memory_class; } else { cls[i] = newv; } } fn classify_struct(tys: [TypeRef], cls: [mut x86_64_reg_class], i: uint, off: uint) { if vec::is_empty(tys) { classify(T_i64(), cls, i, off); } else { let mut field_off = off; for ty in tys { field_off = align(field_off, ty); classify(ty, cls, i, field_off); field_off += ty_size(ty); } } } fn classify(ty: TypeRef, cls: [mut x86_64_reg_class], i: uint, off: uint) { let t_align = ty_align(ty); let t_size = ty_size(ty); let misalign = off % t_align; if misalign != 0u { let mut i = off / 8u; let e = (off + t_size + 7u) / 8u; while i < e { unify(cls, i, memory_class); i += 1u; } ret; } alt llvm::LLVMGetTypeKind(ty) as int { 8 /* integer */ | 12 /* pointer */ { unify(cls, off / 8u, integer_class); } 2 /* float */ { if off % 8u == 4u { unify(cls, off / 8u, sse_fv_class); } else { unify(cls, off / 8u, sse_fs_class); } } 3 /* double */ { unify(cls, off / 8u, sse_ds_class); } 10 /* struct */ { classify_struct(struct_tys(ty), cls, i, off); } _ { fail "classify: unhandled type"; } } } fn fixup(ty: TypeRef, cls: [mut x86_64_reg_class]) { let mut i = 0u; let e = vec::len(cls); if vec::len(cls) > 2u && llvm::LLVMGetTypeKind(ty) as int == 10 /* struct */ { if is_sse(cls[i]) { i += 1u; while i < e { if cls[i] != sseup_class { all_mem(cls); ret; } i += 1u; } } else { all_mem(cls); ret } } else { while i < e { if cls[i] == memory_class { all_mem(cls); ret; } if cls[i] == x87up_class { // for darwin // cls[i] = sse_ds_class; all_mem(cls); ret; } if cls[i] == sseup_class { cls[i] = sse_int_class; } else if is_sse(cls[i]) { i += 1u; while cls[i] == sseup_class { i += 1u; } } else if cls[i] == x87_class { i += 1u; while cls[i] == x87up_class { i += 1u; } } else { i += 1u; } } } } let words = (ty_size(ty) + 7u) / 8u; let cls = vec::to_mut(vec::from_elem(words, no_class)); if words > 4u { all_mem(cls); ret vec::from_mut(cls); } classify(ty, cls, 0u, 0u); fixup(ty, cls); ret vec::from_mut(cls); } fn llreg_ty(cls: [x86_64_reg_class]) -> TypeRef { fn llvec_len(cls: [x86_64_reg_class]) -> uint { let mut len = 1u; for c in cls { if c != sseup_class { break; } len += 1u; } ret len; } let mut tys = []; let mut i = 0u; let e = vec::len(cls); while i < e { alt cls[i] { integer_class { tys += [T_i64()]; } sse_fv_class { let vec_len = llvec_len(vec::tailn(cls, i + 1u)) * 2u; let vec_ty = llvm::LLVMVectorType(T_f32(), vec_len as c_uint); tys += [vec_ty]; i += vec_len; cont; } sse_fs_class { tys += [T_f32()]; } sse_ds_class { tys += [T_f64()]; } _ { fail "llregtype: unhandled class"; } } i += 1u; } ret T_struct(tys); } type x86_64_llty = { cast: bool, ty: TypeRef }; type x86_64_tys = { arg_tys: [x86_64_llty], ret_ty: x86_64_llty, attrs: [option], sret: bool }; fn x86_64_tys(atys: [TypeRef], rty: TypeRef, ret_def: bool) -> x86_64_tys { fn is_reg_ty(ty: TypeRef) -> bool { ret alt llvm::LLVMGetTypeKind(ty) as int { 8 /* integer */ | 12 /* pointer */ | 2 /* float */ | 3 /* double */ { true } _ { false } }; } fn is_pass_byval(cls: [x86_64_reg_class]) -> bool { ret cls[0] == memory_class || cls[0] == x87_class || cls[0] == complex_x87_class; } fn is_ret_bysret(cls: [x86_64_reg_class]) -> bool { ret cls[0] == memory_class; } fn x86_64_ty(ty: TypeRef, is_mem_cls: fn(cls: [x86_64_reg_class]) -> bool, attr: Attribute) -> (x86_64_llty, option) { let mut cast = false; let mut ty_attr = option::none; let mut llty = ty; if !is_reg_ty(ty) { let cls = classify_ty(ty); if is_mem_cls(cls) { llty = T_ptr(ty); ty_attr = option::some(attr); } else { cast = true; llty = llreg_ty(cls); } } ret ({ cast: cast, ty: llty }, ty_attr); } let mut arg_tys = []; let mut attrs = []; for t in atys { let (ty, attr) = x86_64_ty(t, is_pass_byval, ByValAttribute); arg_tys += [ty]; attrs += [attr]; } let mut (ret_ty, ret_attr) = x86_64_ty(rty, is_ret_bysret, StructRetAttribute); let sret = option::is_some(ret_attr); if sret { arg_tys = [ret_ty] + arg_tys; ret_ty = { cast: false, ty: T_void() }; attrs = [ret_attr] + attrs; } else if !ret_def { ret_ty = { cast: false, ty: T_void() }; } ret { arg_tys: arg_tys, ret_ty: ret_ty, attrs: attrs, sret: sret }; } fn decl_x86_64_fn(tys: x86_64_tys, decl: fn(fnty: TypeRef) -> ValueRef) -> ValueRef { let atys = vec::map(tys.arg_tys) {|t| t.ty }; let rty = tys.ret_ty.ty; let fnty = T_fn(atys, rty); let llfn = decl(fnty); vec::iteri(tys.attrs) {|i, a| alt a { option::some(attr) { let llarg = get_param(llfn, i); llvm::LLVMAddAttribute(llarg, attr as c_uint); } _ {} } } ret llfn; } fn link_name(i: @ast::native_item) -> str { alt attr::get_meta_item_value_str_by_name(i.attrs, "link_name") { none { ret i.ident; } option::some(ln) { ret ln; } } } type c_stack_tys = { arg_tys: [TypeRef], ret_ty: TypeRef, ret_def: bool, bundle_ty: TypeRef, shim_fn_ty: TypeRef, x86_64_tys: option }; fn c_arg_and_ret_lltys(ccx: @crate_ctxt, id: ast::node_id) -> ([TypeRef], TypeRef, ty::t) { alt ty::get(ty::node_id_to_type(ccx.tcx, id)).struct { ty::ty_fn({inputs: arg_tys, output: ret_ty, _}) { let llargtys = type_of_explicit_args(ccx, arg_tys); let llretty = type_of::type_of(ccx, ret_ty); (llargtys, llretty, ret_ty) } _ { ccx.sess.bug("c_arg_and_ret_lltys called on non-function type"); } } } fn c_stack_tys(ccx: @crate_ctxt, id: ast::node_id) -> @c_stack_tys { let (llargtys, llretty, ret_ty) = c_arg_and_ret_lltys(ccx, id); let bundle_ty = T_struct(llargtys + [T_ptr(llretty)]); let ret_def = !ty::type_is_bot(ret_ty) && !ty::type_is_nil(ret_ty); let x86_64 = if ccx.sess.targ_cfg.arch == arch_x86_64 { option::some(x86_64_tys(llargtys, llretty, ret_def)) } else { option::none }; ret @{ arg_tys: llargtys, ret_ty: llretty, ret_def: ret_def, bundle_ty: bundle_ty, shim_fn_ty: T_fn([T_ptr(bundle_ty)], T_void()), x86_64_tys: x86_64 }; } type shim_arg_builder = fn(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef) -> [ValueRef]; type shim_ret_builder = fn(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef, llretval: ValueRef); fn build_shim_fn_(ccx: @crate_ctxt, shim_name: str, llbasefn: ValueRef, tys: @c_stack_tys, cc: lib::llvm::CallConv, arg_builder: shim_arg_builder, ret_builder: shim_ret_builder) -> ValueRef { let llshimfn = decl_internal_cdecl_fn( ccx.llmod, shim_name, tys.shim_fn_ty); // Declare the body of the shim function: let fcx = new_fn_ctxt(ccx, [], llshimfn, none); let bcx = top_scope_block(fcx, none); let lltop = bcx.llbb; let llargbundle = get_param(llshimfn, 0u); let llargvals = arg_builder(bcx, tys, llargbundle); // Create the call itself and store the return value: let llretval = CallWithConv(bcx, llbasefn, llargvals, cc); // r ret_builder(bcx, tys, llargbundle, llretval); build_return(bcx); finish_fn(fcx, lltop); ret llshimfn; } type wrap_arg_builder = fn(bcx: block, tys: @c_stack_tys, llwrapfn: ValueRef, llargbundle: ValueRef); type wrap_ret_builder = fn(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef); fn build_wrap_fn_(ccx: @crate_ctxt, tys: @c_stack_tys, llshimfn: ValueRef, llwrapfn: ValueRef, shim_upcall: ValueRef, arg_builder: wrap_arg_builder, ret_builder: wrap_ret_builder) { let _icx = ccx.insn_ctxt("native::build_wrap_fn_"); let fcx = new_fn_ctxt(ccx, [], llwrapfn, none); let bcx = top_scope_block(fcx, none); let lltop = bcx.llbb; // Allocate the struct and write the arguments into it. let llargbundle = alloca(bcx, tys.bundle_ty); arg_builder(bcx, tys, llwrapfn, llargbundle); // Create call itself. let llshimfnptr = PointerCast(bcx, llshimfn, T_ptr(T_i8())); let llrawargbundle = PointerCast(bcx, llargbundle, T_ptr(T_i8())); Call(bcx, shim_upcall, [llrawargbundle, llshimfnptr]); ret_builder(bcx, tys, llargbundle); tie_up_header_blocks(fcx, lltop); // Make sure our standard return block (that we didn't use) is terminated let ret_cx = raw_block(fcx, fcx.llreturn); Unreachable(ret_cx); } // For each native function F, we generate a wrapper function W and a shim // function S that all work together. The wrapper function W is the function // that other rust code actually invokes. Its job is to marshall the // arguments into a struct. It then uses a small bit of assembly to switch // over to the C stack and invoke the shim function. The shim function S then // unpacks the arguments from the struct and invokes the actual function F // according to its specified calling convention. // // Example: Given a native c-stack function F(x: X, y: Y) -> Z, // we generate a wrapper function W that looks like: // // void W(Z* dest, void *env, X x, Y y) { // struct { X x; Y y; Z *z; } args = { x, y, z }; // call_on_c_stack_shim(S, &args); // } // // The shim function S then looks something like: // // void S(struct { X x; Y y; Z *z; } *args) { // *args->z = F(args->x, args->y); // } // // However, if the return type of F is dynamically sized or of aggregate type, // the shim function looks like: // // void S(struct { X x; Y y; Z *z; } *args) { // F(args->z, args->x, args->y); // } // // Note: on i386, the layout of the args struct is generally the same as the // desired layout of the arguments on the C stack. Therefore, we could use // upcall_alloc_c_stack() to allocate the `args` structure and switch the // stack pointer appropriately to avoid a round of copies. (In fact, the shim // function itself is unnecessary). We used to do this, in fact, and will // perhaps do so in the future. fn trans_native_mod(ccx: @crate_ctxt, native_mod: ast::native_mod, abi: ast::native_abi) { let _icx = ccx.insn_ctxt("native::trans_native_mod"); fn build_shim_fn(ccx: @crate_ctxt, native_item: @ast::native_item, tys: @c_stack_tys, cc: lib::llvm::CallConv) -> ValueRef { let _icx = ccx.insn_ctxt("native::build_shim_fn"); fn build_args(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef) -> [ValueRef] { let _icx = bcx.insn_ctxt("native::shim::build_args"); let mut llargvals = []; let mut i = 0u; let n = vec::len(tys.arg_tys); alt tys.x86_64_tys { some(x86_64) { let mut atys = x86_64.arg_tys; let mut attrs = x86_64.attrs; if x86_64.sret { let llretptr = GEPi(bcx, llargbundle, [0, n as int]); let llretloc = Load(bcx, llretptr); llargvals = [llretloc]; atys = vec::tail(atys); attrs = vec::tail(attrs); } while i < n { let llargval = if atys[i].cast { let arg_ptr = GEPi(bcx, llargbundle, [0, i as int]); let arg_ptr = BitCast(bcx, arg_ptr, T_ptr(atys[i].ty)); Load(bcx, arg_ptr) } else if option::is_some(attrs[i]) { GEPi(bcx, llargbundle, [0, i as int]) } else { load_inbounds(bcx, llargbundle, [0, i as int]) }; llargvals += [llargval]; i += 1u; } } _ { while i < n { let llargval = load_inbounds(bcx, llargbundle, [0, i as int]); llargvals += [llargval]; i += 1u; } } } ret llargvals; } fn build_ret(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef, llretval: ValueRef) { let _icx = bcx.insn_ctxt("native::shim::build_ret"); alt tys.x86_64_tys { some(x86_64) { vec::iteri(x86_64.attrs) {|i, a| alt a { some(attr) { llvm::LLVMAddInstrAttribute( llretval, (i + 1u) as c_uint, attr as c_uint); } _ {} } } if x86_64.sret || !tys.ret_def { ret; } let n = vec::len(tys.arg_tys); let llretptr = GEPi(bcx, llargbundle, [0, n as int]); let llretloc = Load(bcx, llretptr); if x86_64.ret_ty.cast { let tmp_ptr = BitCast(bcx, llretloc, T_ptr(x86_64.ret_ty.ty)); Store(bcx, llretval, tmp_ptr); } else { Store(bcx, llretval, llretloc); }; } _ { if tys.ret_def { let n = vec::len(tys.arg_tys); // R** llretptr = &args->r; let llretptr = GEPi(bcx, llargbundle, [0, n as int]); // R* llretloc = *llretptr; /* (args->r) */ let llretloc = Load(bcx, llretptr); // *args->r = r; Store(bcx, llretval, llretloc); } } } } let lname = link_name(native_item); let llbasefn = base_fn(ccx, lname, tys, cc); // Name the shim function let shim_name = lname + "__c_stack_shim"; ret build_shim_fn_(ccx, shim_name, llbasefn, tys, cc, build_args, build_ret); } fn base_fn(ccx: @crate_ctxt, lname: str, tys: @c_stack_tys, cc: lib::llvm::CallConv) -> ValueRef { // Declare the "prototype" for the base function F: alt tys.x86_64_tys { some(x86_64) { decl_x86_64_fn(x86_64) {|fnty| decl_fn(ccx.llmod, lname, cc, fnty) } } _ { let llbasefnty = T_fn(tys.arg_tys, tys.ret_ty); decl_fn(ccx.llmod, lname, cc, llbasefnty) } } } // FIXME this is very shaky and probably gets ABIs wrong all over // the place fn build_direct_fn(ccx: @crate_ctxt, decl: ValueRef, item: @ast::native_item, tys: @c_stack_tys, cc: lib::llvm::CallConv) { let fcx = new_fn_ctxt(ccx, [], decl, none); let bcx = top_scope_block(fcx, none), lltop = bcx.llbb; let llbasefn = base_fn(ccx, link_name(item), tys, cc); let ty = ty::lookup_item_type(ccx.tcx, ast_util::local_def(item.id)).ty; let args = vec::from_fn(ty::ty_fn_args(ty).len(), {|i| get_param(decl, i + first_real_arg) }); let retval = Call(bcx, llbasefn, args); if !ty::type_is_nil(ty::ty_fn_ret(ty)) { Store(bcx, retval, fcx.llretptr); } build_return(bcx); finish_fn(fcx, lltop); } fn build_wrap_fn(ccx: @crate_ctxt, tys: @c_stack_tys, llshimfn: ValueRef, llwrapfn: ValueRef) { let _icx = ccx.insn_ctxt("native::build_wrap_fn"); fn build_args(bcx: block, tys: @c_stack_tys, llwrapfn: ValueRef, llargbundle: ValueRef) { let _icx = bcx.insn_ctxt("native::wrap::build_args"); let mut i = 0u; let n = vec::len(tys.arg_tys); let implicit_args = first_real_arg; // ret + env while i < n { let llargval = get_param(llwrapfn, i + implicit_args); store_inbounds(bcx, llargval, llargbundle, [0, i as int]); i += 1u; } let llretptr = get_param(llwrapfn, 0u); store_inbounds(bcx, llretptr, llargbundle, [0, n as int]); } fn build_ret(bcx: block, _tys: @c_stack_tys, _llargbundle: ValueRef) { let _icx = bcx.insn_ctxt("native::wrap::build_ret"); RetVoid(bcx); } build_wrap_fn_(ccx, tys, llshimfn, llwrapfn, ccx.upcalls.call_shim_on_c_stack, build_args, build_ret); } let mut cc = alt abi { ast::native_abi_rust_intrinsic { ret; } ast::native_abi_cdecl { lib::llvm::CCallConv } ast::native_abi_stdcall { lib::llvm::X86StdcallCallConv } }; for native_item in native_mod.items { alt native_item.node { ast::native_item_fn(fn_decl, _) { let id = native_item.id; let llwrapfn = get_item_val(ccx, id); let tys = c_stack_tys(ccx, id); if attr::attrs_contains_name(native_item.attrs, "rust_stack") { build_direct_fn(ccx, llwrapfn, native_item, tys, cc); } else { let llshimfn = build_shim_fn(ccx, native_item, tys, cc); build_wrap_fn(ccx, tys, llshimfn, llwrapfn); } } } } } fn trans_intrinsic(ccx: @crate_ctxt, decl: ValueRef, item: @ast::native_item, path: ast_map::path, substs: param_substs, ref_id: option) { let fcx = new_fn_ctxt_w_id(ccx, path, decl, item.id, none, some(substs), some(item.span)); let bcx = top_scope_block(fcx, none), lltop = bcx.llbb; let tp_ty = substs.tys[0], lltp_ty = type_of::type_of(ccx, tp_ty); alt check item.ident { "size_of" { Store(bcx, C_uint(ccx, shape::llsize_of_real(ccx, lltp_ty)), fcx.llretptr); } "align_of" { Store(bcx, C_uint(ccx, shape::llalign_of_real(ccx, lltp_ty)), fcx.llretptr); } "get_tydesc" { let td = get_tydesc_simple(ccx, tp_ty); Store(bcx, PointerCast(bcx, td, T_ptr(T_nil())), fcx.llretptr); } "init" { if !ty::type_is_nil(tp_ty) { Store(bcx, C_null(lltp_ty), fcx.llretptr); } } "forget" {} "reinterpret_cast" { let llout_ty = type_of::type_of(ccx, substs.tys[1]); if shape::llsize_of_real(ccx, lltp_ty) != shape::llsize_of_real(ccx, llout_ty) { let sp = alt check ccx.tcx.items.get(option::get(ref_id)) { ast_map::node_expr(e) { e.span } }; ccx.sess.span_fatal(sp, "reinterpret_cast called on types \ with different size: " + ty_to_str(ccx.tcx, tp_ty) + " to " + ty_to_str(ccx.tcx, substs.tys[1])); } if !ty::type_is_nil(substs.tys[1]) { let cast = PointerCast(bcx, get_param(decl, first_real_arg), T_ptr(llout_ty)); Store(bcx, Load(bcx, cast), fcx.llretptr); } } "addr_of" { Store(bcx, get_param(decl, first_real_arg), fcx.llretptr); } } build_return(bcx); finish_fn(fcx, lltop); } fn trans_crust_fn(ccx: @crate_ctxt, path: ast_map::path, decl: ast::fn_decl, body: ast::blk, llwrapfn: ValueRef, id: ast::node_id) { let _icx = ccx.insn_ctxt("native::build_crust_fn"); fn build_rust_fn(ccx: @crate_ctxt, path: ast_map::path, decl: ast::fn_decl, body: ast::blk, id: ast::node_id) -> ValueRef { let _icx = ccx.insn_ctxt("native::crust::build_rust_fn"); let t = ty::node_id_to_type(ccx.tcx, id); let ps = link::mangle_internal_name_by_path( ccx, path + [ast_map::path_name("__rust_abi")]); let llty = type_of_fn_from_ty(ccx, t); let llfndecl = decl_internal_cdecl_fn(ccx.llmod, ps, llty); trans_fn(ccx, path, decl, body, llfndecl, no_self, none, id); ret llfndecl; } fn build_shim_fn(ccx: @crate_ctxt, path: ast_map::path, llrustfn: ValueRef, tys: @c_stack_tys) -> ValueRef { let _icx = ccx.insn_ctxt("native::crust::build_shim_fn"); fn build_args(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef) -> [ValueRef] { let _icx = bcx.insn_ctxt("native::crust::shim::build_args"); let mut llargvals = []; let mut i = 0u; let n = vec::len(tys.arg_tys); let llretptr = load_inbounds(bcx, llargbundle, [0, n as int]); llargvals += [llretptr]; let llenvptr = C_null(T_opaque_box_ptr(bcx.ccx())); llargvals += [llenvptr]; while i < n { let llargval = load_inbounds(bcx, llargbundle, [0, i as int]); llargvals += [llargval]; i += 1u; } ret llargvals; } fn build_ret(_bcx: block, _tys: @c_stack_tys, _llargbundle: ValueRef, _llretval: ValueRef) { // Nop. The return pointer in the Rust ABI function // is wired directly into the return slot in the shim struct } let shim_name = link::mangle_internal_name_by_path( ccx, path + [ast_map::path_name("__rust_stack_shim")]); ret build_shim_fn_(ccx, shim_name, llrustfn, tys, lib::llvm::CCallConv, build_args, build_ret); } fn build_wrap_fn(ccx: @crate_ctxt, llshimfn: ValueRef, llwrapfn: ValueRef, tys: @c_stack_tys) { let _icx = ccx.insn_ctxt("native::crust::build_wrap_fn"); fn build_args(bcx: block, tys: @c_stack_tys, llwrapfn: ValueRef, llargbundle: ValueRef) { let _icx = bcx.insn_ctxt("native::crust::wrap::build_args"); alt tys.x86_64_tys { option::some(x86_64) { let mut atys = x86_64.arg_tys; let mut attrs = x86_64.attrs; let mut j = 0u; let llretptr = if x86_64.sret { atys = vec::tail(atys); attrs = vec::tail(attrs); j = 1u; get_param(llwrapfn, 0u) } else if x86_64.ret_ty.cast { let retptr = alloca(bcx, x86_64.ret_ty.ty); BitCast(bcx, retptr, T_ptr(tys.ret_ty)) } else { alloca(bcx, tys.ret_ty) }; let mut i = 0u; let n = vec::len(atys); while i < n { let mut argval = get_param(llwrapfn, i + j); if option::is_some(attrs[i]) { argval = Load(bcx, argval); store_inbounds(bcx, argval, llargbundle, [0, i as int]); } else if atys[i].cast { let argptr = GEPi(bcx, llargbundle, [0, i as int]); let argptr = BitCast(bcx, argptr, T_ptr(atys[i].ty)); Store(bcx, argval, argptr); } else { store_inbounds(bcx, argval, llargbundle, [0, i as int]); } i += 1u; } store_inbounds(bcx, llretptr, llargbundle, [0, n as int]); } _ { let llretptr = alloca(bcx, tys.ret_ty); let mut i = 0u; let n = vec::len(tys.arg_tys); while i < n { let llargval = get_param(llwrapfn, i); store_inbounds(bcx, llargval, llargbundle, [0, i as int]); i += 1u; } store_inbounds(bcx, llretptr, llargbundle, [0, n as int]); } } } fn build_ret(bcx: block, tys: @c_stack_tys, llargbundle: ValueRef) { let _icx = bcx.insn_ctxt("native::crust::wrap::build_ret"); alt tys.x86_64_tys { option::some(x86_64) { if x86_64.sret || !tys.ret_def { RetVoid(bcx); ret; } let n = vec::len(tys.arg_tys); let llretval = load_inbounds(bcx, llargbundle, [0, n as int]); let llretval = if x86_64.ret_ty.cast { let retptr = BitCast(bcx, llretval, T_ptr(x86_64.ret_ty.ty)); Load(bcx, retptr) } else { Load(bcx, llretval) }; Ret(bcx, llretval); } _ { let n = vec::len(tys.arg_tys); let llretval = load_inbounds(bcx, llargbundle, [0, n as int]); let llretval = Load(bcx, llretval); Ret(bcx, llretval); } } } build_wrap_fn_(ccx, tys, llshimfn, llwrapfn, ccx.upcalls.call_shim_on_rust_stack, build_args, build_ret); } let tys = c_stack_tys(ccx, id); // The internal Rust ABI function - runs on the Rust stack let llrustfn = build_rust_fn(ccx, path, decl, body, id); // The internal shim function - runs on the Rust stack let llshimfn = build_shim_fn(ccx, path, llrustfn, tys); // The external C function - runs on the C stack build_wrap_fn(ccx, llshimfn, llwrapfn, tys) } fn register_crust_fn(ccx: @crate_ctxt, sp: span, path: ast_map::path, node_id: ast::node_id) -> ValueRef { let _icx = ccx.insn_ctxt("native::register_crust_fn"); let t = ty::node_id_to_type(ccx.tcx, node_id); let (llargtys, llretty, ret_ty) = c_arg_and_ret_lltys(ccx, node_id); ret if ccx.sess.targ_cfg.arch == arch_x86_64 { let ret_def = !ty::type_is_bot(ret_ty) && !ty::type_is_nil(ret_ty); let x86_64 = x86_64_tys(llargtys, llretty, ret_def); decl_x86_64_fn(x86_64) {|fnty| register_fn_fuller(ccx, sp, path, node_id, t, lib::llvm::CCallConv, fnty) } } else { let llfty = T_fn(llargtys, llretty); register_fn_fuller(ccx, sp, path, node_id, t, lib::llvm::CCallConv, llfty) } } fn abi_of_native_fn(ccx: @crate_ctxt, i: @ast::native_item) -> ast::native_abi { alt attr::get_meta_item_value_str_by_name(i.attrs, "abi") { none { alt check ccx.tcx.items.get(i.id) { ast_map::node_native_item(_, abi, _) { abi } } } some(_) { alt attr::native_abi(i.attrs) { either::right(abi) { abi } either::left(msg) { ccx.sess.span_fatal(i.span, msg); } } } } }