Blame view

scripts/recordmcount.h 19.3 KB
4317cf95c   Thomas Gleixner   treewide: Replace...
1
  /* SPDX-License-Identifier: GPL-2.0-only */
c28d5077f   Steven Rostedt   ftrace: Remove du...
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
  /*
   * recordmcount.h
   *
   * This code was taken out of recordmcount.c written by
   * Copyright 2009 John F. Reiser <jreiser@BitWagon.com>.  All rights reserved.
   *
   * The original code had the same algorithms for both 32bit
   * and 64bit ELF files, but the code was duplicated to support
   * the difference in structures that were used. This
   * file creates a macro of everything that is different between
   * the 64 and 32 bit code, such that by including this header
   * twice we can create both sets of functions by including this
   * header once with RECORD_MCOUNT_64 undefined, and again with
   * it defined.
   *
   * This conversion to macros was done by:
   * Copyright 2010 Steven Rostedt <srostedt@redhat.com>, Red Hat Inc.
c28d5077f   Steven Rostedt   ftrace: Remove du...
19
20
   */
  #undef append_func
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
21
22
23
  #undef is_fake_mcount
  #undef fn_is_fake_mcount
  #undef MIPS_is_fake_mcount
07d8b595f   Martin Schwidefsky   ftrace/recordmcou...
24
  #undef mcount_adjust
c28d5077f   Steven Rostedt   ftrace: Remove du...
25
  #undef sift_rel_mcount
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
26
  #undef nop_mcount
c28d5077f   Steven Rostedt   ftrace: Remove du...
27
28
29
30
  #undef find_secsym_ndx
  #undef __has_rel_mcount
  #undef has_rel_mcount
  #undef tot_relsize
37762cb99   Steven Rostedt   ftrace/recordmcou...
31
  #undef get_mcountsym
4ef57b21d   Sami Tolvanen   recordmcount: sup...
32
33
34
35
36
  #undef find_symtab
  #undef get_shnum
  #undef set_shnum
  #undef get_shstrndx
  #undef get_symindex
41b402a20   Steven Rostedt   ftrace/recordmcou...
37
  #undef get_sym_str_and_relp
c28d5077f   Steven Rostedt   ftrace: Remove du...
38
  #undef do_func
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
39
  #undef Elf_Addr
c28d5077f   Steven Rostedt   ftrace: Remove du...
40
41
42
43
44
45
  #undef Elf_Ehdr
  #undef Elf_Shdr
  #undef Elf_Rel
  #undef Elf_Rela
  #undef Elf_Sym
  #undef ELF_R_SYM
a2d49358b   John Reiser   ftrace/MIPS: Add ...
46
  #undef Elf_r_sym
c28d5077f   Steven Rostedt   ftrace: Remove du...
47
  #undef ELF_R_INFO
a2d49358b   John Reiser   ftrace/MIPS: Add ...
48
  #undef Elf_r_info
c28d5077f   Steven Rostedt   ftrace: Remove du...
49
  #undef ELF_ST_BIND
9905ce8ad   Rabin Vincent   ftrace/recordmcou...
50
  #undef ELF_ST_TYPE
a2d49358b   John Reiser   ftrace/MIPS: Add ...
51
52
  #undef fn_ELF_R_SYM
  #undef fn_ELF_R_INFO
c28d5077f   Steven Rostedt   ftrace: Remove du...
53
54
55
56
57
58
59
60
  #undef uint_t
  #undef _w
  #undef _align
  #undef _size
  
  #ifdef RECORD_MCOUNT_64
  # define append_func		append64
  # define sift_rel_mcount	sift64_rel_mcount
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
61
  # define nop_mcount		nop_mcount_64
c28d5077f   Steven Rostedt   ftrace: Remove du...
62
63
64
65
  # define find_secsym_ndx	find64_secsym_ndx
  # define __has_rel_mcount	__has64_rel_mcount
  # define has_rel_mcount		has64_rel_mcount
  # define tot_relsize		tot64_relsize
4ef57b21d   Sami Tolvanen   recordmcount: sup...
66
67
68
69
70
  # define find_symtab		find_symtab64
  # define get_shnum		get_shnum64
  # define set_shnum		set_shnum64
  # define get_shstrndx		get_shstrndx64
  # define get_symindex		get_symindex64
41b402a20   Steven Rostedt   ftrace/recordmcou...
71
  # define get_sym_str_and_relp	get_sym_str_and_relp_64
c28d5077f   Steven Rostedt   ftrace: Remove du...
72
  # define do_func		do64
37762cb99   Steven Rostedt   ftrace/recordmcou...
73
  # define get_mcountsym		get_mcountsym_64
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
74
75
76
  # define is_fake_mcount		is_fake_mcount64
  # define fn_is_fake_mcount	fn_is_fake_mcount64
  # define MIPS_is_fake_mcount	MIPS64_is_fake_mcount
07d8b595f   Martin Schwidefsky   ftrace/recordmcou...
77
  # define mcount_adjust		mcount_adjust_64
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
78
  # define Elf_Addr		Elf64_Addr
c28d5077f   Steven Rostedt   ftrace: Remove du...
79
80
81
82
83
84
  # define Elf_Ehdr		Elf64_Ehdr
  # define Elf_Shdr		Elf64_Shdr
  # define Elf_Rel		Elf64_Rel
  # define Elf_Rela		Elf64_Rela
  # define Elf_Sym		Elf64_Sym
  # define ELF_R_SYM		ELF64_R_SYM
