Don't use an allocator shim for #[global_allocator]
This makes it possible to use liballoc/libstd in combination with `--emit obj` if you use `#[global_allocator]`. Making it work for the default libstd allocator would require weak functions, which are not well supported on all systems.
This commit is contained in:
parent
79fa6ce7a1
commit
4ce20663f7
4 changed files with 176 additions and 159 deletions
|
@ -1,6 +1,6 @@
|
||||||
use rustc_span::symbol::{sym, Symbol};
|
use rustc_span::symbol::{sym, Symbol};
|
||||||
|
|
||||||
#[derive(Clone, Debug, Copy, HashStable_Generic)]
|
#[derive(Clone, Debug, Copy, Eq, PartialEq, HashStable_Generic)]
|
||||||
pub enum AllocatorKind {
|
pub enum AllocatorKind {
|
||||||
Global,
|
Global,
|
||||||
Default,
|
Default,
|
||||||
|
@ -9,12 +9,19 @@ pub enum AllocatorKind {
|
||||||
impl AllocatorKind {
|
impl AllocatorKind {
|
||||||
pub fn fn_name(&self, base: Symbol) -> String {
|
pub fn fn_name(&self, base: Symbol) -> String {
|
||||||
match *self {
|
match *self {
|
||||||
AllocatorKind::Global => format!("__rg_{base}"),
|
AllocatorKind::Global => format!("__rust_{base}"),
|
||||||
AllocatorKind::Default => format!("__rdl_{base}"),
|
AllocatorKind::Default => format!("__rdl_{base}"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn alloc_error_handler_name(alloc_error_handler_kind: AllocatorKind) -> &'static str {
|
||||||
|
match alloc_error_handler_kind {
|
||||||
|
AllocatorKind::Global => "__rg_oom",
|
||||||
|
AllocatorKind::Default => "__rdl_oom",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub enum AllocatorTy {
|
pub enum AllocatorTy {
|
||||||
Layout,
|
Layout,
|
||||||
Ptr,
|
Ptr,
|
||||||
|
|
|
@ -3,10 +3,11 @@
|
||||||
|
|
||||||
use crate::prelude::*;
|
use crate::prelude::*;
|
||||||
|
|
||||||
use rustc_ast::expand::allocator::{AllocatorKind, AllocatorTy, ALLOCATOR_METHODS};
|
use rustc_ast::expand::allocator::{
|
||||||
|
alloc_error_handler_name, AllocatorKind, AllocatorTy, ALLOCATOR_METHODS,
|
||||||
|
};
|
||||||
use rustc_codegen_ssa::base::allocator_kind_for_codegen;
|
use rustc_codegen_ssa::base::allocator_kind_for_codegen;
|
||||||
use rustc_session::config::OomStrategy;
|
use rustc_session::config::OomStrategy;
|
||||||
use rustc_span::symbol::sym;
|
|
||||||
|
|
||||||
/// Returns whether an allocator shim was created
|
/// Returns whether an allocator shim was created
|
||||||
pub(crate) fn codegen(
|
pub(crate) fn codegen(
|
||||||
|
@ -34,41 +35,43 @@ fn codegen_inner(
|
||||||
) {
|
) {
|
||||||
let usize_ty = module.target_config().pointer_type();
|
let usize_ty = module.target_config().pointer_type();
|
||||||
|
|
||||||
for method in ALLOCATOR_METHODS {
|
if kind == AllocatorKind::Default {
|
||||||
let mut arg_tys = Vec::with_capacity(method.inputs.len());
|
for method in ALLOCATOR_METHODS {
|
||||||
for ty in method.inputs.iter() {
|
let mut arg_tys = Vec::with_capacity(method.inputs.len());
|
||||||
match *ty {
|
for ty in method.inputs.iter() {
|
||||||
AllocatorTy::Layout => {
|
match *ty {
|
||||||
arg_tys.push(usize_ty); // size
|
AllocatorTy::Layout => {
|
||||||
arg_tys.push(usize_ty); // align
|
arg_tys.push(usize_ty); // size
|
||||||
|
arg_tys.push(usize_ty); // align
|
||||||
|
}
|
||||||
|
AllocatorTy::Ptr => arg_tys.push(usize_ty),
|
||||||
|
AllocatorTy::Usize => arg_tys.push(usize_ty),
|
||||||
|
|
||||||
|
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
||||||
}
|
}
|
||||||
AllocatorTy::Ptr => arg_tys.push(usize_ty),
|
|
||||||
AllocatorTy::Usize => arg_tys.push(usize_ty),
|
|
||||||
|
|
||||||
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
|
||||||
}
|
}
|
||||||
|
let output = match method.output {
|
||||||
|
AllocatorTy::ResultPtr => Some(usize_ty),
|
||||||
|
AllocatorTy::Unit => None,
|
||||||
|
|
||||||
|
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
||||||
|
panic!("invalid allocator output")
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let sig = Signature {
|
||||||
|
call_conv: module.target_config().default_call_conv,
|
||||||
|
params: arg_tys.iter().cloned().map(AbiParam::new).collect(),
|
||||||
|
returns: output.into_iter().map(AbiParam::new).collect(),
|
||||||
|
};
|
||||||
|
crate::common::create_wrapper_function(
|
||||||
|
module,
|
||||||
|
unwind_context,
|
||||||
|
sig,
|
||||||
|
&format!("__rust_{}", method.name),
|
||||||
|
&AllocatorKind::Default.fn_name(method.name),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
let output = match method.output {
|
|
||||||
AllocatorTy::ResultPtr => Some(usize_ty),
|
|
||||||
AllocatorTy::Unit => None,
|
|
||||||
|
|
||||||
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
|
||||||
panic!("invalid allocator output")
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let sig = Signature {
|
|
||||||
call_conv: module.target_config().default_call_conv,
|
|
||||||
params: arg_tys.iter().cloned().map(AbiParam::new).collect(),
|
|
||||||
returns: output.into_iter().map(AbiParam::new).collect(),
|
|
||||||
};
|
|
||||||
crate::common::create_wrapper_function(
|
|
||||||
module,
|
|
||||||
unwind_context,
|
|
||||||
sig,
|
|
||||||
&format!("__rust_{}", method.name),
|
|
||||||
&kind.fn_name(method.name),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let sig = Signature {
|
let sig = Signature {
|
||||||
|
@ -81,7 +84,7 @@ fn codegen_inner(
|
||||||
unwind_context,
|
unwind_context,
|
||||||
sig,
|
sig,
|
||||||
"__rust_alloc_error_handler",
|
"__rust_alloc_error_handler",
|
||||||
&alloc_error_handler_kind.fn_name(sym::oom),
|
&alloc_error_handler_name(alloc_error_handler_kind),
|
||||||
);
|
);
|
||||||
|
|
||||||
let data_id = module.declare_data(OomStrategy::SYMBOL, Linkage::Export, false, false).unwrap();
|
let data_id = module.declare_data(OomStrategy::SYMBOL, Linkage::Export, false, false).unwrap();
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
#[cfg(feature="master")]
|
#[cfg(feature="master")]
|
||||||
use gccjit::FnAttribute;
|
use gccjit::FnAttribute;
|
||||||
use gccjit::{FunctionType, GlobalKind, ToRValue};
|
use gccjit::{FunctionType, GlobalKind, ToRValue};
|
||||||
use rustc_ast::expand::allocator::{AllocatorKind, AllocatorTy, ALLOCATOR_METHODS};
|
use rustc_ast::expand::allocator::{
|
||||||
|
alloc_error_handler_name, AllocatorKind, AllocatorTy, ALLOCATOR_METHODS,
|
||||||
|
};
|
||||||
use rustc_middle::bug;
|
use rustc_middle::bug;
|
||||||
use rustc_middle::ty::TyCtxt;
|
use rustc_middle::ty::TyCtxt;
|
||||||
use rustc_session::config::OomStrategy;
|
use rustc_session::config::OomStrategy;
|
||||||
use rustc_span::symbol::sym;
|
|
||||||
|
|
||||||
use crate::GccContext;
|
use crate::GccContext;
|
||||||
|
|
||||||
|
@ -22,69 +23,71 @@ pub(crate) unsafe fn codegen(tcx: TyCtxt<'_>, mods: &mut GccContext, _module_nam
|
||||||
let i8p = i8.make_pointer();
|
let i8p = i8.make_pointer();
|
||||||
let void = context.new_type::<()>();
|
let void = context.new_type::<()>();
|
||||||
|
|
||||||
for method in ALLOCATOR_METHODS {
|
if kind == AllocatorKind::Default {
|
||||||
let mut types = Vec::with_capacity(method.inputs.len());
|
for method in ALLOCATOR_METHODS {
|
||||||
for ty in method.inputs.iter() {
|
let mut types = Vec::with_capacity(method.inputs.len());
|
||||||
match *ty {
|
for ty in method.inputs.iter() {
|
||||||
AllocatorTy::Layout => {
|
match *ty {
|
||||||
types.push(usize);
|
AllocatorTy::Layout => {
|
||||||
types.push(usize);
|
types.push(usize);
|
||||||
|
types.push(usize);
|
||||||
|
}
|
||||||
|
AllocatorTy::Ptr => types.push(i8p),
|
||||||
|
AllocatorTy::Usize => types.push(usize),
|
||||||
|
|
||||||
|
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
||||||
}
|
}
|
||||||
AllocatorTy::Ptr => types.push(i8p),
|
|
||||||
AllocatorTy::Usize => types.push(usize),
|
|
||||||
|
|
||||||
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
|
||||||
}
|
}
|
||||||
}
|
let output = match method.output {
|
||||||
let output = match method.output {
|
AllocatorTy::ResultPtr => Some(i8p),
|
||||||
AllocatorTy::ResultPtr => Some(i8p),
|
AllocatorTy::Unit => None,
|
||||||
AllocatorTy::Unit => None,
|
|
||||||
|
|
||||||
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
||||||
panic!("invalid allocator output")
|
panic!("invalid allocator output")
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let name = format!("__rust_{}", method.name);
|
||||||
|
|
||||||
|
let args: Vec<_> = types.iter().enumerate()
|
||||||
|
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
||||||
|
.collect();
|
||||||
|
let func = context.new_function(None, FunctionType::Exported, output.unwrap_or(void), &args, name, false);
|
||||||
|
|
||||||
|
if tcx.sess.target.options.default_hidden_visibility {
|
||||||
|
#[cfg(feature="master")]
|
||||||
|
func.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
||||||
|
}
|
||||||
|
if tcx.sess.must_emit_unwind_tables() {
|
||||||
|
// TODO(antoyo): emit unwind tables.
|
||||||
}
|
}
|
||||||
};
|
|
||||||
let name = format!("__rust_{}", method.name);
|
|
||||||
|
|
||||||
let args: Vec<_> = types.iter().enumerate()
|
let callee = AllocatorKind::Default.fn_name(method.name);
|
||||||
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
let args: Vec<_> = types.iter().enumerate()
|
||||||
.collect();
|
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
||||||
let func = context.new_function(None, FunctionType::Exported, output.unwrap_or(void), &args, name, false);
|
.collect();
|
||||||
|
let callee = context.new_function(None, FunctionType::Extern, output.unwrap_or(void), &args, callee, false);
|
||||||
if tcx.sess.target.options.default_hidden_visibility {
|
|
||||||
#[cfg(feature="master")]
|
#[cfg(feature="master")]
|
||||||
func.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
callee.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
||||||
}
|
|
||||||
if tcx.sess.must_emit_unwind_tables() {
|
|
||||||
// TODO(antoyo): emit unwind tables.
|
|
||||||
}
|
|
||||||
|
|
||||||
let callee = kind.fn_name(method.name);
|
let block = func.new_block("entry");
|
||||||
let args: Vec<_> = types.iter().enumerate()
|
|
||||||
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
|
||||||
.collect();
|
|
||||||
let callee = context.new_function(None, FunctionType::Extern, output.unwrap_or(void), &args, callee, false);
|
|
||||||
#[cfg(feature="master")]
|
|
||||||
callee.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
|
||||||
|
|
||||||
let block = func.new_block("entry");
|
let args = args
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(i, _)| func.get_param(i as i32).to_rvalue())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
let ret = context.new_call(None, callee, &args);
|
||||||
|
//llvm::LLVMSetTailCall(ret, True);
|
||||||
|
if output.is_some() {
|
||||||
|
block.end_with_return(None, ret);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
block.end_with_void_return(None);
|
||||||
|
}
|
||||||
|
|
||||||
let args = args
|
// TODO(@Commeownist): Check if we need to emit some extra debugging info in certain circumstances
|
||||||
.iter()
|
// as described in https://github.com/rust-lang/rust/commit/77a96ed5646f7c3ee8897693decc4626fe380643
|
||||||
.enumerate()
|
|
||||||
.map(|(i, _)| func.get_param(i as i32).to_rvalue())
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
let ret = context.new_call(None, callee, &args);
|
|
||||||
//llvm::LLVMSetTailCall(ret, True);
|
|
||||||
if output.is_some() {
|
|
||||||
block.end_with_return(None, ret);
|
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
block.end_with_void_return(None);
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO(@Commeownist): Check if we need to emit some extra debugging info in certain circumstances
|
|
||||||
// as described in https://github.com/rust-lang/rust/commit/77a96ed5646f7c3ee8897693decc4626fe380643
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let types = [usize, usize];
|
let types = [usize, usize];
|
||||||
|
@ -99,7 +102,7 @@ pub(crate) unsafe fn codegen(tcx: TyCtxt<'_>, mods: &mut GccContext, _module_nam
|
||||||
func.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
func.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
|
||||||
}
|
}
|
||||||
|
|
||||||
let callee = alloc_error_handler_kind.fn_name(sym::oom);
|
let callee = alloc_error_handler_name(alloc_error_handler_kind);
|
||||||
let args: Vec<_> = types.iter().enumerate()
|
let args: Vec<_> = types.iter().enumerate()
|
||||||
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
.map(|(index, typ)| context.new_parameter(None, *typ, &format!("param{}", index)))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
|
@ -1,10 +1,11 @@
|
||||||
use crate::attributes;
|
use crate::attributes;
|
||||||
use libc::c_uint;
|
use libc::c_uint;
|
||||||
use rustc_ast::expand::allocator::{AllocatorKind, AllocatorTy, ALLOCATOR_METHODS};
|
use rustc_ast::expand::allocator::{
|
||||||
|
alloc_error_handler_name, AllocatorKind, AllocatorTy, ALLOCATOR_METHODS,
|
||||||
|
};
|
||||||
use rustc_middle::bug;
|
use rustc_middle::bug;
|
||||||
use rustc_middle::ty::TyCtxt;
|
use rustc_middle::ty::TyCtxt;
|
||||||
use rustc_session::config::{DebugInfo, OomStrategy};
|
use rustc_session::config::{DebugInfo, OomStrategy};
|
||||||
use rustc_span::symbol::sym;
|
|
||||||
|
|
||||||
use crate::debuginfo;
|
use crate::debuginfo;
|
||||||
use crate::llvm::{self, False, True};
|
use crate::llvm::{self, False, True};
|
||||||
|
@ -29,75 +30,78 @@ pub(crate) unsafe fn codegen(
|
||||||
let i8p = llvm::LLVMPointerType(i8, 0);
|
let i8p = llvm::LLVMPointerType(i8, 0);
|
||||||
let void = llvm::LLVMVoidTypeInContext(llcx);
|
let void = llvm::LLVMVoidTypeInContext(llcx);
|
||||||
|
|
||||||
for method in ALLOCATOR_METHODS {
|
if kind == AllocatorKind::Default {
|
||||||
let mut args = Vec::with_capacity(method.inputs.len());
|
for method in ALLOCATOR_METHODS {
|
||||||
for ty in method.inputs.iter() {
|
let mut args = Vec::with_capacity(method.inputs.len());
|
||||||
match *ty {
|
for ty in method.inputs.iter() {
|
||||||
AllocatorTy::Layout => {
|
match *ty {
|
||||||
args.push(usize); // size
|
AllocatorTy::Layout => {
|
||||||
args.push(usize); // align
|
args.push(usize); // size
|
||||||
|
args.push(usize); // align
|
||||||
|
}
|
||||||
|
AllocatorTy::Ptr => args.push(i8p),
|
||||||
|
AllocatorTy::Usize => args.push(usize),
|
||||||
|
|
||||||
|
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
||||||
}
|
}
|
||||||
AllocatorTy::Ptr => args.push(i8p),
|
|
||||||
AllocatorTy::Usize => args.push(usize),
|
|
||||||
|
|
||||||
AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
|
|
||||||
}
|
}
|
||||||
}
|
let output = match method.output {
|
||||||
let output = match method.output {
|
AllocatorTy::ResultPtr => Some(i8p),
|
||||||
AllocatorTy::ResultPtr => Some(i8p),
|
AllocatorTy::Unit => None,
|
||||||
AllocatorTy::Unit => None,
|
|
||||||
|
|
||||||
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
|
||||||
panic!("invalid allocator output")
|
panic!("invalid allocator output")
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let ty = llvm::LLVMFunctionType(
|
||||||
|
output.unwrap_or(void),
|
||||||
|
args.as_ptr(),
|
||||||
|
args.len() as c_uint,
|
||||||
|
False,
|
||||||
|
);
|
||||||
|
let name = format!("__rust_{}", method.name);
|
||||||
|
let llfn =
|
||||||
|
llvm::LLVMRustGetOrInsertFunction(llmod, name.as_ptr().cast(), name.len(), ty);
|
||||||
|
|
||||||
|
if tcx.sess.target.default_hidden_visibility {
|
||||||
|
llvm::LLVMRustSetVisibility(llfn, llvm::Visibility::Hidden);
|
||||||
|
}
|
||||||
|
if tcx.sess.must_emit_unwind_tables() {
|
||||||
|
let uwtable = attributes::uwtable_attr(llcx);
|
||||||
|
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Function, &[uwtable]);
|
||||||
}
|
}
|
||||||
};
|
|
||||||
let ty = llvm::LLVMFunctionType(
|
|
||||||
output.unwrap_or(void),
|
|
||||||
args.as_ptr(),
|
|
||||||
args.len() as c_uint,
|
|
||||||
False,
|
|
||||||
);
|
|
||||||
let name = format!("__rust_{}", method.name);
|
|
||||||
let llfn = llvm::LLVMRustGetOrInsertFunction(llmod, name.as_ptr().cast(), name.len(), ty);
|
|
||||||
|
|
||||||
if tcx.sess.target.default_hidden_visibility {
|
let callee = AllocatorKind::Default.fn_name(method.name);
|
||||||
llvm::LLVMRustSetVisibility(llfn, llvm::Visibility::Hidden);
|
let callee =
|
||||||
|
llvm::LLVMRustGetOrInsertFunction(llmod, callee.as_ptr().cast(), callee.len(), ty);
|
||||||
|
llvm::LLVMRustSetVisibility(callee, llvm::Visibility::Hidden);
|
||||||
|
|
||||||
|
let llbb = llvm::LLVMAppendBasicBlockInContext(llcx, llfn, "entry\0".as_ptr().cast());
|
||||||
|
|
||||||
|
let llbuilder = llvm::LLVMCreateBuilderInContext(llcx);
|
||||||
|
llvm::LLVMPositionBuilderAtEnd(llbuilder, llbb);
|
||||||
|
let args = args
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(i, _)| llvm::LLVMGetParam(llfn, i as c_uint))
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
let ret = llvm::LLVMRustBuildCall(
|
||||||
|
llbuilder,
|
||||||
|
ty,
|
||||||
|
callee,
|
||||||
|
args.as_ptr(),
|
||||||
|
args.len() as c_uint,
|
||||||
|
[].as_ptr(),
|
||||||
|
0 as c_uint,
|
||||||
|
);
|
||||||
|
llvm::LLVMSetTailCall(ret, True);
|
||||||
|
if output.is_some() {
|
||||||
|
llvm::LLVMBuildRet(llbuilder, ret);
|
||||||
|
} else {
|
||||||
|
llvm::LLVMBuildRetVoid(llbuilder);
|
||||||
|
}
|
||||||
|
llvm::LLVMDisposeBuilder(llbuilder);
|
||||||
}
|
}
|
||||||
if tcx.sess.must_emit_unwind_tables() {
|
|
||||||
let uwtable = attributes::uwtable_attr(llcx);
|
|
||||||
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Function, &[uwtable]);
|
|
||||||
}
|
|
||||||
|
|
||||||
let callee = kind.fn_name(method.name);
|
|
||||||
let callee =
|
|
||||||
llvm::LLVMRustGetOrInsertFunction(llmod, callee.as_ptr().cast(), callee.len(), ty);
|
|
||||||
llvm::LLVMRustSetVisibility(callee, llvm::Visibility::Hidden);
|
|
||||||
|
|
||||||
let llbb = llvm::LLVMAppendBasicBlockInContext(llcx, llfn, "entry\0".as_ptr().cast());
|
|
||||||
|
|
||||||
let llbuilder = llvm::LLVMCreateBuilderInContext(llcx);
|
|
||||||
llvm::LLVMPositionBuilderAtEnd(llbuilder, llbb);
|
|
||||||
let args = args
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, _)| llvm::LLVMGetParam(llfn, i as c_uint))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
let ret = llvm::LLVMRustBuildCall(
|
|
||||||
llbuilder,
|
|
||||||
ty,
|
|
||||||
callee,
|
|
||||||
args.as_ptr(),
|
|
||||||
args.len() as c_uint,
|
|
||||||
[].as_ptr(),
|
|
||||||
0 as c_uint,
|
|
||||||
);
|
|
||||||
llvm::LLVMSetTailCall(ret, True);
|
|
||||||
if output.is_some() {
|
|
||||||
llvm::LLVMBuildRet(llbuilder, ret);
|
|
||||||
} else {
|
|
||||||
llvm::LLVMBuildRetVoid(llbuilder);
|
|
||||||
}
|
|
||||||
llvm::LLVMDisposeBuilder(llbuilder);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// rust alloc error handler
|
// rust alloc error handler
|
||||||
|
@ -118,7 +122,7 @@ pub(crate) unsafe fn codegen(
|
||||||
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Function, &[uwtable]);
|
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Function, &[uwtable]);
|
||||||
}
|
}
|
||||||
|
|
||||||
let callee = alloc_error_handler_kind.fn_name(sym::oom);
|
let callee = alloc_error_handler_name(alloc_error_handler_kind);
|
||||||
let callee = llvm::LLVMRustGetOrInsertFunction(llmod, callee.as_ptr().cast(), callee.len(), ty);
|
let callee = llvm::LLVMRustGetOrInsertFunction(llmod, callee.as_ptr().cast(), callee.len(), ty);
|
||||||
// -> ! DIFlagNoReturn
|
// -> ! DIFlagNoReturn
|
||||||
attributes::apply_to_llfn(callee, llvm::AttributePlace::Function, &[no_return]);
|
attributes::apply_to_llfn(callee, llvm::AttributePlace::Function, &[no_return]);
|
||||||
|
|
Loading…
Add table
Reference in a new issue