Blame view

tools/perf/util/symbol-minimal.c 7.23 KB
393be2e37   Namhyung Kim   perf symbols: Sup...
1
  #include "symbol.h"
74cf249d5   Arnaldo Carvalho de Melo   perf tools: Use z...
2
  #include "util.h"
393be2e37   Namhyung Kim   perf symbols: Sup...
3

a43783aee   Arnaldo Carvalho de Melo   perf tools: Inclu...
4
  #include <errno.h>
b691f6436   Namhyung Kim   perf symbols: Imp...
5
6
7
8
9
  #include <stdio.h>
  #include <fcntl.h>
  #include <string.h>
  #include <byteswap.h>
  #include <sys/stat.h>
393be2e37   Namhyung Kim   perf symbols: Sup...
10

b691f6436   Namhyung Kim   perf symbols: Imp...
11
12
  
  static bool check_need_swap(int file_endian)
393be2e37   Namhyung Kim   perf symbols: Sup...
13
  {
b691f6436   Namhyung Kim   perf symbols: Imp...
14
15
16
17
18
19
20
21
22
23
  	const int data = 1;
  	u8 *check = (u8 *)&data;
  	int host_endian;
  
  	if (check[0] == 1)
  		host_endian = ELFDATA2LSB;
  	else
  		host_endian = ELFDATA2MSB;
  
  	return host_endian != file_endian;
393be2e37   Namhyung Kim   perf symbols: Sup...
24
  }
b691f6436   Namhyung Kim   perf symbols: Imp...
25
26
27
28
29
30
  #define NOTE_ALIGN(sz) (((sz) + 3) & ~3)
  
  #define NT_GNU_BUILD_ID	3
  
  static int read_build_id(void *note_data, size_t note_len, void *bf,
  			 size_t size, bool need_swap)
393be2e37   Namhyung Kim   perf symbols: Sup...
31
  {
b691f6436   Namhyung Kim   perf symbols: Imp...
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
  	struct {
  		u32 n_namesz;
  		u32 n_descsz;
  		u32 n_type;
  	} *nhdr;
  	void *ptr;
  
  	ptr = note_data;
  	while (ptr < (note_data + note_len)) {
  		const char *name;
  		size_t namesz, descsz;
  
  		nhdr = ptr;
  		if (need_swap) {
  			nhdr->n_namesz = bswap_32(nhdr->n_namesz);
  			nhdr->n_descsz = bswap_32(nhdr->n_descsz);
  			nhdr->n_type = bswap_32(nhdr->n_type);
  		}
  
  		namesz = NOTE_ALIGN(nhdr->n_namesz);
  		descsz = NOTE_ALIGN(nhdr->n_descsz);
  
  		ptr += sizeof(*nhdr);
  		name = ptr;
  		ptr += namesz;
  		if (nhdr->n_type == NT_GNU_BUILD_ID &&
  		    nhdr->n_namesz == sizeof("GNU")) {
  			if (memcmp(name, "GNU", sizeof("GNU")) == 0) {
  				size_t sz = min(size, descsz);
  				memcpy(bf, ptr, sz);
  				memset(bf + sz, 0, size - sz);
  				return 0;
  			}
  		}
  		ptr += descsz;
  	}
393be2e37   Namhyung Kim   perf symbols: Sup...
68
69
  	return -1;
  }
1d037ca16   Irina Tirdea   perf tools: Use _...
70
71
72
  int filename__read_debuglink(const char *filename __maybe_unused,
  			     char *debuglink __maybe_unused,
  			     size_t size __maybe_unused)
393be2e37   Namhyung Kim   perf symbols: Sup...
73
74
75
  {
  	return -1;
  }
