Clean-up: add missing new line after end of function
[babeltrace.git] / lib / bin-info.c
CommitLineData
c40a57e5 1/*
d5ddf820 2 * bin-info.c
c40a57e5
AB
3 *
4 * Babeltrace - Executable and Shared Object Debug Info Reader
5 *
6 * Copyright 2015 Antoine Busque <abusque@efficios.com>
7 *
8 * Author: Antoine Busque <abusque@efficios.com>
9 *
10 * Permission is hereby granted, free of charge, to any person obtaining a copy
11 * of this software and associated documentation files (the "Software"), to deal
12 * in the Software without restriction, including without limitation the rights
13 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
14 * copies of the Software, and to permit persons to whom the Software is
15 * furnished to do so, subject to the following conditions:
16 *
17 * The above copyright notice and this permission notice shall be included in
18 * all copies or substantial portions of the Software.
19 *
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
23 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
26 * SOFTWARE.
27 */
28
29#include <fcntl.h>
30#include <math.h>
31#include <libgen.h>
32#include <stdio.h>
33#include <inttypes.h>
34#include <stdlib.h>
35#include <string.h>
36#include <unistd.h>
37#include <dwarf.h>
38#include <glib.h>
39#include <babeltrace/dwarf.h>
d5ddf820 40#include <babeltrace/bin-info.h>
c40a57e5
AB
41#include <babeltrace/crc32.h>
42#include <babeltrace/babeltrace-internal.h>
55cd033d 43#include <babeltrace/utils.h>
c40a57e5
AB
44
45/*
46 * An address printed in hex is at most 20 bytes (16 for 64-bits +
47 * leading 0x + optional leading '+' if addr is an offset + null
48 * character).
49 */
50#define ADDR_STR_LEN 20
51
52BT_HIDDEN
d5ddf820 53int bin_info_init(void)
c40a57e5
AB
54{
55 int ret = 0;
56
57 if (elf_version(EV_CURRENT) == EV_NONE) {
6764ed54
AB
58 printf_debug("ELF library initialization failed: %s\n",
59 elf_errmsg(-1));
c40a57e5
AB
60 ret = -1;
61 }
62
63 return ret;
64}
65
66BT_HIDDEN
d5ddf820 67struct bin_info *bin_info_create(const char *path, uint64_t low_addr,
9f2b13ca 68 uint64_t memsz, bool is_pic)
c40a57e5 69{
d5ddf820 70 struct bin_info *bin = NULL;
c40a57e5
AB
71
72 if (!path) {
73 goto error;
74 }
75
d5ddf820
AB
76 bin = g_new0(struct bin_info, 1);
77 if (!bin) {
c40a57e5
AB
78 goto error;
79 }
80
5cde0dc1 81 if (opt_debug_info_target_prefix) {
d5ddf820 82 bin->elf_path = g_build_path("/", opt_debug_info_target_prefix,
5cde0dc1
AB
83 path, NULL);
84 } else {
d5ddf820 85 bin->elf_path = strdup(path);
5cde0dc1
AB
86 }
87
d5ddf820 88 if (!bin->elf_path) {
c40a57e5
AB
89 goto error;
90 }
91
d5ddf820
AB
92 bin->is_pic = is_pic;
93 bin->memsz = memsz;
94 bin->low_addr = low_addr;
95 bin->high_addr = bin->low_addr + bin->memsz;
c40a57e5 96
d5ddf820 97 return bin;
c40a57e5
AB
98
99error:
d5ddf820 100 bin_info_destroy(bin);
c40a57e5
AB
101 return NULL;
102}
103
104BT_HIDDEN
d5ddf820 105void bin_info_destroy(struct bin_info *bin)
c40a57e5 106{
d5ddf820 107 if (!bin) {
c40a57e5
AB
108 return;
109 }
110
d5ddf820 111 dwarf_end(bin->dwarf_info);
c40a57e5 112
d5ddf820
AB
113 free(bin->elf_path);
114 free(bin->dwarf_path);
115 free(bin->build_id);
116 free(bin->dbg_link_filename);
c40a57e5 117
d5ddf820 118 elf_end(bin->elf_file);
c40a57e5 119
d5ddf820
AB
120 close(bin->elf_fd);
121 close(bin->dwarf_fd);
c40a57e5 122
d5ddf820 123 g_free(bin);
c40a57e5
AB
124}
125
49824faa 126
c40a57e5 127BT_HIDDEN
d5ddf820 128int bin_info_set_build_id(struct bin_info *bin, uint8_t *build_id,
c40a57e5
AB
129 size_t build_id_len)
130{
d5ddf820 131 if (!bin || !build_id) {
c40a57e5
AB
132 goto error;
133 }
134
d5ddf820
AB
135 bin->build_id = malloc(build_id_len);
136 if (!bin->build_id) {
c40a57e5
AB
137 goto error;
138 }
139
d5ddf820
AB
140 memcpy(bin->build_id, build_id, build_id_len);
141 bin->build_id_len = build_id_len;
c40a57e5
AB
142
143 /*
144 * Reset the is_elf_only flag in case it had been set
145 * previously, because we might find separate debug info using
146 * the new build id information.
147 */
d5ddf820 148 bin->is_elf_only = false;
c40a57e5
AB
149
150 return 0;
151
152error:
153
154 return -1;
155}
156
157BT_HIDDEN
d5ddf820 158int bin_info_set_debug_link(struct bin_info *bin, char *filename, uint32_t crc)
c40a57e5 159{
d5ddf820 160 if (!bin || !filename) {
c40a57e5
AB
161 goto error;
162 }
163
d5ddf820
AB
164 bin->dbg_link_filename = strdup(filename);
165 if (!bin->dbg_link_filename) {
c40a57e5
AB
166 goto error;
167 }
168
d5ddf820 169 bin->dbg_link_crc = crc;
c40a57e5
AB
170
171 /*
172 * Reset the is_elf_only flag in case it had been set
173 * previously, because we might find separate debug info using
174 * the new build id information.
175 */
d5ddf820 176 bin->is_elf_only = false;
c40a57e5
AB
177
178 return 0;
179
180error:
181
182 return -1;
183}
184
185/**
186 * Tries to read DWARF info from the location given by path, and
d5ddf820 187 * attach it to the given bin_info instance if it exists.
c40a57e5 188 *
d5ddf820 189 * @param bin bin_info instance for which to set DWARF info
c40a57e5
AB
190 * @param path Presumed location of the DWARF info
191 * @returns 0 on success, -1 on failure
192 */
193static
d5ddf820 194int bin_info_set_dwarf_info_from_path(struct bin_info *bin, char *path)
c40a57e5
AB
195{
196 int fd = -1, ret = 0;
197 struct bt_dwarf_cu *cu = NULL;
198 Dwarf *dwarf_info = NULL;
199
d5ddf820 200 if (!bin || !path) {
c40a57e5
AB
201 goto error;
202 }
203
204 fd = open(path, O_RDONLY);
205 if (fd < 0) {
206 goto error;
207 }
208
209 dwarf_info = dwarf_begin(fd, DWARF_C_READ);
210 if (!dwarf_info) {
211 goto error;
212 }
213
214 /*
d5ddf820
AB
215 * Check if the dwarf info has any CU. If not, the
216 * executable's object file contains no DWARF info.
c40a57e5
AB
217 */
218 cu = bt_dwarf_cu_create(dwarf_info);
219 if (!cu) {
220 goto error;
221 }
222
223 ret = bt_dwarf_cu_next(cu);
224 if (ret) {
225 goto error;
226 }
227
d5ddf820
AB
228 bin->dwarf_fd = fd;
229 bin->dwarf_path = strdup(path);
230 if (!bin->dwarf_path) {
c40a57e5
AB
231 goto error;
232 }
d5ddf820 233 bin->dwarf_info = dwarf_info;
c40a57e5
AB
234 free(cu);
235
236 return 0;
237
238error:
239 close(fd);
240 dwarf_end(dwarf_info);
241 g_free(dwarf_info);
242 free(cu);
243
244 return -1;
245}
246
247/**
d5ddf820 248 * Try to set the dwarf_info for a given bin_info instance via the
c40a57e5
AB
249 * build ID method.
250 *
d5ddf820 251 * @param bin bin_info instance for which to retrieve the
c40a57e5
AB
252 * DWARF info via build ID
253 * @returns 0 on success (i.e. dwarf_info set), -1 on failure
254 */
255static
d5ddf820 256int bin_info_set_dwarf_info_build_id(struct bin_info *bin)
c40a57e5 257{
f06ce5e5 258 int i = 0, ret = 0;
c40a57e5
AB
259 char *path = NULL, *build_id_file = NULL;
260 const char *dbg_dir = NULL;
f06ce5e5 261 size_t build_id_file_len;
c40a57e5 262
d5ddf820 263 if (!bin || !bin->build_id) {
c40a57e5
AB
264 goto error;
265 }
266
05984e0c 267 dbg_dir = opt_debug_info_dir ? : DEFAULT_DEBUG_DIR;
c40a57e5 268
b3599eb3 269 /* 2 characters per byte printed in hex, +1 for '/' and +1 for '\0' */
d5ddf820 270 build_id_file_len = (2 * bin->build_id_len) + 1 +
f06ce5e5 271 strlen(BUILD_ID_SUFFIX) + 1;
c40a57e5
AB
272 build_id_file = malloc(build_id_file_len);
273 if (!build_id_file) {
274 goto error;
275 }
276
d5ddf820
AB
277 snprintf(build_id_file, 4, "%02x/", bin->build_id[0]);
278 for (i = 1; i < bin->build_id_len; ++i) {
c40a57e5
AB
279 int path_idx = 3 + 2 * (i - 1);
280
d5ddf820 281 snprintf(&build_id_file[path_idx], 3, "%02x", bin->build_id[i]);
c40a57e5 282 }
f06ce5e5 283 strcat(build_id_file, BUILD_ID_SUFFIX);
c40a57e5 284
f06ce5e5 285 path = g_build_path("/", dbg_dir, BUILD_ID_SUBDIR, build_id_file, NULL);
c40a57e5
AB
286 if (!path) {
287 goto error;
288 }
289
d5ddf820 290 ret = bin_info_set_dwarf_info_from_path(bin, path);
c40a57e5
AB
291 if (ret) {
292 goto error;
293 }
294
295 goto end;
296
297error:
298 ret = -1;
299end:
300 free(build_id_file);
301 free(path);
302
303 return ret;
304}
305
306/**
307 * Tests whether the file located at path exists and has the expected
308 * checksum.
309 *
310 * This predicate is used when looking up separate debug info via the
311 * GNU debuglink method. The expected crc can be found .gnu_debuglink
312 * section in the original ELF file, along with the filename for the
313 * file containing the debug info.
314 *
315 * @param path Full path at which to look for the debug file
316 * @param crc Expected checksum for the debug file
317 * @returns 1 if the file exists and has the correct checksum,
318 * 0 otherwise
319 */
320static
321int is_valid_debug_file(char *path, uint32_t crc)
322{
323 int ret = 0, fd = -1;
324 uint32_t _crc = 0;
325
326 if (!path) {
327 goto end;
328 }
329
330 fd = open(path, O_RDONLY);
331 if (fd < 0) {
332 goto end;
333 }
334
335 ret = crc32(fd, &_crc);
336 if (ret) {
337 ret = 0;
338 goto end;
339 }
340
341 ret = (crc == _crc);
342
343end:
344 close(fd);
345 return ret;
346}
347
348/**
d5ddf820 349 * Try to set the dwarf_info for a given bin_info instance via the
c40a57e5
AB
350 * build ID method.
351 *
d5ddf820 352 * @param bin bin_info instance for which to retrieve the
c40a57e5
AB
353 * DWARF info via debug link
354 * @returns 0 on success (i.e. dwarf_info set), -1 on failure
355 */
356static
d5ddf820 357int bin_info_set_dwarf_info_debug_link(struct bin_info *bin)
c40a57e5
AB
358{
359 int ret = 0;
360 const char *dbg_dir = NULL;
d5ddf820 361 char *dir_name = NULL, *bin_dir = NULL, *path = NULL;
c40a57e5
AB
362 size_t max_path_len = 0;
363
d5ddf820 364 if (!bin || !bin->dbg_link_filename) {
c40a57e5
AB
365 goto error;
366 }
367
05984e0c 368 dbg_dir = opt_debug_info_dir ? : DEFAULT_DEBUG_DIR;
c40a57e5 369
d5ddf820 370 dir_name = dirname(bin->elf_path);
c40a57e5
AB
371 if (!dir_name) {
372 goto error;
373 }
374
d5ddf820
AB
375 /* bin_dir is just dir_name with a trailing slash */
376 bin_dir = malloc(strlen(dir_name) + 2);
377 if (!bin_dir) {
c40a57e5
AB
378 goto error;
379 }
380
d5ddf820
AB
381 strcpy(bin_dir, dir_name);
382 strcat(bin_dir, "/");
c40a57e5 383
d5ddf820
AB
384 max_path_len = strlen(dbg_dir) + strlen(bin_dir) +
385 strlen(DEBUG_SUBDIR) + strlen(bin->dbg_link_filename)
c40a57e5
AB
386 + 1;
387 path = malloc(max_path_len);
388 if (!path) {
389 goto error;
390 }
391
d5ddf820
AB
392 /* First look in the executable's dir */
393 strcpy(path, bin_dir);
394 strcat(path, bin->dbg_link_filename);
c40a57e5 395
d5ddf820 396 if (is_valid_debug_file(path, bin->dbg_link_crc)) {
c40a57e5
AB
397 goto found;
398 }
399
400 /* If not found, look in .debug subdir */
d5ddf820 401 strcpy(path, bin_dir);
c40a57e5 402 strcat(path, DEBUG_SUBDIR);
d5ddf820 403 strcat(path, bin->dbg_link_filename);
c40a57e5 404
d5ddf820 405 if (is_valid_debug_file(path, bin->dbg_link_crc)) {
c40a57e5
AB
406 goto found;
407 }
408
409 /* Lastly, look under the global debug directory */
410 strcpy(path, dbg_dir);
d5ddf820
AB
411 strcat(path, bin_dir);
412 strcat(path, bin->dbg_link_filename);
c40a57e5 413
d5ddf820 414 if (is_valid_debug_file(path, bin->dbg_link_crc)) {
c40a57e5
AB
415 goto found;
416 }
417
418error:
419 ret = -1;
420end:
421 free(path);
d5ddf820 422 free(bin_dir);
c40a57e5
AB
423
424 return ret;
425
426found:
d5ddf820 427 ret = bin_info_set_dwarf_info_from_path(bin, path);
c40a57e5
AB
428 if (ret) {
429 goto error;
430 }
431
432 goto end;
433}
434
435/**
436 * Initialize the DWARF info for a given executable.
437 *
d5ddf820 438 * @param bin bin_info instance
c40a57e5
AB
439 * @returns 0 on success, -1 on failure
440 */
441static
d5ddf820 442int bin_info_set_dwarf_info(struct bin_info *bin)
c40a57e5
AB
443{
444 int ret = 0;
445
d5ddf820 446 if (!bin) {
c40a57e5
AB
447 goto error;
448 }
449
450 /* First try to set the DWARF info from the ELF file */
d5ddf820 451 ret = bin_info_set_dwarf_info_from_path(bin, bin->elf_path);
c40a57e5
AB
452 if (!ret) {
453 goto end;
454 }
455
456 /*
457 * If that fails, try to find separate debug info via build ID
458 * and debug link.
459 */
d5ddf820 460 ret = bin_info_set_dwarf_info_build_id(bin);
c40a57e5
AB
461 if (!ret) {
462 goto end;
463 }
464
d5ddf820 465 ret = bin_info_set_dwarf_info_debug_link(bin);
c40a57e5
AB
466 if (!ret) {
467 goto end;
468 }
469
470error:
471 ret = -1;
472end:
473 return ret;
474}
475
49824faa
AB
476/**
477 * Initialize the ELF file for a given executable.
478 *
d5ddf820 479 * @param bin bin_info instance
49824faa
AB
480 * @returns 0 on success, -1 on failure
481 */
482static
d5ddf820 483int bin_info_set_elf_file(struct bin_info *bin)
49824faa
AB
484{
485 int elf_fd;
f99dc219 486 Elf *elf_file = NULL;
49824faa 487
d5ddf820 488 if (!bin) {
49824faa
AB
489 goto error;
490 }
491
d5ddf820 492 elf_fd = open(bin->elf_path, O_RDONLY);
49824faa 493 if (elf_fd < 0) {
d5ddf820 494 printf_verbose("Failed to open %s\n", bin->elf_path);
49824faa
AB
495 goto error;
496 }
497
498 elf_file = elf_begin(elf_fd, ELF_C_READ, NULL);
499 if (!elf_file) {
6764ed54 500 printf_debug("elf_begin failed: %s\n", elf_errmsg(-1));
49824faa
AB
501 goto error;
502 }
503
504 if (elf_kind(elf_file) != ELF_K_ELF) {
6764ed54 505 printf_verbose("Error: %s is not an ELF object\n",
d5ddf820 506 bin->elf_path);
49824faa
AB
507 goto error;
508 }
509
d5ddf820
AB
510 bin->elf_fd = elf_fd;
511 bin->elf_file = elf_file;
49824faa
AB
512 return 0;
513
514error:
515 close(elf_fd);
516 elf_end(elf_file);
517 return -1;
518}
519
c40a57e5
AB
520BT_HIDDEN
521void source_location_destroy(struct source_location *src_loc)
522{
523 if (!src_loc) {
524 return;
525 }
526
527 free(src_loc->filename);
528 g_free(src_loc);
529}
d6d3f4e8 530
d2ac1099
AB
531/**
532 * Append a string representation of an address offset to an existing
533 * string.
534 *
535 * On success, the out parameter `result` will contain the base string
536 * followed by the offset string of the form "+0x1234". On failure,
537 * `result` remains unchanged.
538 *
539 * @param base_str The string to which to append an offset string
540 * @param low_addr The lower virtual memory address, the base from
541 * which the offset is computed
542 * @param high_addr The higher virtual memory address
543 * @param result Out parameter, the base string followed by the
544 * offset string
545 * @returns 0 on success, -1 on failure
546 */
547static
d5ddf820 548int bin_info_append_offset_str(const char *base_str, uint64_t low_addr,
d2ac1099
AB
549 uint64_t high_addr, char **result)
550{
551 int ret;
552 uint64_t offset;
553 char *_result = NULL;
554 char offset_str[ADDR_STR_LEN];
555
556 if (!base_str || !result) {
557 goto error;
558 }
559
560 offset = high_addr - low_addr;
561
562 _result = malloc(strlen(base_str) + ADDR_STR_LEN);
563 if (!_result) {
564 goto error;
565 }
566
567 ret = snprintf(offset_str, ADDR_STR_LEN, "+%#0" PRIx64, offset);
568 if (ret < 0) {
569 goto error;
570 }
571 strcpy(_result, base_str);
572 strcat(_result, offset_str);
573 *result = _result;
574
575 return 0;
576
577error:
578 free(_result);
579 return -1;
580}
c40a57e5
AB
581
582/**
583 * Try to find the symbol closest to an address within a given ELF
584 * section.
585 *
586 * Only function symbols are taken into account. The symbol's address
587 * must precede `addr`. A symbol with a closer address might exist
588 * after `addr` but is irrelevant because it cannot encompass `addr`.
589 *
590 * On success, if found, the out parameters `sym` and `shdr` are
591 * set. On failure or if none are found, they remain unchanged.
592 *
593 * @param scn ELF section in which to look for the address
594 * @param addr Virtual memory address for which to find the
595 * nearest function symbol
596 * @param sym Out parameter, the nearest function symbol
597 * @param shdr Out parameter, the section header for scn
598 * @returns 0 on success, -1 on failure
599 */
600static
d5ddf820 601int bin_info_get_nearest_symbol_from_section(Elf_Scn *scn, uint64_t addr,
c40a57e5
AB
602 GElf_Sym **sym, GElf_Shdr **shdr)
603{
604 int i;
605 size_t symbol_count;
606 Elf_Data *data = NULL;
607 GElf_Shdr *_shdr = NULL;
608 GElf_Sym *nearest_sym = NULL;
609
610 if (!scn || !sym || !shdr) {
611 goto error;
612 }
613
614 _shdr = g_new0(GElf_Shdr, 1);
615 if (!_shdr) {
616 goto error;
617 }
618
619 _shdr = gelf_getshdr(scn, _shdr);
620 if (!_shdr) {
621 goto error;
622 }
623
624 if (_shdr->sh_type != SHT_SYMTAB) {
625 /*
626 * We are only interested in symbol table (symtab)
627 * sections, skip this one.
628 */
629 goto end;
630 }
631
632 data = elf_getdata(scn, NULL);
633 if (!data) {
634 goto error;
635 }
636
637 symbol_count = _shdr->sh_size / _shdr->sh_entsize;
638
639 for (i = 0; i < symbol_count; ++i) {
640 GElf_Sym *cur_sym = NULL;
641
642 cur_sym = g_new0(GElf_Sym, 1);
643 if (!cur_sym) {
644 goto error;
645 }
646 cur_sym = gelf_getsym(data, i, cur_sym);
647 if (!cur_sym) {
648 goto error;
649 }
650 if (GELF_ST_TYPE(cur_sym->st_info) != STT_FUNC) {
651 /* We're only interested in the functions. */
652 g_free(cur_sym);
653 continue;
654 }
655
656 if (cur_sym->st_value <= addr &&
657 (!nearest_sym ||
658 cur_sym->st_value > nearest_sym->st_value)) {
659 g_free(nearest_sym);
660 nearest_sym = cur_sym;
661 } else {
662 g_free(cur_sym);
663 }
664 }
665
666end:
667 if (nearest_sym) {
668 *sym = nearest_sym;
669 *shdr = _shdr;
670 } else {
671 g_free(_shdr);
672 }
673
674 return 0;
675
676error:
677 g_free(nearest_sym);
678 g_free(_shdr);
679 return -1;
680}
681
682/**
683 * Get the name of the function containing a given address within an
684 * executable using ELF symbols.
685 *
686 * The function name is in fact the name of the nearest ELF symbol,
687 * followed by the offset in bytes between the address and the symbol
688 * (in hex), separated by a '+' character.
689 *
690 * If found, the out parameter `func_name` is set on success. On failure,
691 * it remains unchanged.
692 *
d5ddf820 693 * @param bin bin_info instance for the executable containing
c40a57e5
AB
694 * the address
695 * @param addr Virtual memory address for which to find the
696 * function name
697 * @param func_name Out parameter, the function name
698 * @returns 0 on success, -1 on failure
699 */
700static
d5ddf820 701int bin_info_lookup_elf_function_name(struct bin_info *bin, uint64_t addr,
c40a57e5
AB
702 char **func_name)
703{
704 /*
705 * TODO (possible optimisation): if an ELF has no symtab
706 * section, it has been stripped. Therefore, it would be wise
707 * to store a flag indicating the stripped status after the
708 * first iteration to prevent subsequent ones.
709 */
710 int ret = 0;
711 Elf_Scn *scn = NULL;
712 GElf_Sym *sym = NULL;
713 GElf_Shdr *shdr = NULL;
714 char *sym_name = NULL;
c40a57e5 715
49824faa 716 /* Set ELF file if it hasn't been accessed yet. */
d5ddf820
AB
717 if (!bin->elf_file) {
718 ret = bin_info_set_elf_file(bin);
49824faa
AB
719 if (ret) {
720 /* Failed to set ELF file. */
721 goto error;
722 }
723 }
724
d5ddf820 725 scn = elf_nextscn(bin->elf_file, scn);
c40a57e5
AB
726 if (!scn) {
727 goto error;
728 }
729
730 while (scn && !sym) {
d5ddf820 731 ret = bin_info_get_nearest_symbol_from_section(
c40a57e5
AB
732 scn, addr, &sym, &shdr);
733 if (ret) {
734 goto error;
735 }
736
d5ddf820 737 scn = elf_nextscn(bin->elf_file, scn);
c40a57e5
AB
738 }
739
740 if (sym) {
d5ddf820 741 sym_name = elf_strptr(bin->elf_file, shdr->sh_link,
c40a57e5
AB
742 sym->st_name);
743 if (!sym_name) {
744 goto error;
745 }
746
d5ddf820 747 ret = bin_info_append_offset_str(sym_name, sym->st_value, addr,
d2ac1099
AB
748 func_name);
749 if (ret) {
c40a57e5
AB
750 goto error;
751 }
c40a57e5
AB
752 }
753
754 g_free(shdr);
755 g_free(sym);
756 return 0;
757
758error:
759 g_free(shdr);
760 g_free(sym);
c40a57e5
AB
761 return -1;
762}
763
764/**
765 * Get the name of the function containing a given address within a
766 * given compile unit (CU).
767 *
768 * If found, the out parameter `func_name` is set on success. On
769 * failure, it remains unchanged.
770 *
771 * @param cu bt_dwarf_cu instance which may contain the address
772 * @param addr Virtual memory address for which to find the
773 * function name
774 * @param func_name Out parameter, the function name
775 * @returns 0 on success, -1 on failure
776 */
777static
d5ddf820 778int bin_info_lookup_cu_function_name(struct bt_dwarf_cu *cu, uint64_t addr,
c40a57e5
AB
779 char **func_name)
780{
a54aa699
AB
781 int ret = 0;
782 bool found = false;
c40a57e5
AB
783 struct bt_dwarf_die *die = NULL;
784
785 if (!cu || !func_name) {
786 goto error;
787 }
788
789 die = bt_dwarf_die_create(cu);
790 if (!die) {
791 goto error;
792 }
793
794 while (bt_dwarf_die_next(die) == 0) {
795 int tag;
796
797 ret = bt_dwarf_die_get_tag(die, &tag);
798 if (ret) {
799 goto error;
800 }
801
802 if (tag == DW_TAG_subprogram) {
803 ret = bt_dwarf_die_contains_addr(die, addr, &found);
804 if (ret) {
805 goto error;
806 }
807
808 if (found) {
809 break;
810 }
811 }
812 }
813
814 if (found) {
d2ac1099
AB
815 uint64_t low_addr = 0;
816 char *die_name = NULL;
817
7935ee7a
AB
818 ret = bt_dwarf_die_get_name(die, &die_name);
819 if (ret) {
820 goto error;
821 }
822
823 ret = dwarf_lowpc(die->dwarf_die, &low_addr);
c40a57e5
AB
824 if (ret) {
825 goto error;
826 }
827
d5ddf820 828 ret = bin_info_append_offset_str(die_name, low_addr, addr,
d2ac1099
AB
829 func_name);
830 if (ret) {
7935ee7a
AB
831 goto error;
832 }
c40a57e5
AB
833 }
834
835 bt_dwarf_die_destroy(die);
836 return 0;
837
838error:
839 bt_dwarf_die_destroy(die);
840 return -1;
841}
842
843/**
844 * Get the name of the function containing a given address within an
845 * executable using DWARF debug info.
846 *
847 * If found, the out parameter `func_name` is set on success. On
848 * failure, it remains unchanged.
849 *
d5ddf820 850 * @param bin bin_info instance for the executable containing
c40a57e5
AB
851 * the address
852 * @param addr Virtual memory address for which to find the
853 * function name
854 * @param func_name Out parameter, the function name
855 * @returns 0 on success, -1 on failure
856 */
857static
d5ddf820 858int bin_info_lookup_dwarf_function_name(struct bin_info *bin, uint64_t addr,
c40a57e5
AB
859 char **func_name)
860{
861 int ret = 0;
862 char *_func_name = NULL;
863 struct bt_dwarf_cu *cu = NULL;
864
d5ddf820 865 if (!bin || !func_name) {
c40a57e5
AB
866 goto error;
867 }
868
d5ddf820 869 cu = bt_dwarf_cu_create(bin->dwarf_info);
c40a57e5
AB
870 if (!cu) {
871 goto error;
872 }
873
874 while (bt_dwarf_cu_next(cu) == 0) {
d5ddf820 875 ret = bin_info_lookup_cu_function_name(cu, addr, &_func_name);
c40a57e5
AB
876 if (ret) {
877 goto error;
878 }
879
880 if (_func_name) {
881 break;
882 }
883 }
884
885 if (_func_name) {
886 *func_name = _func_name;
93d65223
JG
887 } else {
888 goto error;
c40a57e5
AB
889 }
890
891 bt_dwarf_cu_destroy(cu);
892 return 0;
893
894error:
895 bt_dwarf_cu_destroy(cu);
896 return -1;
897}
898
899BT_HIDDEN
d5ddf820 900int bin_info_lookup_function_name(struct bin_info *bin, uint64_t addr,
c40a57e5
AB
901 char **func_name)
902{
903 int ret = 0;
904 char *_func_name = NULL;
905
d5ddf820 906 if (!bin || !func_name) {
c40a57e5
AB
907 goto error;
908 }
909
910 /* Set DWARF info if it hasn't been accessed yet. */
d5ddf820
AB
911 if (!bin->dwarf_info && !bin->is_elf_only) {
912 ret = bin_info_set_dwarf_info(bin);
c40a57e5
AB
913 if (ret) {
914 /* Failed to set DWARF info, fallback to ELF. */
d5ddf820 915 bin->is_elf_only = true;
c40a57e5
AB
916 }
917 }
918
d5ddf820 919 if (!bin_info_has_address(bin, addr)) {
c40a57e5
AB
920 goto error;
921 }
922
923 /*
924 * Addresses in ELF and DWARF are relative to base address for
925 * PIC, so make the address argument relative too if needed.
926 */
d5ddf820
AB
927 if (bin->is_pic) {
928 addr -= bin->low_addr;
36ae9941
AB
929 }
930
d5ddf820
AB
931 if (bin->is_elf_only) {
932 ret = bin_info_lookup_elf_function_name(bin, addr, &_func_name);
c40a57e5 933 } else {
d5ddf820 934 ret = bin_info_lookup_dwarf_function_name(bin, addr, &_func_name);
c40a57e5
AB
935 }
936
36ae9941 937 if (ret || !_func_name) {
c40a57e5
AB
938 goto error;
939 }
940
36ae9941
AB
941 *func_name = _func_name;
942 return 0;
55cd033d 943
36ae9941
AB
944error:
945 return -1;
946}
947
948BT_HIDDEN
d5ddf820 949int bin_info_get_bin_loc(struct bin_info *bin, uint64_t addr, char **bin_loc)
36ae9941
AB
950{
951 int ret = 0;
952 char *_bin_loc = NULL;
953
d5ddf820 954 if (!bin || !bin_loc) {
36ae9941 955 goto error;
c40a57e5
AB
956 }
957
d5ddf820
AB
958 if (bin->is_pic) {
959 addr -= bin->low_addr;
36ae9941
AB
960 ret = asprintf(&_bin_loc, "+%#0" PRIx64, addr);
961 } else {
962 ret = asprintf(&_bin_loc, "@%#0" PRIx64, addr);
963 }
964
965 if (ret == -1 || !_bin_loc) {
966 goto error;
967 }
968
969 *bin_loc = _bin_loc;
c40a57e5
AB
970 return 0;
971
972error:
973 return -1;
974}
975
976/**
977 * Predicate used to determine whether the children of a given DIE
978 * contain a specific address.
979 *
980 * More specifically, the parameter `die` is expected to be a
981 * subprogram (function) DIE, and this predicate tells whether any
982 * subroutines are inlined within this function and would contain
983 * `addr`.
984 *
a54aa699
AB
985 * On success, the out parameter `contains` is set with the boolean
986 * value indicating whether the DIE's range covers `addr`. On failure,
987 * it remains unchanged.
988 *
c40a57e5
AB
989 * Do note that this function advances the position of `die`. If the
990 * address is found within one of its children, `die` will be pointing
991 * to that child upon returning from the function, allowing to extract
992 * the information deemed necessary.
993 *
a54aa699
AB
994 * @param die The parent DIE in whose children the address will be
995 * looked for
996 * @param addr The address for which to look for in the DIEs
997 * @param contains Out parameter, true if addr is contained,
998 * false if not
999 * @returns Returns 0 on success, -1 on failure
c40a57e5
AB
1000 */
1001static
a54aa699 1002int bin_info_child_die_has_address(struct bt_dwarf_die *die, uint64_t addr, bool *contains)
c40a57e5 1003{
a54aa699
AB
1004 int ret = 0;
1005 bool _contains = false;
c40a57e5
AB
1006
1007 if (!die) {
1008 goto error;
1009 }
1010
1011 ret = bt_dwarf_die_child(die);
1012 if (ret) {
1013 goto error;
1014 }
1015
1016 do {
1017 int tag;
1018
1019 ret = bt_dwarf_die_get_tag(die, &tag);
1020 if (ret) {
1021 goto error;
1022 }
1023
1024 if (tag == DW_TAG_inlined_subroutine) {
a54aa699 1025 ret = bt_dwarf_die_contains_addr(die, addr, &_contains);
c40a57e5
AB
1026 if (ret) {
1027 goto error;
1028 }
1029
1030 if (contains) {
c40a57e5
AB
1031 goto end;
1032 }
1033 }
1034 } while (bt_dwarf_die_next(die) == 0);
1035
1036end:
a54aa699
AB
1037 *contains = _contains;
1038 return 0;
c40a57e5
AB
1039
1040error:
a54aa699 1041 return -1;
c40a57e5
AB
1042}
1043
1044/**
1045 * Lookup the source location for a given address within a CU, making
1046 * the assumption that it is contained within an inline routine in a
1047 * function.
1048 *
1049 * @param cu bt_dwarf_cu instance in which to look for the address
1050 * @param addr The address for which to look for
1051 * @param src_loc Out parameter, the source location (filename and
1052 * line number) for the address
1053 * @returns 0 on success, -1 on failure
1054 */
1055static
d5ddf820 1056int bin_info_lookup_cu_src_loc_inl(struct bt_dwarf_cu *cu, uint64_t addr,
c40a57e5
AB
1057 struct source_location **src_loc)
1058{
a54aa699
AB
1059 int ret = 0;
1060 bool found = false;
c40a57e5
AB
1061 struct bt_dwarf_die *die = NULL;
1062 struct source_location *_src_loc = NULL;
1063
1064 if (!cu || !src_loc) {
1065 goto error;
1066 }
1067
1068 die = bt_dwarf_die_create(cu);
1069 if (!die) {
1070 goto error;
1071 }
1072
1073 while (bt_dwarf_die_next(die) == 0) {
1074 int tag;
1075
1076 ret = bt_dwarf_die_get_tag(die, &tag);
1077 if (ret) {
1078 goto error;
1079 }
1080
1081 if (tag == DW_TAG_subprogram) {
a54aa699 1082 bool contains = false;
c40a57e5
AB
1083
1084 ret = bt_dwarf_die_contains_addr(die, addr, &contains);
1085 if (ret) {
1086 goto error;
1087 }
1088
1089 if (contains) {
1090 /*
1091 * Try to find an inlined subroutine
1092 * child of this DIE containing addr.
1093 */
a54aa699
AB
1094 ret = bin_info_child_die_has_address(die, addr,
1095 &found);
1096 if(ret) {
1097 goto error;
1098 }
1099
c40a57e5
AB
1100 goto end;
1101 }
1102 }
1103 }
1104
1105end:
1106 if (found) {
1107 char *filename = NULL;
1108 uint64_t line_no;
1109
1110 _src_loc = g_new0(struct source_location, 1);
1111 if (!_src_loc) {
1112 goto error;
1113 }
1114
1115 ret = bt_dwarf_die_get_call_file(die, &filename);
1116 if (ret) {
1117 goto error;
1118 }
1119 ret = bt_dwarf_die_get_call_line(die, &line_no);
1120 if (ret) {
1121 free(filename);
1122 goto error;
1123 }
1124
1125 _src_loc->filename = filename;
1126 _src_loc->line_no = line_no;
1127 *src_loc = _src_loc;
1128 }
1129
1130 bt_dwarf_die_destroy(die);
1131 return 0;
1132
1133error:
1134 source_location_destroy(_src_loc);
1135 bt_dwarf_die_destroy(die);
1136 return -1;
1137}
1138
1139/**
1140 * Lookup the source location for a given address within a CU,
1141 * assuming that it is contained within an inlined function.
1142 *
1143 * A source location can be found regardless of inlining status for
1144 * this method, but in the case of an inlined function, the returned
1145 * source location will point not to the callsite but rather to the
1146 * definition site of the inline function.
1147 *
1148 * @param cu bt_dwarf_cu instance in which to look for the address
1149 * @param addr The address for which to look for
1150 * @param src_loc Out parameter, the source location (filename and
1151 * line number) for the address
1152 * @returns 0 on success, -1 on failure
1153 */
1154static
d5ddf820 1155int bin_info_lookup_cu_src_loc_no_inl(struct bt_dwarf_cu *cu, uint64_t addr,
c40a57e5
AB
1156 struct source_location **src_loc)
1157{
1158 struct source_location *_src_loc = NULL;
1159 struct bt_dwarf_die *die = NULL;
1160 const char *filename = NULL;
1161 Dwarf_Line *line = NULL;
1162 Dwarf_Addr line_addr;
1163 int ret, line_no;
1164
1165 if (!cu || !src_loc) {
1166 goto error;
1167 }
1168
1169 die = bt_dwarf_die_create(cu);
1170 if (!die) {
1171 goto error;
1172 }
1173
1174 line = dwarf_getsrc_die(die->dwarf_die, addr);
1175 if (!line) {
1176 goto error;
1177 }
1178
1179 ret = dwarf_lineaddr(line, &line_addr);
1180 if (ret) {
1181 goto error;
1182 }
1183
1184 filename = dwarf_linesrc(line, NULL, NULL);
1185 if (!filename) {
1186 goto error;
1187 }
1188
1189 if (addr == line_addr) {
1190 _src_loc = g_new0(struct source_location, 1);
1191 if (!_src_loc) {
1192 goto error;
1193 }
1194
1195 ret = dwarf_lineno(line, &line_no);
1196 if (ret) {
1197 goto error;
1198 }
1199
1200 _src_loc->line_no = line_no;
1201 _src_loc->filename = strdup(filename);
1202 }
1203
1204 bt_dwarf_die_destroy(die);
1205
1206 if (_src_loc) {
1207 *src_loc = _src_loc;
1208 }
1209
1210 return 0;
1211
1212error:
1213 source_location_destroy(_src_loc);
1214 bt_dwarf_die_destroy(die);
1215 return -1;
1216}
1217
1218/**
1219 * Get the source location (file name and line number) for a given
1220 * address within a compile unit (CU).
1221 *
1222 * On success, the out parameter `src_loc` is set if found. On
1223 * failure, it remains unchanged.
1224 *
d5ddf820 1225 * @param cu bt_dwarf_cu instance for the compile unit which
c40a57e5
AB
1226 * may contain the address
1227 * @param addr Virtual memory address for which to find the
1228 * source location
1229 * @param src_loc Out parameter, the source location
1230 * @returns 0 on success, -1 on failure
1231 */
1232static
d5ddf820 1233int bin_info_lookup_cu_src_loc(struct bt_dwarf_cu *cu, uint64_t addr,
c40a57e5
AB
1234 struct source_location **src_loc)
1235{
1236 int ret = 0;
1237 struct source_location *_src_loc = NULL;
1238
1239 if (!cu || !src_loc) {
1240 goto error;
1241 }
1242
d5ddf820 1243 ret = bin_info_lookup_cu_src_loc_inl(cu, addr, &_src_loc);
c40a57e5
AB
1244 if (ret) {
1245 goto error;
1246 }
1247
1248 if (_src_loc) {
1249 goto end;
1250 }
1251
d5ddf820 1252 ret = bin_info_lookup_cu_src_loc_no_inl(cu, addr, &_src_loc);
c40a57e5
AB
1253 if (ret) {
1254 goto error;
1255 }
1256
1257 if (_src_loc) {
1258 goto end;
1259 }
1260
1261end:
1262 if (_src_loc) {
1263 *src_loc = _src_loc;
1264 }
1265
1266 return 0;
1267
1268error:
1269 source_location_destroy(_src_loc);
1270 return -1;
1271}
1272
1273BT_HIDDEN
d5ddf820 1274int bin_info_lookup_source_location(struct bin_info *bin, uint64_t addr,
c40a57e5
AB
1275 struct source_location **src_loc)
1276{
1277 struct bt_dwarf_cu *cu = NULL;
1278 struct source_location *_src_loc = NULL;
1279
d5ddf820 1280 if (!bin || !src_loc) {
c40a57e5
AB
1281 goto error;
1282 }
1283
1284 /* Set DWARF info if it hasn't been accessed yet. */
d5ddf820
AB
1285 if (!bin->dwarf_info && !bin->is_elf_only) {
1286 if (bin_info_set_dwarf_info(bin)) {
c40a57e5 1287 /* Failed to set DWARF info. */
d5ddf820 1288 bin->is_elf_only = true;
c40a57e5
AB
1289 }
1290 }
1291
d5ddf820 1292 if (bin->is_elf_only) {
c40a57e5
AB
1293 /* We cannot lookup source location without DWARF info. */
1294 goto error;
1295 }
1296
d5ddf820 1297 if (!bin_info_has_address(bin, addr)) {
c40a57e5
AB
1298 goto error;
1299 }
1300
1301 /*
1302 * Addresses in ELF and DWARF are relative to base address for
1303 * PIC, so make the address argument relative too if needed.
1304 */
d5ddf820
AB
1305 if (bin->is_pic) {
1306 addr -= bin->low_addr;
c40a57e5
AB
1307 }
1308
d5ddf820 1309 cu = bt_dwarf_cu_create(bin->dwarf_info);
c40a57e5
AB
1310 if (!cu) {
1311 goto error;
1312 }
1313
1314 while (bt_dwarf_cu_next(cu) == 0) {
1315 int ret;
1316
d5ddf820 1317 ret = bin_info_lookup_cu_src_loc(cu, addr, &_src_loc);
c40a57e5
AB
1318 if (ret) {
1319 goto error;
1320 }
1321
1322 if (_src_loc) {
1323 break;
1324 }
1325 }
1326
1327 bt_dwarf_cu_destroy(cu);
1328 if (_src_loc) {
1329 *src_loc = _src_loc;
1330 }
1331
1332 return 0;
1333
1334error:
1335 source_location_destroy(_src_loc);
1336 bt_dwarf_cu_destroy(cu);
1337 return -1;
1338}
This page took 0.078624 seconds and 4 git commands to generate.