a2d49358b   John Reiser   ftrace/MIPS: Add ...
85
  # define Elf_r_sym		Elf64_r_sym
c28d5077f   Steven Rostedt   ftrace: Remove du...
86
  # define ELF_R_INFO		ELF64_R_INFO
a2d49358b   John Reiser   ftrace/MIPS: Add ...
87
  # define Elf_r_info		Elf64_r_info
c28d5077f   Steven Rostedt   ftrace: Remove du...
88
  # define ELF_ST_BIND		ELF64_ST_BIND
9905ce8ad   Rabin Vincent   ftrace/recordmcou...
89
  # define ELF_ST_TYPE		ELF64_ST_TYPE
a2d49358b   John Reiser   ftrace/MIPS: Add ...
90
91
  # define fn_ELF_R_SYM		fn_ELF64_R_SYM
  # define fn_ELF_R_INFO		fn_ELF64_R_INFO
c28d5077f   Steven Rostedt   ftrace: Remove du...
92
93
94
95
96
97
98
  # define uint_t			uint64_t
  # define _w			w8
  # define _align			7u
  # define _size			8
  #else
  # define append_func		append32
  # define sift_rel_mcount	sift32_rel_mcount
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
99
  # define nop_mcount		nop_mcount_32
c28d5077f   Steven Rostedt   ftrace: Remove du...
100
101
102
103
  # define find_secsym_ndx	find32_secsym_ndx
  # define __has_rel_mcount	__has32_rel_mcount
  # define has_rel_mcount		has32_rel_mcount
  # define tot_relsize		tot32_relsize
4ef57b21d   Sami Tolvanen   recordmcount: sup...
104
105
106
107
108
  # define find_symtab		find_symtab32
  # define get_shnum		get_shnum32
  # define set_shnum		set_shnum32
  # define get_shstrndx		get_shstrndx32
  # define get_symindex		get_symindex32
41b402a20   Steven Rostedt   ftrace/recordmcou...
109
  # define get_sym_str_and_relp	get_sym_str_and_relp_32
c28d5077f   Steven Rostedt   ftrace: Remove du...
110
  # define do_func		do32
37762cb99   Steven Rostedt   ftrace/recordmcou...
111
  # define get_mcountsym		get_mcountsym_32
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
112
113
114
  # define is_fake_mcount		is_fake_mcount32
  # define fn_is_fake_mcount	fn_is_fake_mcount32
  # define MIPS_is_fake_mcount	MIPS32_is_fake_mcount
07d8b595f   Martin Schwidefsky   ftrace/recordmcou...
115
  # define mcount_adjust		mcount_adjust_32
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
116
  # define Elf_Addr		Elf32_Addr
c28d5077f   Steven Rostedt   ftrace: Remove du...
117
118
119
120
121
122
  # define Elf_Ehdr		Elf32_Ehdr
  # define Elf_Shdr		Elf32_Shdr
  # define Elf_Rel		Elf32_Rel
  # define Elf_Rela		Elf32_Rela
  # define Elf_Sym		Elf32_Sym
  # define ELF_R_SYM		ELF32_R_SYM
a2d49358b   John Reiser   ftrace/MIPS: Add ...
123
  # define Elf_r_sym		Elf32_r_sym
c28d5077f   Steven Rostedt   ftrace: Remove du...
124
  # define ELF_R_INFO		ELF32_R_INFO
a2d49358b   John Reiser   ftrace/MIPS: Add ...
125
  # define Elf_r_info		Elf32_r_info
c28d5077f   Steven Rostedt   ftrace: Remove du...
126
  # define ELF_ST_BIND		ELF32_ST_BIND
9905ce8ad   Rabin Vincent   ftrace/recordmcou...
127
  # define ELF_ST_TYPE		ELF32_ST_TYPE
a2d49358b   John Reiser   ftrace/MIPS: Add ...
128
129
  # define fn_ELF_R_SYM		fn_ELF32_R_SYM
  # define fn_ELF_R_INFO		fn_ELF32_R_INFO
c28d5077f   Steven Rostedt   ftrace: Remove du...
130
131
132
133
134
  # define uint_t			uint32_t
  # define _w			w
  # define _align			3u
  # define _size			4
  #endif
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
135
136
137
138
139
140
  /* Functions and pointers that do_file() may override for specific e_machine. */
  static int fn_is_fake_mcount(Elf_Rel const *rp)
  {
  	return 0;
  }
  static int (*is_fake_mcount)(Elf_Rel const *rp) = fn_is_fake_mcount;
a2d49358b   John Reiser   ftrace/MIPS: Add ...
141
142
143
144
145
146
147
148
  static uint_t fn_ELF_R_SYM(Elf_Rel const *rp)
  {
  	return ELF_R_SYM(_w(rp->r_info));
  }
  static uint_t (*Elf_r_sym)(Elf_Rel const *rp) = fn_ELF_R_SYM;
  
  static void fn_ELF_R_INFO(Elf_Rel *const rp, unsigned sym, unsigned type)
  {
e63233f75   John Reiser   ftrace: Have reco...
149
  	rp->r_info = _w(ELF_R_INFO(sym, type));
a2d49358b   John Reiser   ftrace/MIPS: Add ...
150
151
  }
  static void (*Elf_r_info)(Elf_Rel *const rp, unsigned sym, unsigned type) = fn_ELF_R_INFO;
