1*b055f4c4SSteven Rostedt /* SPDX-License-Identifier: GPL-2.0-only */
2*b055f4c4SSteven Rostedt #ifndef _SCRIPTS_ELF_PARSE_H
3*b055f4c4SSteven Rostedt #define _SCRIPTS_ELF_PARSE_H
4*b055f4c4SSteven Rostedt
5*b055f4c4SSteven Rostedt #include <elf.h>
6*b055f4c4SSteven Rostedt
7*b055f4c4SSteven Rostedt #include <tools/be_byteshift.h>
8*b055f4c4SSteven Rostedt #include <tools/le_byteshift.h>
9*b055f4c4SSteven Rostedt
10*b055f4c4SSteven Rostedt typedef union {
11*b055f4c4SSteven Rostedt Elf32_Ehdr e32;
12*b055f4c4SSteven Rostedt Elf64_Ehdr e64;
13*b055f4c4SSteven Rostedt } Elf_Ehdr;
14*b055f4c4SSteven Rostedt
15*b055f4c4SSteven Rostedt typedef union {
16*b055f4c4SSteven Rostedt Elf32_Shdr e32;
17*b055f4c4SSteven Rostedt Elf64_Shdr e64;
18*b055f4c4SSteven Rostedt } Elf_Shdr;
19*b055f4c4SSteven Rostedt
20*b055f4c4SSteven Rostedt typedef union {
21*b055f4c4SSteven Rostedt Elf32_Sym e32;
22*b055f4c4SSteven Rostedt Elf64_Sym e64;
23*b055f4c4SSteven Rostedt } Elf_Sym;
24*b055f4c4SSteven Rostedt
25*b055f4c4SSteven Rostedt typedef union {
26*b055f4c4SSteven Rostedt Elf32_Rela e32;
27*b055f4c4SSteven Rostedt Elf64_Rela e64;
28*b055f4c4SSteven Rostedt } Elf_Rela;
29*b055f4c4SSteven Rostedt
30*b055f4c4SSteven Rostedt struct elf_funcs {
31*b055f4c4SSteven Rostedt int (*compare_extable)(const void *a, const void *b);
32*b055f4c4SSteven Rostedt uint64_t (*ehdr_shoff)(Elf_Ehdr *ehdr);
33*b055f4c4SSteven Rostedt uint16_t (*ehdr_shstrndx)(Elf_Ehdr *ehdr);
34*b055f4c4SSteven Rostedt uint16_t (*ehdr_shentsize)(Elf_Ehdr *ehdr);
35*b055f4c4SSteven Rostedt uint16_t (*ehdr_shnum)(Elf_Ehdr *ehdr);
36*b055f4c4SSteven Rostedt uint64_t (*shdr_addr)(Elf_Shdr *shdr);
37*b055f4c4SSteven Rostedt uint64_t (*shdr_offset)(Elf_Shdr *shdr);
38*b055f4c4SSteven Rostedt uint64_t (*shdr_size)(Elf_Shdr *shdr);
39*b055f4c4SSteven Rostedt uint64_t (*shdr_entsize)(Elf_Shdr *shdr);
40*b055f4c4SSteven Rostedt uint32_t (*shdr_link)(Elf_Shdr *shdr);
41*b055f4c4SSteven Rostedt uint32_t (*shdr_name)(Elf_Shdr *shdr);
42*b055f4c4SSteven Rostedt uint32_t (*shdr_type)(Elf_Shdr *shdr);
43*b055f4c4SSteven Rostedt uint8_t (*sym_type)(Elf_Sym *sym);
44*b055f4c4SSteven Rostedt uint32_t (*sym_name)(Elf_Sym *sym);
45*b055f4c4SSteven Rostedt uint64_t (*sym_value)(Elf_Sym *sym);
46*b055f4c4SSteven Rostedt uint16_t (*sym_shndx)(Elf_Sym *sym);
47*b055f4c4SSteven Rostedt uint64_t (*rela_offset)(Elf_Rela *rela);
48*b055f4c4SSteven Rostedt uint64_t (*rela_info)(Elf_Rela *rela);
49*b055f4c4SSteven Rostedt uint64_t (*rela_addend)(Elf_Rela *rela);
50*b055f4c4SSteven Rostedt void (*rela_write_addend)(Elf_Rela *rela, uint64_t val);
51*b055f4c4SSteven Rostedt uint32_t (*r)(const uint32_t *);
52*b055f4c4SSteven Rostedt uint16_t (*r2)(const uint16_t *);
53*b055f4c4SSteven Rostedt uint64_t (*r8)(const uint64_t *);
54*b055f4c4SSteven Rostedt void (*w)(uint32_t, uint32_t *);
55*b055f4c4SSteven Rostedt void (*w8)(uint64_t, uint64_t *);
56*b055f4c4SSteven Rostedt };
57*b055f4c4SSteven Rostedt
58*b055f4c4SSteven Rostedt extern struct elf_funcs elf_parser;
59*b055f4c4SSteven Rostedt
ehdr64_shoff(Elf_Ehdr * ehdr)60*b055f4c4SSteven Rostedt static inline uint64_t ehdr64_shoff(Elf_Ehdr *ehdr)
61*b055f4c4SSteven Rostedt {
62*b055f4c4SSteven Rostedt return elf_parser.r8(&ehdr->e64.e_shoff);
63*b055f4c4SSteven Rostedt }
64*b055f4c4SSteven Rostedt
ehdr32_shoff(Elf_Ehdr * ehdr)65*b055f4c4SSteven Rostedt static inline uint64_t ehdr32_shoff(Elf_Ehdr *ehdr)
66*b055f4c4SSteven Rostedt {
67*b055f4c4SSteven Rostedt return elf_parser.r(&ehdr->e32.e_shoff);
68*b055f4c4SSteven Rostedt }
69*b055f4c4SSteven Rostedt
ehdr_shoff(Elf_Ehdr * ehdr)70*b055f4c4SSteven Rostedt static inline uint64_t ehdr_shoff(Elf_Ehdr *ehdr)
71*b055f4c4SSteven Rostedt {
72*b055f4c4SSteven Rostedt return elf_parser.ehdr_shoff(ehdr);
73*b055f4c4SSteven Rostedt }
74*b055f4c4SSteven Rostedt
75*b055f4c4SSteven Rostedt #define EHDR_HALF(fn_name) \
76*b055f4c4SSteven Rostedt static inline uint16_t ehdr64_##fn_name(Elf_Ehdr *ehdr) \
77*b055f4c4SSteven Rostedt { \
78*b055f4c4SSteven Rostedt return elf_parser.r2(&ehdr->e64.e_##fn_name); \
79*b055f4c4SSteven Rostedt } \
80*b055f4c4SSteven Rostedt \
81*b055f4c4SSteven Rostedt static inline uint16_t ehdr32_##fn_name(Elf_Ehdr *ehdr) \
82*b055f4c4SSteven Rostedt { \
83*b055f4c4SSteven Rostedt return elf_parser.r2(&ehdr->e32.e_##fn_name); \
84*b055f4c4SSteven Rostedt } \
85*b055f4c4SSteven Rostedt \
86*b055f4c4SSteven Rostedt static inline uint16_t ehdr_##fn_name(Elf_Ehdr *ehdr) \
87*b055f4c4SSteven Rostedt { \
88*b055f4c4SSteven Rostedt return elf_parser.ehdr_##fn_name(ehdr); \
89*b055f4c4SSteven Rostedt }
90*b055f4c4SSteven Rostedt
91*b055f4c4SSteven Rostedt EHDR_HALF(shentsize)
EHDR_HALF(shstrndx)92*b055f4c4SSteven Rostedt EHDR_HALF(shstrndx)
93*b055f4c4SSteven Rostedt EHDR_HALF(shnum)
94*b055f4c4SSteven Rostedt
95*b055f4c4SSteven Rostedt #define SHDR_WORD(fn_name) \
96*b055f4c4SSteven Rostedt static inline uint32_t shdr64_##fn_name(Elf_Shdr *shdr) \
97*b055f4c4SSteven Rostedt { \
98*b055f4c4SSteven Rostedt return elf_parser.r(&shdr->e64.sh_##fn_name); \
99*b055f4c4SSteven Rostedt } \
100*b055f4c4SSteven Rostedt \
101*b055f4c4SSteven Rostedt static inline uint32_t shdr32_##fn_name(Elf_Shdr *shdr) \
102*b055f4c4SSteven Rostedt { \
103*b055f4c4SSteven Rostedt return elf_parser.r(&shdr->e32.sh_##fn_name); \
104*b055f4c4SSteven Rostedt } \
105*b055f4c4SSteven Rostedt \
106*b055f4c4SSteven Rostedt static inline uint32_t shdr_##fn_name(Elf_Shdr *shdr) \
107*b055f4c4SSteven Rostedt { \
108*b055f4c4SSteven Rostedt return elf_parser.shdr_##fn_name(shdr); \
109*b055f4c4SSteven Rostedt }
110*b055f4c4SSteven Rostedt
111*b055f4c4SSteven Rostedt #define SHDR_ADDR(fn_name) \
112*b055f4c4SSteven Rostedt static inline uint64_t shdr64_##fn_name(Elf_Shdr *shdr) \
113*b055f4c4SSteven Rostedt { \
114*b055f4c4SSteven Rostedt return elf_parser.r8(&shdr->e64.sh_##fn_name); \
115*b055f4c4SSteven Rostedt } \
116*b055f4c4SSteven Rostedt \
117*b055f4c4SSteven Rostedt static inline uint64_t shdr32_##fn_name(Elf_Shdr *shdr) \
118*b055f4c4SSteven Rostedt { \
119*b055f4c4SSteven Rostedt return elf_parser.r(&shdr->e32.sh_##fn_name); \
120*b055f4c4SSteven Rostedt } \
121*b055f4c4SSteven Rostedt \
122*b055f4c4SSteven Rostedt static inline uint64_t shdr_##fn_name(Elf_Shdr *shdr) \
123*b055f4c4SSteven Rostedt { \
124*b055f4c4SSteven Rostedt return elf_parser.shdr_##fn_name(shdr); \
125*b055f4c4SSteven Rostedt }
126*b055f4c4SSteven Rostedt
127*b055f4c4SSteven Rostedt #define SHDR_WORD(fn_name) \
128*b055f4c4SSteven Rostedt static inline uint32_t shdr64_##fn_name(Elf_Shdr *shdr) \
129*b055f4c4SSteven Rostedt { \
130*b055f4c4SSteven Rostedt return elf_parser.r(&shdr->e64.sh_##fn_name); \
131*b055f4c4SSteven Rostedt } \
132*b055f4c4SSteven Rostedt \
133*b055f4c4SSteven Rostedt static inline uint32_t shdr32_##fn_name(Elf_Shdr *shdr) \
134*b055f4c4SSteven Rostedt { \
135*b055f4c4SSteven Rostedt return elf_parser.r(&shdr->e32.sh_##fn_name); \
136*b055f4c4SSteven Rostedt } \
137*b055f4c4SSteven Rostedt static inline uint32_t shdr_##fn_name(Elf_Shdr *shdr) \
138*b055f4c4SSteven Rostedt { \
139*b055f4c4SSteven Rostedt return elf_parser.shdr_##fn_name(shdr); \
140*b055f4c4SSteven Rostedt }
141*b055f4c4SSteven Rostedt
142*b055f4c4SSteven Rostedt SHDR_ADDR(addr)
143*b055f4c4SSteven Rostedt SHDR_ADDR(offset)
144*b055f4c4SSteven Rostedt SHDR_ADDR(size)
145*b055f4c4SSteven Rostedt SHDR_ADDR(entsize)
146*b055f4c4SSteven Rostedt
147*b055f4c4SSteven Rostedt SHDR_WORD(link)
148*b055f4c4SSteven Rostedt SHDR_WORD(name)
149*b055f4c4SSteven Rostedt SHDR_WORD(type)
150*b055f4c4SSteven Rostedt
151*b055f4c4SSteven Rostedt #define SYM_ADDR(fn_name) \
152*b055f4c4SSteven Rostedt static inline uint64_t sym64_##fn_name(Elf_Sym *sym) \
153*b055f4c4SSteven Rostedt { \
154*b055f4c4SSteven Rostedt return elf_parser.r8(&sym->e64.st_##fn_name); \
155*b055f4c4SSteven Rostedt } \
156*b055f4c4SSteven Rostedt \
157*b055f4c4SSteven Rostedt static inline uint64_t sym32_##fn_name(Elf_Sym *sym) \
158*b055f4c4SSteven Rostedt { \
159*b055f4c4SSteven Rostedt return elf_parser.r(&sym->e32.st_##fn_name); \
160*b055f4c4SSteven Rostedt } \
161*b055f4c4SSteven Rostedt \
162*b055f4c4SSteven Rostedt static inline uint64_t sym_##fn_name(Elf_Sym *sym) \
163*b055f4c4SSteven Rostedt { \
164*b055f4c4SSteven Rostedt return elf_parser.sym_##fn_name(sym); \
165*b055f4c4SSteven Rostedt }
166*b055f4c4SSteven Rostedt
167*b055f4c4SSteven Rostedt #define SYM_WORD(fn_name) \
168*b055f4c4SSteven Rostedt static inline uint32_t sym64_##fn_name(Elf_Sym *sym) \
169*b055f4c4SSteven Rostedt { \
170*b055f4c4SSteven Rostedt return elf_parser.r(&sym->e64.st_##fn_name); \
171*b055f4c4SSteven Rostedt } \
172*b055f4c4SSteven Rostedt \
173*b055f4c4SSteven Rostedt static inline uint32_t sym32_##fn_name(Elf_Sym *sym) \
174*b055f4c4SSteven Rostedt { \
175*b055f4c4SSteven Rostedt return elf_parser.r(&sym->e32.st_##fn_name); \
176*b055f4c4SSteven Rostedt } \
177*b055f4c4SSteven Rostedt \
178*b055f4c4SSteven Rostedt static inline uint32_t sym_##fn_name(Elf_Sym *sym) \
179*b055f4c4SSteven Rostedt { \
180*b055f4c4SSteven Rostedt return elf_parser.sym_##fn_name(sym); \
181*b055f4c4SSteven Rostedt }
182*b055f4c4SSteven Rostedt
183*b055f4c4SSteven Rostedt #define SYM_HALF(fn_name) \
184*b055f4c4SSteven Rostedt static inline uint16_t sym64_##fn_name(Elf_Sym *sym) \
185*b055f4c4SSteven Rostedt { \
186*b055f4c4SSteven Rostedt return elf_parser.r2(&sym->e64.st_##fn_name); \
187*b055f4c4SSteven Rostedt } \
188*b055f4c4SSteven Rostedt \
189*b055f4c4SSteven Rostedt static inline uint16_t sym32_##fn_name(Elf_Sym *sym) \
190*b055f4c4SSteven Rostedt { \
191*b055f4c4SSteven Rostedt return elf_parser.r2(&sym->e32.st_##fn_name); \
192*b055f4c4SSteven Rostedt } \
193*b055f4c4SSteven Rostedt \
194*b055f4c4SSteven Rostedt static inline uint16_t sym_##fn_name(Elf_Sym *sym) \
195*b055f4c4SSteven Rostedt { \
196*b055f4c4SSteven Rostedt return elf_parser.sym_##fn_name(sym); \
197*b055f4c4SSteven Rostedt }
198*b055f4c4SSteven Rostedt
199*b055f4c4SSteven Rostedt static inline uint8_t sym64_type(Elf_Sym *sym)
200*b055f4c4SSteven Rostedt {
201*b055f4c4SSteven Rostedt return ELF64_ST_TYPE(sym->e64.st_info);
202*b055f4c4SSteven Rostedt }
203*b055f4c4SSteven Rostedt
sym32_type(Elf_Sym * sym)204*b055f4c4SSteven Rostedt static inline uint8_t sym32_type(Elf_Sym *sym)
205*b055f4c4SSteven Rostedt {
206*b055f4c4SSteven Rostedt return ELF32_ST_TYPE(sym->e32.st_info);
207*b055f4c4SSteven Rostedt }
208*b055f4c4SSteven Rostedt
sym_type(Elf_Sym * sym)209*b055f4c4SSteven Rostedt static inline uint8_t sym_type(Elf_Sym *sym)
210*b055f4c4SSteven Rostedt {
211*b055f4c4SSteven Rostedt return elf_parser.sym_type(sym);
212*b055f4c4SSteven Rostedt }
213*b055f4c4SSteven Rostedt
214*b055f4c4SSteven Rostedt SYM_ADDR(value)
SYM_WORD(name)215*b055f4c4SSteven Rostedt SYM_WORD(name)
216*b055f4c4SSteven Rostedt SYM_HALF(shndx)
217*b055f4c4SSteven Rostedt
218*b055f4c4SSteven Rostedt #define __maybe_unused __attribute__((__unused__))
219*b055f4c4SSteven Rostedt
220*b055f4c4SSteven Rostedt #define RELA_ADDR(fn_name) \
221*b055f4c4SSteven Rostedt static inline uint64_t rela64_##fn_name(Elf_Rela *rela) \
222*b055f4c4SSteven Rostedt { \
223*b055f4c4SSteven Rostedt return elf_parser.r8((uint64_t *)&rela->e64.r_##fn_name); \
224*b055f4c4SSteven Rostedt } \
225*b055f4c4SSteven Rostedt \
226*b055f4c4SSteven Rostedt static inline uint64_t rela32_##fn_name(Elf_Rela *rela) \
227*b055f4c4SSteven Rostedt { \
228*b055f4c4SSteven Rostedt return elf_parser.r((uint32_t *)&rela->e32.r_##fn_name); \
229*b055f4c4SSteven Rostedt } \
230*b055f4c4SSteven Rostedt \
231*b055f4c4SSteven Rostedt static inline uint64_t __maybe_unused rela_##fn_name(Elf_Rela *rela) \
232*b055f4c4SSteven Rostedt { \
233*b055f4c4SSteven Rostedt return elf_parser.rela_##fn_name(rela); \
234*b055f4c4SSteven Rostedt }
235*b055f4c4SSteven Rostedt
236*b055f4c4SSteven Rostedt RELA_ADDR(offset)
237*b055f4c4SSteven Rostedt RELA_ADDR(info)
238*b055f4c4SSteven Rostedt RELA_ADDR(addend)
239*b055f4c4SSteven Rostedt
240*b055f4c4SSteven Rostedt static inline void rela64_write_addend(Elf_Rela *rela, uint64_t val)
241*b055f4c4SSteven Rostedt {
242*b055f4c4SSteven Rostedt elf_parser.w8(val, (uint64_t *)&rela->e64.r_addend);
243*b055f4c4SSteven Rostedt }
244*b055f4c4SSteven Rostedt
rela32_write_addend(Elf_Rela * rela,uint64_t val)245*b055f4c4SSteven Rostedt static inline void rela32_write_addend(Elf_Rela *rela, uint64_t val)
246*b055f4c4SSteven Rostedt {
247*b055f4c4SSteven Rostedt elf_parser.w(val, (uint32_t *)&rela->e32.r_addend);
248*b055f4c4SSteven Rostedt }
249*b055f4c4SSteven Rostedt
rbe(const uint32_t * x)250*b055f4c4SSteven Rostedt static inline uint32_t rbe(const uint32_t *x)
251*b055f4c4SSteven Rostedt {
252*b055f4c4SSteven Rostedt return get_unaligned_be32(x);
253*b055f4c4SSteven Rostedt }
254*b055f4c4SSteven Rostedt
r2be(const uint16_t * x)255*b055f4c4SSteven Rostedt static inline uint16_t r2be(const uint16_t *x)
256*b055f4c4SSteven Rostedt {
257*b055f4c4SSteven Rostedt return get_unaligned_be16(x);
258*b055f4c4SSteven Rostedt }
259*b055f4c4SSteven Rostedt
r8be(const uint64_t * x)260*b055f4c4SSteven Rostedt static inline uint64_t r8be(const uint64_t *x)
261*b055f4c4SSteven Rostedt {
262*b055f4c4SSteven Rostedt return get_unaligned_be64(x);
263*b055f4c4SSteven Rostedt }
264*b055f4c4SSteven Rostedt
rle(const uint32_t * x)265*b055f4c4SSteven Rostedt static inline uint32_t rle(const uint32_t *x)
266*b055f4c4SSteven Rostedt {
267*b055f4c4SSteven Rostedt return get_unaligned_le32(x);
268*b055f4c4SSteven Rostedt }
269*b055f4c4SSteven Rostedt
r2le(const uint16_t * x)270*b055f4c4SSteven Rostedt static inline uint16_t r2le(const uint16_t *x)
271*b055f4c4SSteven Rostedt {
272*b055f4c4SSteven Rostedt return get_unaligned_le16(x);
273*b055f4c4SSteven Rostedt }
274*b055f4c4SSteven Rostedt
r8le(const uint64_t * x)275*b055f4c4SSteven Rostedt static inline uint64_t r8le(const uint64_t *x)
276*b055f4c4SSteven Rostedt {
277*b055f4c4SSteven Rostedt return get_unaligned_le64(x);
278*b055f4c4SSteven Rostedt }
279*b055f4c4SSteven Rostedt
wbe(uint32_t val,uint32_t * x)280*b055f4c4SSteven Rostedt static inline void wbe(uint32_t val, uint32_t *x)
281*b055f4c4SSteven Rostedt {
282*b055f4c4SSteven Rostedt put_unaligned_be32(val, x);
283*b055f4c4SSteven Rostedt }
284*b055f4c4SSteven Rostedt
wle(uint32_t val,uint32_t * x)285*b055f4c4SSteven Rostedt static inline void wle(uint32_t val, uint32_t *x)
286*b055f4c4SSteven Rostedt {
287*b055f4c4SSteven Rostedt put_unaligned_le32(val, x);
288*b055f4c4SSteven Rostedt }
289*b055f4c4SSteven Rostedt
w8be(uint64_t val,uint64_t * x)290*b055f4c4SSteven Rostedt static inline void w8be(uint64_t val, uint64_t *x)
291*b055f4c4SSteven Rostedt {
292*b055f4c4SSteven Rostedt put_unaligned_be64(val, x);
293*b055f4c4SSteven Rostedt }
294*b055f4c4SSteven Rostedt
w8le(uint64_t val,uint64_t * x)295*b055f4c4SSteven Rostedt static inline void w8le(uint64_t val, uint64_t *x)
296*b055f4c4SSteven Rostedt {
297*b055f4c4SSteven Rostedt put_unaligned_le64(val, x);
298*b055f4c4SSteven Rostedt }
299*b055f4c4SSteven Rostedt
300*b055f4c4SSteven Rostedt void *elf_map(char const *fname, size_t *size, uint32_t types);
301*b055f4c4SSteven Rostedt void elf_unmap(void *addr, size_t size);
302*b055f4c4SSteven Rostedt int elf_map_machine(void *addr);
303*b055f4c4SSteven Rostedt int elf_map_long_size(void *addr);
304*b055f4c4SSteven Rostedt
305*b055f4c4SSteven Rostedt #endif /* _SCRIPTS_ELF_PARSE_H */
306