]> git.lizzy.rs Git - rust.git/blob - src/rtstartup/rsbegin.rs
Rollup merge of #68469 - ollie27:skip_count, r=sfackler
[rust.git] / src / rtstartup / rsbegin.rs
1 // rsbegin.o and rsend.o are the so called "compiler runtime startup objects".
2 // They contain code needed to correctly initialize the compiler runtime.
3 //
4 // When an executable or dylib image is linked, all user code and libraries are
5 // "sandwiched" between these two object files, so code or data from rsbegin.o
6 // become first in the respective sections of the image, whereas code and data
7 // from rsend.o become the last ones. This effect can be used to place symbols
8 // at the beginning or at the end of a section, as well as to insert any required
9 // headers or footers.
10 //
11 // Note that the actual module entry point is located in the C runtime startup
12 // object (usually called `crtX.o), which then invokes initialization callbacks
13 // of other runtime components (registered via yet another special image section).
14
15 #![feature(no_core, lang_items, optin_builtin_traits)]
16 #![crate_type = "rlib"]
17 #![no_core]
18 #![allow(non_camel_case_types)]
19
20 #[lang = "sized"]
21 trait Sized {}
22 #[lang = "sync"]
23 auto trait Sync {}
24 #[lang = "copy"]
25 trait Copy {}
26 #[lang = "freeze"]
27 auto trait Freeze {}
28
29 #[lang = "drop_in_place"]
30 #[inline]
31 #[allow(unconditional_recursion)]
32 pub unsafe fn drop_in_place<T: ?Sized>(to_drop: *mut T) {
33     drop_in_place(to_drop);
34 }
35
36 #[cfg(all(target_os = "windows", target_arch = "x86", target_env = "gnu"))]
37 pub mod eh_frames {
38     #[no_mangle]
39     #[link_section = ".eh_frame"]
40     // Marks beginning of the stack frame unwind info section
41     pub static __EH_FRAME_BEGIN__: [u8; 0] = [];
42
43     // Scratch space for unwinder's internal book-keeping.
44     // This is defined as `struct object` in $GCC/libgcc/unwind-dw2-fde.h.
45     static mut OBJ: [isize; 6] = [0; 6];
46
47     macro_rules! impl_copy {
48         ($($t:ty)*) => {
49             $(
50                 impl ::Copy for $t {}
51             )*
52         }
53     }
54
55     impl_copy! {
56         usize u8 u16 u32 u64 u128
57         isize i8 i16 i32 i64 i128
58         f32 f64
59         bool char
60     }
61
62     // Unwind info registration/deregistration routines.
63     // See the docs of `unwind` module in libstd.
64     extern "C" {
65         fn rust_eh_register_frames(eh_frame_begin: *const u8, object: *mut u8);
66         fn rust_eh_unregister_frames(eh_frame_begin: *const u8, object: *mut u8);
67     }
68
69     unsafe fn init() {
70         // register unwind info on module startup
71         rust_eh_register_frames(&__EH_FRAME_BEGIN__ as *const u8, &mut OBJ as *mut _ as *mut u8);
72     }
73
74     unsafe fn uninit() {
75         // unregister on shutdown
76         rust_eh_unregister_frames(&__EH_FRAME_BEGIN__ as *const u8, &mut OBJ as *mut _ as *mut u8);
77     }
78
79     // MSVC-specific init/uninit routine registration
80     pub mod ms_init {
81         // .CRT$X?? sections are roughly analogous to ELF's .init_array and .fini_array,
82         // except that they exploit the fact that linker will sort them alphabitically,
83         // so e.g., sections with names between .CRT$XIA and .CRT$XIZ are guaranteed to be
84         // placed between those two, without requiring any ordering of objects on the linker
85         // command line.
86         // Note that ordering of same-named sections from different objects is not guaranteed.
87         // Since .CRT$XIA contains init array's header symbol, which must always come first,
88         // we place our initialization callback into .CRT$XIB.
89
90         #[link_section = ".CRT$XIB"] // .CRT$XI? : C initialization callbacks
91         pub static P_INIT: unsafe fn() = super::init;
92
93         #[link_section = ".CRT$XTY"] // .CRT$XT? : C termination callbacks
94         pub static P_UNINIT: unsafe fn() = super::uninit;
95     }
96 }