about summary refs log tree commit diff
path: root/compiler/rustc_codegen_gcc/src/allocator.rs
blob: f4105b91e9848adde21aa12a87edec51ae63f0da (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#[cfg(feature="master")]
use gccjit::FnAttribute;
use gccjit::{FunctionType, GlobalKind, ToRValue};
use rustc_ast::expand::allocator::{
    alloc_error_handler_name, AllocatorKind, AllocatorTy, ALLOCATOR_METHODS,
};
use rustc_middle::bug;
use rustc_middle::ty::TyCtxt;
use rustc_session::config::OomStrategy;

use crate::GccContext;

pub(crate) unsafe fn codegen(tcx: TyCtxt<'_>, mods: &mut GccContext, _module_name: &str, kind: AllocatorKind, alloc_error_handler_kind: AllocatorKind) {
    let context = &mods.context;
    let usize =
        match tcx.sess.target.pointer_width {
            16 => context.new_type::<u16>(),
            32 => context.new_type::<u32>(),
            64 => context.new_type::<u64>(),
            tws => bug!("Unsupported target word size for int: {}", tws),
        };
    let i8 = context.new_type::<i8>();
    let i8p = i8.make_pointer();
    let void = context.new_type::<()>();

    if kind == AllocatorKind::Default {
        for method in ALLOCATOR_METHODS {
            let mut types = Vec::with_capacity(method.inputs.len());
            for ty in method.inputs.iter() {
                match *ty {
                    AllocatorTy::Layout => {
                        types.push(usize);
                        types.push(usize);
                    }
                    AllocatorTy::Ptr => types.push(i8p),
                    AllocatorTy::Usize => types.push(usize),

                    AllocatorTy::ResultPtr | AllocatorTy::Unit => panic!("invalid allocator arg"),
                }
            }
            let output = match method.output {
                AllocatorTy::ResultPtr => Some(i8p),
                AllocatorTy::Unit => None,

                AllocatorTy::Layout | AllocatorTy::Usize | AllocatorTy::Ptr => {
                    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 callee = AllocatorKind::Default.fn_name(method.name);
            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);
            }

            // 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 name = "__rust_alloc_error_handler".to_string();
    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, void, &args, name, false);

    if tcx.sess.target.default_hidden_visibility {
        #[cfg(feature="master")]
        func.add_attribute(FnAttribute::Visibility(gccjit::Visibility::Hidden));
    }

    let callee = alloc_error_handler_name(alloc_error_handler_kind);
    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, 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);
    block.end_with_void_return(None);

    let name = OomStrategy::SYMBOL.to_string();
    let global = context.new_global(None, GlobalKind::Exported, i8, name);
    let value = tcx.sess.opts.unstable_opts.oom.should_panic();
    let value = context.new_rvalue_from_int(i8, value as i32);
    global.global_set_initializer_rvalue(value);
}