07d8b595f   Martin Schwidefsky   ftrace/recordmcou...
152
  static int mcount_adjust = 0;
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
  /*
   * MIPS mcount long call has 2 _mcount symbols, only the position of the 1st
   * _mcount symbol is needed for dynamic function tracer, with it, to disable
   * tracing(ftrace_make_nop), the instruction in the position is replaced with
   * the "b label" instruction, to enable tracing(ftrace_make_call), replace the
   * instruction back. So, here, we set the 2nd one as fake and filter it.
   *
   * c:	3c030000	lui	v1,0x0		<-->	b	label
   *		c: R_MIPS_HI16	_mcount
   *		c: R_MIPS_NONE	*ABS*
   *		c: R_MIPS_NONE	*ABS*
   * 10:	64630000	daddiu	v1,v1,0
   *		10: R_MIPS_LO16	_mcount
   *		10: R_MIPS_NONE	*ABS*
   *		10: R_MIPS_NONE	*ABS*
   * 14:	03e0082d	move	at,ra
   * 18:	0060f809	jalr	v1
   * label:
   */
  #define MIPS_FAKEMCOUNT_OFFSET	4
  
  static int MIPS_is_fake_mcount(Elf_Rel const *rp)
  {
91ad11d7c   Alex Smith   recordmcount/MIPS...
176
  	static Elf_Addr old_r_offset = ~(Elf_Addr)0;
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
177
178
  	Elf_Addr current_r_offset = _w(rp->r_offset);
  	int is_fake;
91ad11d7c   Alex Smith   recordmcount/MIPS...
179
  	is_fake = (old_r_offset != ~(Elf_Addr)0) &&
412910cd0   Wu Zhangjin   ftrace/MIPS: Add ...
180
181
182
183
184
  		(current_r_offset - old_r_offset == MIPS_FAKEMCOUNT_OFFSET);
  	old_r_offset = current_r_offset;
  
  	return is_fake;
  }
a2d49358b   John Reiser   ftrace/MIPS: Add ...
185

4ef57b21d   Sami Tolvanen   recordmcount: sup...
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
  static unsigned int get_symindex(Elf_Sym const *sym, Elf32_Word const *symtab,
  				 Elf32_Word const *symtab_shndx)
  {
  	unsigned long offset;
  	int index;
  
  	if (sym->st_shndx != SHN_XINDEX)
  		return w2(sym->st_shndx);
  
  	offset = (unsigned long)sym - (unsigned long)symtab;
  	index = offset / sizeof(*sym);
  
  	return w(symtab_shndx[index]);
  }
  
  static unsigned int get_shnum(Elf_Ehdr const *ehdr, Elf_Shdr const *shdr0)
  {
  	if (shdr0 && !ehdr->e_shnum)
  		return w(shdr0->sh_size);
  
  	return w2(ehdr->e_shnum);
  }
  
  static void set_shnum(Elf_Ehdr *ehdr, Elf_Shdr *shdr0, unsigned int new_shnum)
  {
  	if (new_shnum >= SHN_LORESERVE) {
  		ehdr->e_shnum = 0;
  		shdr0->sh_size = w(new_shnum);
  	} else
  		ehdr->e_shnum = w2(new_shnum);
  }
  
  static int get_shstrndx(Elf_Ehdr const *ehdr, Elf_Shdr const *shdr0)
  {
  	if (ehdr->e_shstrndx != SHN_XINDEX)
  		return w2(ehdr->e_shstrndx);
  
  	return w(shdr0->sh_link);
  }
  
  static void find_symtab(Elf_Ehdr *const ehdr, Elf_Shdr const *shdr0,
  			unsigned const nhdr, Elf32_Word **symtab,
  			Elf32_Word **symtab_shndx)
  {
  	Elf_Shdr const *relhdr;
  	unsigned k;
  
  	*symtab = NULL;
  	*symtab_shndx = NULL;
  
  	for (relhdr = shdr0, k = nhdr; k; --k, ++relhdr) {
  		if (relhdr->sh_type == SHT_SYMTAB)
  			*symtab = (void *)ehdr + relhdr->sh_offset;
  		else if (relhdr->sh_type == SHT_SYMTAB_SHNDX)
  			*symtab_shndx = (void *)ehdr + relhdr->sh_offset;
  
  		if (*symtab && *symtab_shndx)
  			break;
  	}
  }
c28d5077f   Steven Rostedt   ftrace: Remove du...
246
  /* Append the new shstrtab, Elf_Shdr[], __mcount_loc and its relocations. */
