1use proc_macro::TokenStream;
2use proc_macro2::Span;
3use quote::quote;
4use syn::{
5 parse, parse_macro_input, spanned::Spanned, Ident, ItemFn, ReturnType, Type, Visibility,
6};
7
8#[proc_macro_attribute]
9pub fn bentry(args: TokenStream, input: TokenStream) -> TokenStream {
10 let f = parse_macro_input!(input as ItemFn);
11
12 let valid_signature = f.sig.constness.is_none()
13 && f.vis == Visibility::Inherited
14 && f.sig.abi.is_none()
15 && f.sig.inputs.is_empty()
16 && f.sig.generics.params.is_empty()
17 && f.sig.generics.where_clause.is_none()
18 && f.sig.variadic.is_none()
19 && match f.sig.output {
20 ReturnType::Default => false,
21 ReturnType::Type(_, ref ty) => matches! {**ty, Type::Never(_)},
22 };
23
24 if !valid_signature {
25 return parse::Error::new(
26 f.span(),
27 "`#[entry]` function must have signature `[unsafe] fn() -> !`",
28 )
29 .to_compile_error()
30 .into();
31 }
32
33 if !args.is_empty() {
34 return parse::Error::new(Span::call_site(), "This attribute accepts no arguments")
35 .to_compile_error()
36 .into();
37 }
38
39 let tramp_ident = Ident::new(&format!("{}_trampoline", f.sig.ident), Span::call_site());
40 let ident = &f.sig.ident;
41
42 quote! {
43 global_asm!("
44 .macro enable_dbg
45 msr daifclr, #8
46 .endm
47
48 .macro mov_q, reg, val
49 .if (((\\val) >> 31) == 0 || ((\\val) >> 31) == 0x1ffffffff)
50 movz \\reg, :abs_g1_s:\\val
51 .else
52 .if (((\\val) >> 47) == 0 || ((\\val) >> 47) == 0x1ffff)
53 movz \\reg, :abs_g2_s:\\val
54 .else
55 movz \\reg, :abs_g3:\\val
56 movk \\reg, :abs_g2_nc:\\val
57 .endif
58 movk \\reg, :abs_g1_nc:\\val
59 .endif
60 movk \\reg, :abs_g0_nc:\\val
61 .endm
62
63 .macro reset_pmuserenr_el0, tmpreg
64 mrs \\tmpreg, id_aa64dfr0_el1
65 sbfx \\tmpreg, \\tmpreg, #8, #4
66 cmp \\tmpreg, #1
67 b.lt 9000f
68 msr pmuserenr_el0, xzr
699000:
70 .endm
71.pushsection \".head.text\",\"ax\"
72_head:
73 b stext
74 .long 0
75.globl stext; .align 2; stext:
76 mov x21, x0
77 bl el2_setup
78 bl __cpu_setup
79 b __primary_switch
80.type stext, @function; .size stext, .-stext
81.globl el2_setup; .align 2; el2_setup:
82
83 msr SPsel, #1
84 mrs x0, CurrentEL
85 cmp x0, #(2 << 2)
86 b.eq 1f
87 mov_q x0, (((((1) << (11))) | (((1) << (20))) | (((1) << (22))) | (((1) << (28))) | (((1) << (29)))) | 0)
88 msr sctlr_el1, x0
89 mov w0, #(0xe11)
90 isb
91 ret
92
93 1: mov_q x0, (((((1) << (4))) | (((1) << (5))) | (((1) << (11))) | (((1) << (16))) | (((1) << (18))) | (((1) << (22))) | (((1) << (23))) | (((1) << (28))) | (((1) << (29)))) | 0)
94 msr sctlr_el2, x0
95
96 mov_q x0, (((((1) << (11))) | (((1) << (20))) | (((1) << (22))) | (((1) << (28))) | (((1) << (29)))) | 0)
97 msr sctlr_el1, x0
98
99
100 mov x0, #(0x00000040 | 0x00000080 | 0x00000100 | 0x00000200 | 0x00000005)
101
102 msr spsr_el2, x0
103 msr elr_el2, lr
104 mov w0, #(0xe12)
105 eret
106.type el2_setup, @function; .size el2_setup, .-el2_setup
107.globl __cpu_setup; .align 2; __cpu_setup:
108
109 tlbi vmalle1
110 dsb nsh
111
112 mov x0, #3 << 20
113 msr cpacr_el1, x0
114 mov x0, #1 << 12
115 msr mdscr_el1, x0
116 isb
117 enable_dbg
118 reset_pmuserenr_el0 x0
119
120 ldr x5, =((0x00) << ((0) * 8)) | ((0x04) << ((1) * 8)) | ((0x0c) << ((2) * 8)) | ((0x44) << ((3) * 8)) | ((0xff) << ((4) * 8)) | ((0xbb) << ((5) * 8))
121 msr mair_el1, x5
122 ret
123.type __cpu_setup, @function; .size __cpu_setup, .-__cpu_setup
124__primary_switch:
125 adrp x4, __top_stack
126 mov x1, x4
127 mov sp, x1
128 bl clear_bss
129 b main
130.type __primary_switch, @function; .size __primary_switch, .-__primary_switch
131.popsection");
132
133 #[doc(hidden)]
134 #[export_name = "main"]
135 pub unsafe extern "C" fn #tramp_ident() {
136 #ident()
137 }
138
139 #f
140 }
141 .into()
142}