xref: /linux/scripts/elf-parse.h (revision 36492b7141b9abc967e92c991af32c670351dc16)
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