3f1df1201   Matt Helsley   recordmcount: Rew...
247
  static int append_func(Elf_Ehdr *const ehdr,
c28d5077f   Steven Rostedt   ftrace: Remove du...
248
249
250
251
252
253
254
255
256
257
258
259
260
  			Elf_Shdr *const shstr,
  			uint_t const *const mloc0,
  			uint_t const *const mlocp,
  			Elf_Rel const *const mrel0,
  			Elf_Rel const *const mrelp,
  			unsigned int const rel_entsize,
  			unsigned int const symsec_sh_link)
  {
  	/* Begin constructing output file */
  	Elf_Shdr mcsec;
  	char const *mc_name = (sizeof(Elf_Rela) == rel_entsize)
  		? ".rela__mcount_loc"
  		:  ".rel__mcount_loc";
c28d5077f   Steven Rostedt   ftrace: Remove du...
261
262
263
  	uint_t const old_shoff = _w(ehdr->e_shoff);
  	uint_t const old_shstr_sh_size   = _w(shstr->sh_size);
  	uint_t const old_shstr_sh_offset = _w(shstr->sh_offset);
4ef57b21d   Sami Tolvanen   recordmcount: sup...
264
265
266
  	Elf_Shdr *const shdr0 = (Elf_Shdr *)(old_shoff + (void *)ehdr);
  	unsigned int const old_shnum = get_shnum(ehdr, shdr0);
  	unsigned int const new_shnum = 2 + old_shnum; /* {.rel,}__mcount_loc */
c28d5077f   Steven Rostedt   ftrace: Remove du...
267
268
269
270
271
272
273
274
  	uint_t t = 1 + strlen(mc_name) + _w(shstr->sh_size);
  	uint_t new_e_shoff;
  
  	shstr->sh_size = _w(t);
  	shstr->sh_offset = _w(sb.st_size);
  	t += sb.st_size;
  	t += (_align & -t);  /* word-byte align */
  	new_e_shoff = t;
4ef57b21d   Sami Tolvanen   recordmcount: sup...
275
  	set_shnum(ehdr, shdr0, new_shnum);
c28d5077f   Steven Rostedt   ftrace: Remove du...
276
  	/* body for new shstrtab */
3f1df1201   Matt Helsley   recordmcount: Rew...
277
278
279
280
281
282
  	if (ulseek(sb.st_size, SEEK_SET) < 0)
  		return -1;
  	if (uwrite(old_shstr_sh_offset + (void *)ehdr, old_shstr_sh_size) < 0)
  		return -1;
  	if (uwrite(mc_name, 1 + strlen(mc_name)) < 0)
  		return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
283
284
  
  	/* old(modified) Elf_Shdr table, word-byte aligned */
3f1df1201   Matt Helsley   recordmcount: Rew...
285
286
  	if (ulseek(t, SEEK_SET) < 0)
  		return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
287
  	t += sizeof(Elf_Shdr) * old_shnum;
3f1df1201   Matt Helsley   recordmcount: Rew...
288
289
290
  	if (uwrite(old_shoff + (void *)ehdr,
  	       sizeof(Elf_Shdr) * old_shnum) < 0)
  		return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
291
292
293
294
295
296
297
298
299
300
301
302
303
304
  
  	/* new sections __mcount_loc and .rel__mcount_loc */
  	t += 2*sizeof(mcsec);
  	mcsec.sh_name = w((sizeof(Elf_Rela) == rel_entsize) + strlen(".rel")
  		+ old_shstr_sh_size);
  	mcsec.sh_type = w(SHT_PROGBITS);
  	mcsec.sh_flags = _w(SHF_ALLOC);
  	mcsec.sh_addr = 0;
  	mcsec.sh_offset = _w(t);
  	mcsec.sh_size = _w((void *)mlocp - (void *)mloc0);
  	mcsec.sh_link = 0;
  	mcsec.sh_info = 0;
  	mcsec.sh_addralign = _w(_size);
  	mcsec.sh_entsize = _w(_size);
3f1df1201   Matt Helsley   recordmcount: Rew...
305
306
  	if (uwrite(&mcsec, sizeof(mcsec)) < 0)
  		return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
307
308
309
310
311
312
313
314
315
316
317
318
319
  
  	mcsec.sh_name = w(old_shstr_sh_size);
  	mcsec.sh_type = (sizeof(Elf_Rela) == rel_entsize)
  		? w(SHT_RELA)
  		: w(SHT_REL);
  	mcsec.sh_flags = 0;
  	mcsec.sh_addr = 0;
  	mcsec.sh_offset = _w((void *)mlocp - (void *)mloc0 + t);
  	mcsec.sh_size   = _w((void *)mrelp - (void *)mrel0);
  	mcsec.sh_link = w(symsec_sh_link);
  	mcsec.sh_info = w(old_shnum);
  	mcsec.sh_addralign = _w(_size);
  	mcsec.sh_entsize = _w(rel_entsize);
c28d5077f   Steven Rostedt   ftrace: Remove du...
320

3f1df1201   Matt Helsley   recordmcount: Rew...
321
322
323
324
325
326
327
  	if (uwrite(&mcsec, sizeof(mcsec)) < 0)
  		return -1;
  
  	if (uwrite(mloc0, (void *)mlocp - (void *)mloc0) < 0)
  		return -1;
  	if (uwrite(mrel0, (void *)mrelp - (void *)mrel0) < 0)
  		return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
328
329
  
  	ehdr->e_shoff = _w(new_e_shoff);
3f1df1201   Matt Helsley   recordmcount: Rew...
330
331
332
333
334
  	if (ulseek(0, SEEK_SET) < 0)
  		return -1;
  	if (uwrite(ehdr, sizeof(*ehdr)) < 0)
  		return -1;
  	return 0;
c28d5077f   Steven Rostedt   ftrace: Remove du...
335
  }