b691f6436   Namhyung Kim   perf symbols: Imp...
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
  /*
   * Just try PT_NOTE header otherwise fails
   */
  int filename__read_build_id(const char *filename, void *bf, size_t size)
  {
  	FILE *fp;
  	int ret = -1;
  	bool need_swap = false;
  	u8 e_ident[EI_NIDENT];
  	size_t buf_size;
  	void *buf;
  	int i;
  
  	fp = fopen(filename, "r");
  	if (fp == NULL)
  		return -1;
  
  	if (fread(e_ident, sizeof(e_ident), 1, fp) != 1)
  		goto out;
  
  	if (memcmp(e_ident, ELFMAG, SELFMAG) ||
  	    e_ident[EI_VERSION] != EV_CURRENT)
  		goto out;
  
  	need_swap = check_need_swap(e_ident[EI_DATA]);
  
  	/* for simplicity */
  	fseek(fp, 0, SEEK_SET);
  
  	if (e_ident[EI_CLASS] == ELFCLASS32) {
  		Elf32_Ehdr ehdr;
  		Elf32_Phdr *phdr;
  
  		if (fread(&ehdr, sizeof(ehdr), 1, fp) != 1)
  			goto out;
  
  		if (need_swap) {
  			ehdr.e_phoff = bswap_32(ehdr.e_phoff);
  			ehdr.e_phentsize = bswap_16(ehdr.e_phentsize);
  			ehdr.e_phnum = bswap_16(ehdr.e_phnum);
  		}
  
  		buf_size = ehdr.e_phentsize * ehdr.e_phnum;
  		buf = malloc(buf_size);
  		if (buf == NULL)
  			goto out;
  
  		fseek(fp, ehdr.e_phoff, SEEK_SET);
  		if (fread(buf, buf_size, 1, fp) != 1)
  			goto out_free;
  
  		for (i = 0, phdr = buf; i < ehdr.e_phnum; i++, phdr++) {
  			void *tmp;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
129
  			long offset;
b691f6436   Namhyung Kim   perf symbols: Imp...
130
131
132
133
134
135
136
137
138
139
140
  
  			if (need_swap) {
  				phdr->p_type = bswap_32(phdr->p_type);
  				phdr->p_offset = bswap_32(phdr->p_offset);
  				phdr->p_filesz = bswap_32(phdr->p_filesz);
  			}
  
  			if (phdr->p_type != PT_NOTE)
  				continue;
  
  			buf_size = phdr->p_filesz;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
141
  			offset = phdr->p_offset;
b691f6436   Namhyung Kim   perf symbols: Imp...
142
143
144
145
146
  			tmp = realloc(buf, buf_size);
  			if (tmp == NULL)
  				goto out_free;
  
  			buf = tmp;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
147
  			fseek(fp, offset, SEEK_SET);
b691f6436   Namhyung Kim   perf symbols: Imp...
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
  			if (fread(buf, buf_size, 1, fp) != 1)
  				goto out_free;
  
  			ret = read_build_id(buf, buf_size, bf, size, need_swap);
  			if (ret == 0)
  				ret = size;
  			break;
  		}
  	} else {
  		Elf64_Ehdr ehdr;
  		Elf64_Phdr *phdr;
  
  		if (fread(&ehdr, sizeof(ehdr), 1, fp) != 1)
  			goto out;
  
  		if (need_swap) {
  			ehdr.e_phoff = bswap_64(ehdr.e_phoff);
  			ehdr.e_phentsize = bswap_16(ehdr.e_phentsize);
  			ehdr.e_phnum = bswap_16(ehdr.e_phnum);
  		}
  
  		buf_size = ehdr.e_phentsize * ehdr.e_phnum;
  		buf = malloc(buf_size);
  		if (buf == NULL)
  			goto out;
  
  		fseek(fp, ehdr.e_phoff, SEEK_SET);
  		if (fread(buf, buf_size, 1, fp) != 1)
  			goto out_free;
  
  		for (i = 0, phdr = buf; i < ehdr.e_phnum; i++, phdr++) {
  			void *tmp;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
180
  			long offset;
b691f6436   Namhyung Kim   perf symbols: Imp...
181
182
183
184
185
186
187
188
189
190
191
  
  			if (need_swap) {
  				phdr->p_type = bswap_32(phdr->p_type);
  				phdr->p_offset = bswap_64(phdr->p_offset);
  				phdr->p_filesz = bswap_64(phdr->p_filesz);
  			}
  
  			if (phdr->p_type != PT_NOTE)
  				continue;
  
  			buf_size = phdr->p_filesz;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
192
  			offset = phdr->p_offset;
b691f6436   Namhyung Kim   perf symbols: Imp...
193
194
195
196
197
  			tmp = realloc(buf, buf_size);
  			if (tmp == NULL)
  				goto out_free;
  
  			buf = tmp;
7ad74b41e   Mitchell Krome   perf symbols: Fix...
198
  			fseek(fp, offset, SEEK_SET);
b691f6436   Namhyung Kim   perf symbols: Imp...
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
  			if (fread(buf, buf_size, 1, fp) != 1)
  				goto out_free;
  
  			ret = read_build_id(buf, buf_size, bf, size, need_swap);
  			if (ret == 0)
  				ret = size;
  			break;
  		}
  	}
  out_free:
  	free(buf);
  out:
  	fclose(fp);
  	return ret;
  }
  
  int sysfs__read_build_id(const char *filename, void *build_id, size_t size)
  {
  	int fd;
  	int ret = -1;
  	struct stat stbuf;
  	size_t buf_size;
  	void *buf;
  
  	fd = open(filename, O_RDONLY);
  	if (fd < 0)
  		return -1;
  
  	if (fstat(fd, &stbuf) < 0)
  		goto out;
  
  	buf_size = stbuf.st_size;
  	buf = malloc(buf_size);
  	if (buf == NULL)
  		goto out;
  
  	if (read(fd, buf, buf_size) != (ssize_t) buf_size)
  		goto out_free;
  
  	ret = read_build_id(buf, buf_size, build_id, size, false);
  out_free:
  	free(buf);
  out:
  	close(fd);
  	return ret;
  }