37762cb99   Steven Rostedt   ftrace/recordmcou...
336
337
338
339
340
341
342
343
344
345
  static unsigned get_mcountsym(Elf_Sym const *const sym0,
  			      Elf_Rel const *relp,
  			      char const *const str0)
  {
  	unsigned mcountsym = 0;
  
  	Elf_Sym const *const symp =
  		&sym0[Elf_r_sym(relp)];
  	char const *symname = &str0[w(symp->st_name)];
  	char const *mcount = gpfx == '_' ? "_mcount" : "mcount";
48bb5dc6c   Steven Rostedt   ftrace: Make reco...
346
  	char const *fentry = "__fentry__";
37762cb99   Steven Rostedt   ftrace/recordmcou...
347
348
349
350
  
  	if (symname[0] == '.')
  		++symname;  /* ppc64 hack */
  	if (strcmp(mcount, symname) == 0 ||
48bb5dc6c   Steven Rostedt   ftrace: Make reco...
351
352
  	    (altmcount && strcmp(altmcount, symname) == 0) ||
  	    (strcmp(fentry, symname) == 0))
37762cb99   Steven Rostedt   ftrace/recordmcou...
353
354
355
356
  		mcountsym = Elf_r_sym(relp);
  
  	return mcountsym;
  }
41b402a20   Steven Rostedt   ftrace/recordmcou...
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
  static void get_sym_str_and_relp(Elf_Shdr const *const relhdr,
  				 Elf_Ehdr const *const ehdr,
  				 Elf_Sym const **sym0,
  				 char const **str0,
  				 Elf_Rel const **relp)
  {
  	Elf_Shdr *const shdr0 = (Elf_Shdr *)(_w(ehdr->e_shoff)
  		+ (void *)ehdr);
  	unsigned const symsec_sh_link = w(relhdr->sh_link);
  	Elf_Shdr const *const symsec = &shdr0[symsec_sh_link];
  	Elf_Shdr const *const strsec = &shdr0[w(symsec->sh_link)];
  	Elf_Rel const *const rel0 = (Elf_Rel const *)(_w(relhdr->sh_offset)
  		+ (void *)ehdr);
  
  	*sym0 = (Elf_Sym const *)(_w(symsec->sh_offset)
  				  + (void *)ehdr);
  
  	*str0 = (char const *)(_w(strsec->sh_offset)
  			       + (void *)ehdr);
  
  	*relp = rel0;
  }
c28d5077f   Steven Rostedt   ftrace: Remove du...
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
  /*
   * Look at the relocations in order to find the calls to mcount.
   * Accumulate the section offsets that are found, and their relocation info,
   * onto the end of the existing arrays.
   */
  static uint_t *sift_rel_mcount(uint_t *mlocp,
  			       unsigned const offbase,
  			       Elf_Rel **const mrelpp,
  			       Elf_Shdr const *const relhdr,
  			       Elf_Ehdr const *const ehdr,
  			       unsigned const recsym,
  			       uint_t const recval,
  			       unsigned const reltype)
  {
  	uint_t *const mloc0 = mlocp;
  	Elf_Rel *mrelp = *mrelpp;
41b402a20   Steven Rostedt   ftrace/recordmcou...
395
396
397
  	Elf_Sym const *sym0;
  	char const *str0;
  	Elf_Rel const *relp;
c28d5077f   Steven Rostedt   ftrace: Remove du...
398
399
  	unsigned rel_entsize = _w(relhdr->sh_entsize);
  	unsigned const nrel = _w(relhdr->sh_size) / rel_entsize;
c28d5077f   Steven Rostedt   ftrace: Remove du...
400
401
  	unsigned mcountsym = 0;
  	unsigned t;
41b402a20   Steven Rostedt   ftrace/recordmcou...
402
  	get_sym_str_and_relp(relhdr, ehdr, &sym0, &str0, &relp);
c28d5077f   Steven Rostedt   ftrace: Remove du...
403
  	for (t = nrel; t; --t) {
37762cb99   Steven Rostedt   ftrace/recordmcou...
404
405
  		if (!mcountsym)
  			mcountsym = get_mcountsym(sym0, relp, str0);
c28d5077f   Steven Rostedt   ftrace: Remove du...
406

80e5302e4   Naveen N. Rao   recordmcount: Fix...
407
408
  		if (mcountsym && mcountsym == Elf_r_sym(relp) &&
  				!is_fake_mcount(relp)) {
07d8b595f   Martin Schwidefsky   ftrace/recordmcou...
409
410
  			uint_t const addend =
  				_w(_w(relp->r_offset) - recval + mcount_adjust);
c28d5077f   Steven Rostedt   ftrace: Remove du...
411
412
  			mrelp->r_offset = _w(offbase
  				+ ((void *)mlocp - (void *)mloc0));
a2d49358b   John Reiser   ftrace/MIPS: Add ...
413
  			Elf_r_info(mrelp, recsym, reltype);
dd5477ff3   Steven Rostedt   ftrace/trivial: C...
414
  			if (rel_entsize == sizeof(Elf_Rela)) {
c28d5077f   Steven Rostedt   ftrace: Remove du...
415
416
417
418
419
420
421
422
423
424
425
426
  				((Elf_Rela *)mrelp)->r_addend = addend;
  				*mlocp++ = 0;
  			} else
  				*mlocp++ = addend;
  
  			mrelp = (Elf_Rel *)(rel_entsize + (void *)mrelp);
  		}
  		relp = (Elf_Rel const *)(rel_entsize + (void *)relp);
  	}
  	*mrelpp = mrelp;
  	return mlocp;
  }
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
427
428
429
430
431
  /*
   * Read the relocation table again, but this time its called on sections
   * that are not going to be traced. The mcount calls here will be converted
   * into nops.
   */
3f1df1201   Matt Helsley   recordmcount: Rew...
432
433
434
  static int nop_mcount(Elf_Shdr const *const relhdr,
  		      Elf_Ehdr const *const ehdr,
  		      const char *const txtname)
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
435
436
437
  {
  	Elf_Shdr *const shdr0 = (Elf_Shdr *)(_w(ehdr->e_shoff)
  		+ (void *)ehdr);
41b402a20   Steven Rostedt   ftrace/recordmcou...
438
439
440
441
  	Elf_Sym const *sym0;
  	char const *str0;
  	Elf_Rel const *relp;
  	Elf_Shdr const *const shdr = &shdr0[w(relhdr->sh_info)];
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
442
443
  	unsigned rel_entsize = _w(relhdr->sh_entsize);
  	unsigned const nrel = _w(relhdr->sh_size) / rel_entsize;
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
444
445
  	unsigned mcountsym = 0;
  	unsigned t;
dfad3d598   Steven Rostedt   ftrace/recordmcou...
446
  	int once = 0;
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
447

41b402a20   Steven Rostedt   ftrace/recordmcou...
448
  	get_sym_str_and_relp(relhdr, ehdr, &sym0, &str0, &relp);
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
449
450
  	for (t = nrel; t; --t) {
  		int ret = -1;
37762cb99   Steven Rostedt   ftrace/recordmcou...
451
452
  		if (!mcountsym)
  			mcountsym = get_mcountsym(sym0, relp, str0);
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
453

dfad3d598   Steven Rostedt   ftrace/recordmcou...
454
  		if (mcountsym == Elf_r_sym(relp) && !is_fake_mcount(relp)) {
7f8557b88   Steven Rostedt (VMware)   recordmcount: Fix...
455
  			if (make_nop)
c84da8b9a   libin   recordmcount: Fix...
456
  				ret = make_nop((void *)ehdr, _w(shdr->sh_offset) + _w(relp->r_offset));
dfad3d598   Steven Rostedt   ftrace/recordmcou...
457
458
459
460
461
462
463
  			if (warn_on_notrace_sect && !once) {
  				printf("Section %s has mcount callers being ignored
  ",
  				       txtname);
  				once = 1;
  				/* just warn? */
  				if (!make_nop)
3f1df1201   Matt Helsley   recordmcount: Rew...
464
  					return 0;
dfad3d598   Steven Rostedt   ftrace/recordmcou...
465
466
  			}
  		}
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
467
468
469
470
471
472
473
474
475
  
  		/*
  		 * If we successfully removed the mcount, mark the relocation
  		 * as a nop (don't do anything with it).
  		 */
  		if (!ret) {
  			Elf_Rel rel;
  			rel = *(Elf_Rel *)relp;
  			Elf_r_info(&rel, Elf_r_sym(relp), rel_type_nop);
3f1df1201   Matt Helsley   recordmcount: Rew...
476
477
478
479
  			if (ulseek((void *)relp - (void *)ehdr, SEEK_SET) < 0)
  				return -1;
  			if (uwrite(&rel, sizeof(rel)) < 0)
  				return -1;
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
480
481
482
  		}
  		relp = (Elf_Rel const *)(rel_entsize + (void *)relp);
  	}
3f1df1201   Matt Helsley   recordmcount: Rew...
483
  	return 0;
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
484
  }
c28d5077f   Steven Rostedt   ftrace: Remove du...
485
486
487
488
489
490
491
492
493
494
  /*
   * Find a symbol in the given section, to be used as the base for relocating
   * the table of offsets of calls to mcount.  A local or global symbol suffices,
   * but avoid a Weak symbol because it may be overridden; the change in value
   * would invalidate the relocations of the offsets of the calls to mcount.
   * Often the found symbol will be the unnamed local symbol generated by
   * GNU 'as' for the start of each section.  For example:
   *    Num:    Value  Size Type    Bind   Vis      Ndx Name
   *      2: 00000000     0 SECTION LOCAL  DEFAULT    1
   */