18425f13a   Arnaldo Carvalho de Melo   perf symbols: Sav...
245
  int symsrc__init(struct symsrc *ss, struct dso *dso, const char *name,
b68e2f919   Cody P Schafer   perf symbols: Int...
246
247
248
249
  	         enum dso_binary_type type)
  {
  	int fd = open(name, O_RDONLY);
  	if (fd < 0)
18425f13a   Arnaldo Carvalho de Melo   perf symbols: Sav...
250
  		goto out_errno;
b68e2f919   Cody P Schafer   perf symbols: Int...
251
252
253
254
  
  	ss->name = strdup(name);
  	if (!ss->name)
  		goto out_close;
779e24e2c   Adrian Hunter   perf symbols: Fix...
255
  	ss->fd = fd;
b68e2f919   Cody P Schafer   perf symbols: Int...
256
257
258
259
260
  	ss->type = type;
  
  	return 0;
  out_close:
  	close(fd);
18425f13a   Arnaldo Carvalho de Melo   perf symbols: Sav...
261
262
  out_errno:
  	dso->load_errno = errno;
b68e2f919   Cody P Schafer   perf symbols: Int...
263
264
  	return -1;
  }
1d037ca16   Irina Tirdea   perf tools: Use _...
265
  bool symsrc__possibly_runtime(struct symsrc *ss __maybe_unused)
3aafe5ae0   Cody P Schafer   perf symbols: Use...
266
267
268
269
  {
  	/* Assume all sym sources could be a runtime image. */
  	return true;
  }
1d037ca16   Irina Tirdea   perf tools: Use _...
270
  bool symsrc__has_symtab(struct symsrc *ss __maybe_unused)
d26cd12b4   Cody P Schafer   perf symbols: Fac...
271
272
273
  {
  	return false;
  }
b68e2f919   Cody P Schafer   perf symbols: Int...
274
275
  void symsrc__destroy(struct symsrc *ss)
  {
74cf249d5   Arnaldo Carvalho de Melo   perf tools: Use z...
276
  	zfree(&ss->name);
b68e2f919   Cody P Schafer   perf symbols: Int...
277
278
  	close(ss->fd);
  }
1d037ca16   Irina Tirdea   perf tools: Use _...
279
280
  int dso__synthesize_plt_symbols(struct dso *dso __maybe_unused,
  				struct symsrc *ss __maybe_unused,
be39db9f2   Arnaldo Carvalho de Melo   perf symbols: Rem...
281
  				struct map *map __maybe_unused)