3f1df1201   Matt Helsley   recordmcount: Rew...
495
  static int find_secsym_ndx(unsigned const txtndx,
c28d5077f   Steven Rostedt   ftrace: Remove du...
496
497
  				char const *const txtname,
  				uint_t *const recvalp,
3f1df1201   Matt Helsley   recordmcount: Rew...
498
  				unsigned int *sym_index,
c28d5077f   Steven Rostedt   ftrace: Remove du...
499
  				Elf_Shdr const *const symhdr,
4ef57b21d   Sami Tolvanen   recordmcount: sup...
500
501
  				Elf32_Word const *symtab,
  				Elf32_Word const *symtab_shndx,
c28d5077f   Steven Rostedt   ftrace: Remove du...
502
503
504
505
506
507
508
509
510
511
  				Elf_Ehdr const *const ehdr)
  {
  	Elf_Sym const *const sym0 = (Elf_Sym const *)(_w(symhdr->sh_offset)
  		+ (void *)ehdr);
  	unsigned const nsym = _w(symhdr->sh_size) / _w(symhdr->sh_entsize);
  	Elf_Sym const *symp;
  	unsigned t;
  
  	for (symp = sym0, t = nsym; t; --t, ++symp) {
  		unsigned int const st_bind = ELF_ST_BIND(symp->st_info);
4ef57b21d   Sami Tolvanen   recordmcount: sup...
512
  		if (txtndx == get_symindex(symp, symtab, symtab_shndx)
c28d5077f   Steven Rostedt   ftrace: Remove du...
513
514
  			/* avoid STB_WEAK */
  		    && (STB_LOCAL == st_bind || STB_GLOBAL == st_bind)) {
9905ce8ad   Rabin Vincent   ftrace/recordmcou...
515
516
517
518
  			/* function symbols on ARM have quirks, avoid them */
  			if (w2(ehdr->e_machine) == EM_ARM
  			    && ELF_ST_TYPE(symp->st_info) == STT_FUNC)
  				continue;
c28d5077f   Steven Rostedt   ftrace: Remove du...
519
  			*recvalp = _w(symp->st_value);
3f1df1201   Matt Helsley   recordmcount: Rew...
520
521
  			*sym_index = symp - sym0;
  			return 0;
c28d5077f   Steven Rostedt   ftrace: Remove du...
522
523
  		}
  	}
ac5db1fc8   nixiaoming   scripts: Fixed pr...
524
525
  	fprintf(stderr, "Cannot find symbol for section %u: %s.
  ",
c28d5077f   Steven Rostedt   ftrace: Remove du...
526
  		txtndx, txtname);
3f1df1201   Matt Helsley   recordmcount: Rew...
527
  	return -1;
c28d5077f   Steven Rostedt   ftrace: Remove du...
528
  }
c28d5077f   Steven Rostedt   ftrace: Remove du...
529
  /* Evade ISO C restriction: no declaration after statement in has_rel_mcount. */
3aec86382   Matt Helsley   recordmcount: Ker...
530
531
532
533
  static char const * __has_rel_mcount(Elf_Shdr const *const relhdr, /* reltype */
  				     Elf_Shdr const *const shdr0,
  				     char const *const shstrtab,
  				     char const *const fname)
c28d5077f   Steven Rostedt   ftrace: Remove du...
534
535
536
537
  {
  	/* .sh_info depends on .sh_type == SHT_REL[,A] */
  	Elf_Shdr const *const txthdr = &shdr0[w(relhdr->sh_info)];
  	char const *const txtname = &shstrtab[w(txthdr->sh_name)];
dd5477ff3   Steven Rostedt   ftrace/trivial: C...
538
  	if (strcmp("__mcount_loc", txtname) == 0) {
c28d5077f   Steven Rostedt   ftrace: Remove du...
539
540
541
  		fprintf(stderr, "warning: __mcount_loc already exists: %s
  ",
  			fname);
3f1df1201   Matt Helsley   recordmcount: Rew...
542
  		return already_has_rel_mcount;
c28d5077f   Steven Rostedt   ftrace: Remove du...
543
  	}
dd5477ff3   Steven Rostedt   ftrace/trivial: C...
544
  	if (w(txthdr->sh_type) != SHT_PROGBITS ||
2e885057b   David Daney   recordmcount: Fix...
545
  	    !(_w(txthdr->sh_flags) & SHF_EXECINSTR))
c28d5077f   Steven Rostedt   ftrace: Remove du...
546
547
548
549
550
551
552
553
554
  		return NULL;
  	return txtname;
  }
  
  static char const *has_rel_mcount(Elf_Shdr const *const relhdr,
  				  Elf_Shdr const *const shdr0,
  				  char const *const shstrtab,
  				  char const *const fname)
  {
dd5477ff3   Steven Rostedt   ftrace/trivial: C...
555
  	if (w(relhdr->sh_type) != SHT_REL && w(relhdr->sh_type) != SHT_RELA)
c28d5077f   Steven Rostedt   ftrace: Remove du...
556
557
558
559
560
561
562
563
564
565
566
567
  		return NULL;
  	return __has_rel_mcount(relhdr, shdr0, shstrtab, fname);
  }
  
  
  static unsigned tot_relsize(Elf_Shdr const *const shdr0,
  			    unsigned nhdr,
  			    const char *const shstrtab,
  			    const char *const fname)
  {
  	unsigned totrelsz = 0;
  	Elf_Shdr const *shdrp = shdr0;
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
568
  	char const *txtname;
c28d5077f   Steven Rostedt   ftrace: Remove du...
569
570
  
  	for (; nhdr; --nhdr, ++shdrp) {
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
571
  		txtname = has_rel_mcount(shdrp, shdr0, shstrtab, fname);
3f1df1201   Matt Helsley   recordmcount: Rew...
572
573
574
575
  		if (txtname == already_has_rel_mcount) {
  			totrelsz = 0;
  			break;
  		}
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
576
  		if (txtname && is_mcounted_section_name(txtname))
c28d5077f   Steven Rostedt   ftrace: Remove du...
577
578
579
580
  			totrelsz += _w(shdrp->sh_size);
  	}
  	return totrelsz;
  }
c28d5077f   Steven Rostedt   ftrace: Remove du...
581
  /* Overall supervision for Elf32 ET_REL file. */
3aec86382   Matt Helsley   recordmcount: Ker...
582
583
  static int do_func(Elf_Ehdr *const ehdr, char const *const fname,
  		   unsigned const reltype)
c28d5077f   Steven Rostedt   ftrace: Remove du...
584
585
586
  {
  	Elf_Shdr *const shdr0 = (Elf_Shdr *)(_w(ehdr->e_shoff)
  		+ (void *)ehdr);
4ef57b21d   Sami Tolvanen   recordmcount: sup...
587
588
  	unsigned const nhdr = get_shnum(ehdr, shdr0);
  	Elf_Shdr *const shstr = &shdr0[get_shstrndx(ehdr, shdr0)];
c28d5077f   Steven Rostedt   ftrace: Remove du...
589
590
591
592
593
  	char const *const shstrtab = (char const *)(_w(shstr->sh_offset)
  		+ (void *)ehdr);
  
  	Elf_Shdr const *relhdr;
  	unsigned k;
4ef57b21d   Sami Tolvanen   recordmcount: sup...
594
595
  	Elf32_Word *symtab;
  	Elf32_Word *symtab_shndx;
c28d5077f   Steven Rostedt   ftrace: Remove du...
596
  	/* Upper bound on space: assume all relevant relocs are for mcount. */
3f1df1201   Matt Helsley   recordmcount: Rew...
597
  	unsigned       totrelsz;
c28d5077f   Steven Rostedt   ftrace: Remove du...
598

3f1df1201   Matt Helsley   recordmcount: Rew...
599
600
601
602
603
  	Elf_Rel *      mrel0;
  	Elf_Rel *      mrelp;
  
  	uint_t *      mloc0;
  	uint_t *      mlocp;
c28d5077f   Steven Rostedt   ftrace: Remove du...
604
605
606
  
  	unsigned rel_entsize = 0;
  	unsigned symsec_sh_link = 0;
3f1df1201   Matt Helsley   recordmcount: Rew...
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
  	int result = 0;
  
  	totrelsz = tot_relsize(shdr0, nhdr, shstrtab, fname);
  	if (totrelsz == 0)
  		return 0;
  	mrel0 = umalloc(totrelsz);
  	mrelp = mrel0;
  	if (!mrel0)
  		return -1;
  
  	/* 2*sizeof(address) <= sizeof(Elf_Rel) */
  	mloc0 = umalloc(totrelsz>>1);
  	mlocp = mloc0;
  	if (!mloc0) {
  		free(mrel0);
  		return -1;
  	}
4ef57b21d   Sami Tolvanen   recordmcount: sup...
624
  	find_symtab(ehdr, shdr0, nhdr, &symtab, &symtab_shndx);
c28d5077f   Steven Rostedt   ftrace: Remove du...
625
626
627
  	for (relhdr = shdr0, k = nhdr; k; --k, ++relhdr) {
  		char const *const txtname = has_rel_mcount(relhdr, shdr0,
  			shstrtab, fname);
3f1df1201   Matt Helsley   recordmcount: Rew...
628
629
630
631
632
  		if (txtname == already_has_rel_mcount) {
  			result = 0;
  			file_updated = 0;
  			goto out; /* Nothing to be done; don't append! */
  		}
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
633
  		if (txtname && is_mcounted_section_name(txtname)) {
3f1df1201   Matt Helsley   recordmcount: Rew...
634
  			unsigned int recsym;
c28d5077f   Steven Rostedt   ftrace: Remove du...
635
  			uint_t recval = 0;
3f1df1201   Matt Helsley   recordmcount: Rew...
636
637
638
639
640
  
  			symsec_sh_link = w(relhdr->sh_link);
  			result = find_secsym_ndx(w(relhdr->sh_info), txtname,
  						&recval, &recsym,
  						&shdr0[symsec_sh_link],
4ef57b21d   Sami Tolvanen   recordmcount: sup...
641
  						symtab, symtab_shndx,
3f1df1201   Matt Helsley   recordmcount: Rew...
642
643
644
  						ehdr);
  			if (result)
  				goto out;
c28d5077f   Steven Rostedt   ftrace: Remove du...
645
646
647
648
649
  
  			rel_entsize = _w(relhdr->sh_entsize);
  			mlocp = sift_rel_mcount(mlocp,
  				(void *)mlocp - (void *)mloc0, &mrelp,
  				relhdr, ehdr, recsym, recval, reltype);
dfad3d598   Steven Rostedt   ftrace/recordmcou...
650
  		} else if (txtname && (warn_on_notrace_sect || make_nop)) {
ffd618fa3   Steven Rostedt   ftrace/recordmcou...
651
652
653
654
  			/*
  			 * This section is ignored by ftrace, but still
  			 * has mcount calls. Convert them to nops now.
  			 */
3f1df1201   Matt Helsley   recordmcount: Rew...
655
656
657
658
  			if (nop_mcount(relhdr, ehdr, txtname) < 0) {
  				result = -1;
  				goto out;
  			}
c28d5077f   Steven Rostedt   ftrace: Remove du...
659
660
  		}
  	}
3f1df1201   Matt Helsley   recordmcount: Rew...
661
662
663
664
  	if (!result && mloc0 != mlocp)
  		result = append_func(ehdr, shstr, mloc0, mlocp, mrel0, mrelp,
  				     rel_entsize, symsec_sh_link);
  out:
c28d5077f   Steven Rostedt   ftrace: Remove du...
665
666
  	free(mrel0);
  	free(mloc0);
3f1df1201   Matt Helsley   recordmcount: Rew...
667
  	return result;
c28d5077f   Steven Rostedt   ftrace: Remove du...
668
  }