393be2e37   Namhyung Kim   perf symbols: Sup...
282
283
284
  {
  	return 0;
  }
c6d8f2a4a   Adrian Hunter   perf symbols: Rec...
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
  static int fd__is_64_bit(int fd)
  {
  	u8 e_ident[EI_NIDENT];
  
  	if (lseek(fd, 0, SEEK_SET))
  		return -1;
  
  	if (readn(fd, e_ident, sizeof(e_ident)) != sizeof(e_ident))
  		return -1;
  
  	if (memcmp(e_ident, ELFMAG, SELFMAG) ||
  	    e_ident[EI_VERSION] != EV_CURRENT)
  		return -1;
  
  	return e_ident[EI_CLASS] == ELFCLASS64;
  }
2b5b8bb27   Adrian Hunter   perf tools: Add d...
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
  enum dso_type dso__type_fd(int fd)
  {
  	Elf64_Ehdr ehdr;
  	int ret;
  
  	ret = fd__is_64_bit(fd);
  	if (ret < 0)
  		return DSO__TYPE_UNKNOWN;
  
  	if (ret)
  		return DSO__TYPE_64BIT;
  
  	if (readn(fd, &ehdr, sizeof(ehdr)) != sizeof(ehdr))
  		return DSO__TYPE_UNKNOWN;
  
  	if (ehdr.e_machine == EM_X86_64)
  		return DSO__TYPE_X32BIT;
  
  	return DSO__TYPE_32BIT;
  }
1d037ca16   Irina Tirdea   perf tools: Use _...
321
322
323
  int dso__load_sym(struct dso *dso, struct map *map __maybe_unused,
  		  struct symsrc *ss,
  		  struct symsrc *runtime_ss __maybe_unused,
1d037ca16   Irina Tirdea   perf tools: Use _...
324
  		  int kmodule __maybe_unused)
393be2e37   Namhyung Kim   perf symbols: Sup...
325
  {
f2f309688   Dima Kogan   perf symbols: Fix...
326
  	unsigned char build_id[BUILD_ID_SIZE];
c6d8f2a4a   Adrian Hunter   perf symbols: Rec...
327
328
329
330
331
  	int ret;
  
  	ret = fd__is_64_bit(ss->fd);
  	if (ret >= 0)
  		dso->is_64_bit = ret;
b691f6436   Namhyung Kim   perf symbols: Imp...
332

b68e2f919   Cody P Schafer   perf symbols: Int...
333
  	if (filename__read_build_id(ss->name, build_id, BUILD_ID_SIZE) > 0) {
b691f6436   Namhyung Kim   perf symbols: Imp...
334
  		dso__set_build_id(dso, build_id);
b691f6436   Namhyung Kim   perf symbols: Imp...
335
  	}
393be2e37   Namhyung Kim   perf symbols: Sup...
336
337
  	return 0;
  }
8e0cf965f   Adrian Hunter   perf symbols: Add...
338
339
340
341
342
343
  int file__read_maps(int fd __maybe_unused, bool exe __maybe_unused,
  		    mapfn_t mapfn __maybe_unused, void *data __maybe_unused,
  		    bool *is_64_bit __maybe_unused)
  {
  	return -1;
  }
afba19d9d   Adrian Hunter   perf symbols: Wor...
344
345
346
347
348
349
350
351
  int kcore_extract__create(struct kcore_extract *kce __maybe_unused)
  {
  	return -1;
  }
  
  void kcore_extract__delete(struct kcore_extract *kce __maybe_unused)
  {
  }
fc1b691d7   Adrian Hunter   perf buildid-cach...
352
353
354
355
356
  int kcore_copy(const char *from_dir __maybe_unused,
  	       const char *to_dir __maybe_unused)
  {
  	return -1;
  }
393be2e37   Namhyung Kim   perf symbols: Sup...
357
358
359
  void symbol__elf_init(void)
  {
  }
a64489c56   Jin Yao   perf report: Find...
360
361
362
363
364
365
366
  
  char *dso__demangle_sym(struct dso *dso __maybe_unused,
  			int kmodule __maybe_unused,
  			char *elf_name __maybe_unused)
  {
  	return NULL;
  }