Branch data Line data Source code
1 : : /* Print symbol information from ELF file in human-readable form.
2 : : Copyright (C) 2000-2008, 2009, 2011, 2012 Red Hat, Inc.
3 : : This file is part of elfutils.
4 : : Written by Ulrich Drepper <drepper@redhat.com>, 2000.
5 : :
6 : : This file is free software; you can redistribute it and/or modify
7 : : it under the terms of the GNU General Public License as published by
8 : : the Free Software Foundation; either version 3 of the License, or
9 : : (at your option) any later version.
10 : :
11 : : elfutils is distributed in the hope that it will be useful, but
12 : : WITHOUT ANY WARRANTY; without even the implied warranty of
13 : : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : : GNU General Public License for more details.
15 : :
16 : : You should have received a copy of the GNU General Public License
17 : : along with this program. If not, see <http://www.gnu.org/licenses/>. */
18 : :
19 : : #ifdef HAVE_CONFIG_H
20 : : # include <config.h>
21 : : #endif
22 : :
23 : : #include <ar.h>
24 : : #include <argp.h>
25 : : #include <assert.h>
26 : : #include <ctype.h>
27 : : #include <dwarf.h>
28 : : #include <errno.h>
29 : : #include <error.h>
30 : : #include <fcntl.h>
31 : : #include <gelf.h>
32 : : #include <inttypes.h>
33 : : #include <libdw.h>
34 : : #include <libintl.h>
35 : : #include <locale.h>
36 : : #include <mcheck.h>
37 : : #include <obstack.h>
38 : : #include <search.h>
39 : : #include <stdbool.h>
40 : : #include <stdio.h>
41 : : #include <stdio_ext.h>
42 : : #include <stdlib.h>
43 : : #include <string.h>
44 : : #include <unistd.h>
45 : : #include <sys/param.h>
46 : :
47 : : #include <system.h>
48 : : #include "../libebl/libeblP.h"
49 : :
50 : :
51 : : /* Name and version of program. */
52 : : static void print_version (FILE *stream, struct argp_state *state);
53 : : ARGP_PROGRAM_VERSION_HOOK_DEF = print_version;
54 : :
55 : : /* Bug report address. */
56 : : ARGP_PROGRAM_BUG_ADDRESS_DEF = PACKAGE_BUGREPORT;
57 : :
58 : :
59 : : /* Values for the parameters which have no short form. */
60 : : #define OPT_DEFINED 0x100
61 : : #define OPT_MARK_SPECIAL 0x101
62 : :
63 : : /* Definitions of arguments for argp functions. */
64 : : static const struct argp_option options[] =
65 : : {
66 : : { NULL, 0, NULL, 0, N_("Output selection:"), 0 },
67 : : { "debug-syms", 'a', NULL, 0, N_("Display debugger-only symbols"), 0 },
68 : : { "defined-only", OPT_DEFINED, NULL, 0, N_("Display only defined symbols"),
69 : : 0 },
70 : : { "dynamic", 'D', NULL, 0,
71 : : N_("Display dynamic symbols instead of normal symbols"), 0 },
72 : : { "extern-only", 'g', NULL, 0, N_("Display only external symbols"), 0 },
73 : : { "undefined-only", 'u', NULL, 0, N_("Display only undefined symbols"), 0 },
74 : : { "print-armap", 's', NULL, 0,
75 : : N_("Include index for symbols from archive members"), 0 },
76 : :
77 : : { NULL, 0, NULL, 0, N_("Output format:"), 0 },
78 : : { "print-file-name", 'A', NULL, 0,
79 : : N_("Print name of the input file before every symbol"), 0 },
80 : : { NULL, 'o', NULL, OPTION_HIDDEN, "Same as -A", 0 },
81 : : { "format", 'f', "FORMAT", 0,
82 : : N_("Use the output format FORMAT. FORMAT can be `bsd', `sysv' or `posix'. The default is `sysv'"),
83 : : 0 },
84 : : { NULL, 'B', NULL, 0, N_("Same as --format=bsd"), 0 },
85 : : { "portability", 'P', NULL, 0, N_("Same as --format=posix"), 0 },
86 : : { "radix", 't', "RADIX", 0, N_("Use RADIX for printing symbol values"), 0 },
87 : : { "mark-special", OPT_MARK_SPECIAL, NULL, 0, N_("Mark special symbols"), 0 },
88 : : { "mark-weak", OPT_MARK_SPECIAL, NULL, OPTION_HIDDEN, "", 0 },
89 : : { "print-size", 'S', NULL, 0, N_("Print size of defined symbols"), 0 },
90 : :
91 : : { NULL, 0, NULL, 0, N_("Output options:"), 0 },
92 : : { "numeric-sort", 'n', NULL, 0, N_("Sort symbols numerically by address"),
93 : : 0 },
94 : : { "no-sort", 'p', NULL, 0, N_("Do not sort the symbols"), 0 },
95 : : { "reverse-sort", 'r', NULL, 0, N_("Reverse the sense of the sort"), 0 },
96 : : #ifdef USE_DEMANGLE
97 : : { "demangle", 'C', NULL, 0,
98 : : N_("Decode low-level symbol names into source code names"), 0 },
99 : : #endif
100 : : { NULL, 0, NULL, 0, N_("Miscellaneous:"), 0 },
101 : : { NULL, 0, NULL, 0, NULL, 0 }
102 : : };
103 : :
104 : : /* Short description of program. */
105 : : static const char doc[] = N_("List symbols from FILEs (a.out by default).");
106 : :
107 : : /* Strings for arguments in help texts. */
108 : : static const char args_doc[] = N_("[FILE...]");
109 : :
110 : : /* Prototype for option handler. */
111 : : static error_t parse_opt (int key, char *arg, struct argp_state *state);
112 : :
113 : : /* Parser children. */
114 : : static struct argp_child argp_children[] =
115 : : {
116 : : { &color_argp, 0, N_("Output formatting"), 2 },
117 : : { NULL, 0, NULL, 0}
118 : : };
119 : :
120 : : /* Data structure to communicate with argp functions. */
121 : : static struct argp argp =
122 : : {
123 : : options, parse_opt, args_doc, doc, argp_children, NULL, NULL
124 : : };
125 : :
126 : :
127 : : /* Print symbols in file named FNAME. */
128 : : static int process_file (const char *fname, bool more_than_one);
129 : :
130 : : /* Handle content of archive. */
131 : : static int handle_ar (int fd, Elf *elf, const char *prefix, const char *fname,
132 : : const char *suffix);
133 : :
134 : : /* Handle ELF file. */
135 : : static int handle_elf (Elf *elf, const char *prefix, const char *fname,
136 : : const char *suffix);
137 : :
138 : :
139 : : #define INTERNAL_ERROR(fname) \
140 : : error (EXIT_FAILURE, 0, gettext ("%s: INTERNAL ERROR %d (%s-%s): %s"), \
141 : : fname, __LINE__, PACKAGE_VERSION, __DATE__, elf_errmsg (-1))
142 : :
143 : :
144 : : /* Internal representation of symbols. */
145 : : typedef struct GElf_SymX
146 : : {
147 : : GElf_Sym sym;
148 : : Elf32_Word xndx;
149 : : char *where;
150 : : } GElf_SymX;
151 : :
152 : :
153 : : /* User-selectable options. */
154 : :
155 : : /* The selected output format. */
156 : : static enum
157 : : {
158 : : format_sysv = 0,
159 : : format_bsd,
160 : : format_posix
161 : : } format;
162 : :
163 : : /* Print defined, undefined, or both? */
164 : : static bool hide_undefined;
165 : : static bool hide_defined;
166 : :
167 : : /* Print local symbols also? */
168 : : static bool hide_local;
169 : :
170 : : /* Nonzero if full filename should precede every symbol. */
171 : : static bool print_file_name;
172 : :
173 : : /* If true print size of defined symbols in BSD format. */
174 : : static bool print_size;
175 : :
176 : : /* If true print archive index. */
177 : : static bool print_armap;
178 : :
179 : : /* If true reverse sorting. */
180 : : static bool reverse_sort;
181 : :
182 : : #ifdef USE_DEMANGLE
183 : : /* If true demangle symbols. */
184 : : static bool demangle;
185 : : #endif
186 : :
187 : : /* Type of the section we are printing. */
188 : : static GElf_Word symsec_type = SHT_SYMTAB;
189 : :
190 : : /* Sorting selection. */
191 : : static enum
192 : : {
193 : : sort_name = 0,
194 : : sort_numeric,
195 : : sort_nosort
196 : : } sort;
197 : :
198 : : /* Radix for printed numbers. */
199 : : static enum
200 : : {
201 : : radix_hex = 0,
202 : : radix_decimal,
203 : : radix_octal
204 : : } radix;
205 : :
206 : : /* If nonzero mark special symbols:
207 : : - weak symbols are distinguished from global symbols by adding
208 : : a `*' after the identifying letter for the symbol class and type.
209 : : - TLS symbols are distinguished from normal symbols by adding
210 : : a '@' after the identifying letter for the symbol class and type. */
211 : : static bool mark_special;
212 : :
213 : :
214 : : int
215 : 864 : main (int argc, char *argv[])
216 : : {
217 : : int remaining;
218 : 864 : int result = 0;
219 : :
220 : : /* Make memory leak detection possible. */
221 : 864 : mtrace ();
222 : :
223 : : /* We use no threads here which can interfere with handling a stream. */
224 : 864 : (void) __fsetlocking (stdin, FSETLOCKING_BYCALLER);
225 : 864 : (void) __fsetlocking (stdout, FSETLOCKING_BYCALLER);
226 : 864 : (void) __fsetlocking (stderr, FSETLOCKING_BYCALLER);
227 : :
228 : : /* Set locale. */
229 : 864 : (void) setlocale (LC_ALL, "");
230 : :
231 : : /* Make sure the message catalog can be found. */
232 : 864 : (void) bindtextdomain (PACKAGE_TARNAME, LOCALEDIR);
233 : :
234 : : /* Initialize the message catalog. */
235 : 864 : (void) textdomain (PACKAGE_TARNAME);
236 : :
237 : : /* Parse and process arguments. */
238 : 864 : (void) argp_parse (&argp, argc, argv, 0, &remaining, NULL);
239 : :
240 : : /* Tell the library which version we are expecting. */
241 : 864 : (void) elf_version (EV_CURRENT);
242 : :
243 [ - + ]: 864 : if (remaining == argc)
244 : : /* The user didn't specify a name so we use a.out. */
245 : 0 : result = process_file ("a.out", false);
246 : : else
247 : : {
248 : : /* Process all the remaining files. */
249 : 864 : const bool more_than_one = remaining + 1 < argc;
250 : :
251 : : do
252 : 864 : result |= process_file (argv[remaining], more_than_one);
253 [ - + ]: 864 : while (++remaining < argc);
254 : : }
255 : :
256 : : return result;
257 : : }
258 : :
259 : :
260 : : /* Print the version information. */
261 : : static void
262 : 0 : print_version (FILE *stream, struct argp_state *state __attribute__ ((unused)))
263 : : {
264 : 0 : fprintf (stream, "nm (%s) %s\n", PACKAGE_NAME, PACKAGE_VERSION);
265 : 0 : fprintf (stream, gettext ("\
266 : : Copyright (C) %s Red Hat, Inc.\n\
267 : : This is free software; see the source for copying conditions. There is NO\n\
268 : : warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
269 : : "), "2012");
270 : 0 : fprintf (stream, gettext ("Written by %s.\n"), "Ulrich Drepper");
271 : 0 : }
272 : :
273 : :
274 : : /* Handle program arguments. */
275 : : static error_t
276 : 6912 : parse_opt (int key, char *arg,
277 : : struct argp_state *state __attribute__ ((unused)))
278 : : {
279 [ + - + + : 6912 : switch (key)
+ + - - -
- + - + -
- - + + ]
280 : : {
281 : : case 'a':
282 : : /* XXX */
283 : : break;
284 : :
285 : : #ifdef USE_DEMANGLE
286 : : case 'C':
287 : 0 : demangle = true;
288 : 0 : break;
289 : : #endif
290 : :
291 : : case 'f':
292 [ + + ][ + - ]: 864 : if (strcmp (arg, "bsd") == 0)
[ + - ][ + + ]
293 : 288 : format = format_bsd;
294 [ + + ]: 576 : else if (strcmp (arg, "posix") == 0)
295 : 288 : format = format_posix;
296 : : else
297 : : /* Be bug compatible. The BFD implementation also defaulted to
298 : : using the SysV format if nothing else matches. */
299 : 288 : format = format_sysv;
300 : : break;
301 : :
302 : : case 'g':
303 : 216 : hide_local = true;
304 : 216 : break;
305 : :
306 : : case 'n':
307 : 288 : sort = sort_numeric;
308 : 288 : break;
309 : :
310 : : case 'p':
311 : 288 : sort = sort_nosort;
312 : 288 : break;
313 : :
314 : : case 't':
315 [ # # ][ # # ]: 0 : if (strcmp (arg, "10") == 0 || strcmp (arg, "d") == 0)
[ # # ][ # # ]
[ # # ]
316 : 0 : radix = radix_decimal;
317 [ # # ][ # # ]: 0 : else if (strcmp (arg, "8") == 0 || strcmp (arg, "o") == 0)
[ # # ][ # # ]
318 : 0 : radix = radix_octal;
319 : : else
320 : 0 : radix = radix_hex;
321 : : break;
322 : :
323 : : case 'u':
324 : 0 : hide_undefined = false;
325 : 0 : hide_defined = true;
326 : 0 : break;
327 : :
328 : : case 'A':
329 : : case 'o':
330 : 0 : print_file_name = true;
331 : 0 : break;
332 : :
333 : : case 'B':
334 : 0 : format = format_bsd;
335 : 0 : break;
336 : :
337 : : case 'D':
338 : 216 : symsec_type = SHT_DYNSYM;
339 : 216 : break;
340 : :
341 : : case 'P':
342 : 0 : format = format_posix;
343 : 0 : break;
344 : :
345 : : case OPT_DEFINED:
346 : 216 : hide_undefined = true;
347 : 216 : hide_defined = false;
348 : 216 : break;
349 : :
350 : : case OPT_MARK_SPECIAL:
351 : 0 : mark_special = true;
352 : 0 : break;
353 : :
354 : : case 'S':
355 : 0 : print_size = true;
356 : 0 : break;
357 : :
358 : : case 's':
359 : 0 : print_armap = true;
360 : 0 : break;
361 : :
362 : : case 'r':
363 : 288 : reverse_sort = true;
364 : 6912 : break;
365 : :
366 : : default:
367 : : return ARGP_ERR_UNKNOWN;
368 : : }
369 : : return 0;
370 : : }
371 : :
372 : :
373 : : /* Open the file and determine the type. */
374 : : static int
375 : 864 : process_file (const char *fname, bool more_than_one)
376 : : {
377 : : /* Open the file. */
378 : 864 : int fd = open (fname, O_RDONLY);
379 [ - + ]: 864 : if (fd == -1)
380 : : {
381 : 0 : error (0, errno, gettext ("cannot open '%s'"), fname);
382 : 0 : return 1;
383 : : }
384 : :
385 : : /* Now get the ELF descriptor. */
386 : 864 : Elf *elf = elf_begin (fd, ELF_C_READ_MMAP, NULL);
387 [ + - ]: 864 : if (elf != NULL)
388 : : {
389 [ + - ]: 864 : if (elf_kind (elf) == ELF_K_ELF)
390 : : {
391 [ + - ]: 864 : int result = handle_elf (elf, more_than_one ? "" : NULL,
392 : : fname, NULL);
393 : :
394 [ - + ]: 864 : if (elf_end (elf) != 0)
395 : 0 : INTERNAL_ERROR (fname);
396 : :
397 [ - + ]: 864 : if (close (fd) != 0)
398 : 0 : error (EXIT_FAILURE, errno, gettext ("while closing '%s'"), fname);
399 : :
400 : : return result;
401 : : }
402 [ # # ]: 0 : else if (elf_kind (elf) == ELF_K_AR)
403 : : {
404 : 0 : int result = handle_ar (fd, elf, NULL, fname, NULL);
405 : :
406 [ # # ]: 0 : if (elf_end (elf) != 0)
407 : 0 : INTERNAL_ERROR (fname);
408 : :
409 [ # # ]: 0 : if (close (fd) != 0)
410 : 0 : error (EXIT_FAILURE, errno, gettext ("while closing '%s'"), fname);
411 : :
412 : : return result;
413 : : }
414 : :
415 : : /* We cannot handle this type. Close the descriptor anyway. */
416 [ # # ]: 0 : if (elf_end (elf) != 0)
417 : 0 : INTERNAL_ERROR (fname);
418 : : }
419 : :
420 : 0 : error (0, 0, gettext ("%s: File format not recognized"), fname);
421 : :
422 : 864 : return 1;
423 : : }
424 : :
425 : :
426 : : static int
427 : 0 : handle_ar (int fd, Elf *elf, const char *prefix, const char *fname,
428 : : const char *suffix)
429 : : {
430 : 0 : size_t fname_len = strlen (fname) + 1;
431 [ # # ]: 0 : size_t prefix_len = prefix != NULL ? strlen (prefix) : 0;
432 : 0 : char new_prefix[prefix_len + fname_len + 2];
433 [ # # ]: 0 : size_t suffix_len = suffix != NULL ? strlen (suffix) : 0;
434 : 0 : char new_suffix[suffix_len + 2];
435 : : Elf *subelf;
436 : 0 : Elf_Cmd cmd = ELF_C_READ_MMAP;
437 : 0 : int result = 0;
438 : :
439 : 0 : char *cp = new_prefix;
440 [ # # ]: 0 : if (prefix != NULL)
441 : 0 : cp = stpcpy (cp, prefix);
442 : 0 : cp = stpcpy (cp, fname);
443 : 0 : stpcpy (cp, "[");
444 : :
445 : 0 : cp = new_suffix;
446 [ # # ]: 0 : if (suffix != NULL)
447 : 0 : cp = stpcpy (cp, suffix);
448 : 0 : stpcpy (cp, "]");
449 : :
450 : : /* First print the archive index if this is wanted. */
451 [ # # ]: 0 : if (print_armap)
452 : : {
453 : 0 : Elf_Arsym *arsym = elf_getarsym (elf, NULL);
454 : :
455 [ # # ]: 0 : if (arsym != NULL)
456 : : {
457 : 0 : Elf_Arhdr *arhdr = NULL;
458 : 0 : size_t arhdr_off = 0; /* Note: 0 is no valid offset. */
459 : :
460 : 0 : fputs_unlocked (gettext("\nArchive index:\n"), stdout);
461 : :
462 [ # # ]: 0 : while (arsym->as_off != 0)
463 : : {
464 [ # # ]: 0 : if (arhdr_off != arsym->as_off
465 [ # # ]: 0 : && (elf_rand (elf, arsym->as_off) != arsym->as_off
466 [ # # ]: 0 : || (subelf = elf_begin (fd, cmd, elf)) == NULL
467 [ # # ]: 0 : || (arhdr = elf_getarhdr (subelf)) == NULL))
468 : : {
469 : 0 : error (0, 0, gettext ("invalid offset %zu for symbol %s"),
470 : : arsym->as_off, arsym->as_name);
471 : 0 : continue;
472 : : }
473 : :
474 : 0 : printf (gettext ("%s in %s\n"), arsym->as_name, arhdr->ar_name);
475 : :
476 : 0 : ++arsym;
477 : : }
478 : :
479 [ # # ]: 0 : if (elf_rand (elf, SARMAG) != SARMAG)
480 : : {
481 : : error (0, 0,
482 : 0 : gettext ("cannot reset archive offset to beginning"));
483 : : return 1;
484 : : }
485 : : }
486 : : }
487 : :
488 : : /* Process all the files contained in the archive. */
489 [ # # ]: 0 : while ((subelf = elf_begin (fd, cmd, elf)) != NULL)
490 : : {
491 : : /* The the header for this element. */
492 : 0 : Elf_Arhdr *arhdr = elf_getarhdr (subelf);
493 : :
494 : : /* Skip over the index entries. */
495 [ # # ][ # # ]: 0 : if (strcmp (arhdr->ar_name, "/") != 0
496 [ # # ][ # # ]: 0 : && strcmp (arhdr->ar_name, "//") != 0)
[ # # ]
497 : : {
498 [ # # ]: 0 : if (elf_kind (subelf) == ELF_K_ELF)
499 : 0 : result |= handle_elf (subelf, new_prefix, arhdr->ar_name,
500 : : new_suffix);
501 [ # # ]: 0 : else if (elf_kind (subelf) == ELF_K_AR)
502 : 0 : result |= handle_ar (fd, subelf, new_prefix, arhdr->ar_name,
503 : : new_suffix);
504 : : else
505 : : {
506 : 0 : error (0, 0, gettext ("%s%s%s: file format not recognized"),
507 : : new_prefix, arhdr->ar_name, new_suffix);
508 : 0 : result = 1;
509 : : }
510 : : }
511 : :
512 : : /* Get next archive element. */
513 : 0 : cmd = elf_next (subelf);
514 [ # # ]: 0 : if (elf_end (subelf) != 0)
515 : 0 : INTERNAL_ERROR (fname);
516 : : }
517 : :
518 : : return result;
519 : : }
520 : :
521 : :
522 : : /* Mapping of radix and binary class to length. */
523 : : static const int length_map[2][3] =
524 : : {
525 : : [ELFCLASS32 - 1] =
526 : : {
527 : : [radix_hex] = 8,
528 : : [radix_decimal] = 10,
529 : : [radix_octal] = 11
530 : : },
531 : : [ELFCLASS64 - 1] =
532 : : {
533 : : [radix_hex] = 16,
534 : : [radix_decimal] = 20,
535 : : [radix_octal] = 22
536 : : }
537 : : };
538 : :
539 : :
540 : : static int
541 : 0 : global_compare (const void *p1, const void *p2)
542 : : {
543 : 0 : const Dwarf_Global *g1 = (const Dwarf_Global *) p1;
544 : 0 : const Dwarf_Global *g2 = (const Dwarf_Global *) p2;
545 : :
546 : 0 : return strcmp (g1->name, g2->name);
547 : : }
548 : :
549 : :
550 : : static void *global_root;
551 : :
552 : :
553 : : static int
554 : 0 : get_global (Dwarf *dbg __attribute__ ((unused)), Dwarf_Global *global,
555 : : void *arg __attribute__ ((unused)))
556 : : {
557 : 0 : tsearch (memcpy (xmalloc (sizeof (Dwarf_Global)), global,
558 : : sizeof (Dwarf_Global)),
559 : : &global_root, global_compare);
560 : :
561 : 0 : return DWARF_CB_OK;
562 : : }
563 : :
564 : :
565 : : struct local_name
566 : : {
567 : : const char *name;
568 : : const char *file;
569 : : Dwarf_Word lineno;
570 : : Dwarf_Addr lowpc;
571 : : Dwarf_Addr highpc;
572 : : };
573 : :
574 : :
575 : : static int
576 : 1146507 : local_compare (const void *p1, const void *p2)
577 : : {
578 : 1146507 : struct local_name *g1 = (struct local_name *) p1;
579 : 1146507 : struct local_name *g2 = (struct local_name *) p2;
580 : : int result;
581 : :
582 : 1146507 : result = strcmp (g1->name, g2->name);
583 [ + + ]: 1146507 : if (result == 0)
584 : : {
585 [ + + ][ + + ]: 28812 : if (g1->lowpc <= g2->lowpc && g1->highpc >= g2->highpc)
586 : : {
587 : : /* g2 is contained in g1. Update the data. */
588 : 3654 : g2->lowpc = g1->lowpc;
589 : 3654 : g2->highpc = g1->highpc;
590 : 3654 : result = 0;
591 : : }
592 [ + + ][ + + ]: 25158 : else if (g2->lowpc <= g1->lowpc && g2->highpc >= g1->highpc)
593 : : {
594 : : /* g1 is contained in g2. Update the data. */
595 : 24003 : g1->lowpc = g2->lowpc;
596 : 24003 : g1->highpc = g2->highpc;
597 : 24003 : result = 0;
598 : : }
599 : : else
600 [ + + ]: 1155 : result = g1->lowpc < g2->lowpc ? -1 : 1;
601 : : }
602 : :
603 : 1146507 : return result;
604 : : }
605 : :
606 : :
607 : : static int
608 : 30048 : get_var_range (Dwarf_Die *die, Dwarf_Word *lowpc, Dwarf_Word *highpc)
609 : : {
610 : : Dwarf_Attribute locattr_mem;
611 : 30048 : Dwarf_Attribute *locattr = dwarf_attr (die, DW_AT_location, &locattr_mem);
612 [ + + ]: 30048 : if (locattr == NULL)
613 : : return 1;
614 : :
615 : : Dwarf_Op *loc;
616 : : size_t nloc;
617 [ + - ]: 7080 : if (dwarf_getlocation (locattr, &loc, &nloc) != 0)
618 : : return 1;
619 : :
620 : : /* Interpret the location expressions. */
621 : : // XXX For now just the simple one:
622 [ + + ][ + - ]: 7080 : if (nloc == 1 && loc[0].atom == DW_OP_addr)
623 : : {
624 : 30048 : *lowpc = *highpc = loc[0].number;
625 : : return 0;
626 : : }
627 : :
628 : : return 1;
629 : : }
630 : :
631 : :
632 : :
633 : : static void *local_root;
634 : :
635 : :
636 : : static void
637 : 288 : get_local_names (Dwarf *dbg)
638 : : {
639 : 288 : Dwarf_Off offset = 0;
640 : : Dwarf_Off old_offset;
641 : : size_t hsize;
642 : :
643 [ + + ]: 14316 : while (dwarf_nextcu (dbg, old_offset = offset, &offset, &hsize, NULL, NULL,
644 : : NULL) == 0)
645 : : {
646 : : Dwarf_Die cudie_mem;
647 : 14028 : Dwarf_Die *cudie = dwarf_offdie (dbg, old_offset + hsize, &cudie_mem);
648 : :
649 : : /* If we cannot get the CU DIE there is no need to go on with
650 : : this CU. */
651 [ - + ]: 14028 : if (cudie == NULL)
652 : 0 : continue;
653 : : /* This better be a CU DIE. */
654 [ - + ]: 14028 : if (dwarf_tag (cudie) != DW_TAG_compile_unit)
655 : 0 : continue;
656 : :
657 : : /* Get the line information. */
658 : : Dwarf_Files *files;
659 : : size_t nfiles;
660 [ - + ]: 14028 : if (dwarf_getsrcfiles (cudie, &files, &nfiles) != 0)
661 : 0 : continue;
662 : :
663 : : Dwarf_Die die_mem;
664 : 14028 : Dwarf_Die *die = &die_mem;
665 [ + - ]: 14028 : if (dwarf_child (cudie, die) == 0)
666 : : /* Iterate over all immediate children of the CU DIE. */
667 : : do
668 : : {
669 : 1914240 : int tag = dwarf_tag (die);
670 [ + + ]: 1914240 : if (tag != DW_TAG_subprogram && tag != DW_TAG_variable)
671 : 1750860 : continue;
672 : :
673 : : /* We are interested in five attributes: name, decl_file,
674 : : decl_line, low_pc, and high_pc. */
675 : : Dwarf_Attribute attr_mem;
676 : 163380 : Dwarf_Attribute *attr = dwarf_attr (die, DW_AT_name, &attr_mem);
677 : 163380 : const char *name = dwarf_formstring (attr);
678 [ + + ]: 163380 : if (name == NULL)
679 : 1740 : continue;
680 : :
681 : : Dwarf_Word fileidx;
682 : 161640 : attr = dwarf_attr (die, DW_AT_decl_file, &attr_mem);
683 [ + + ][ - + ]: 161640 : if (dwarf_formudata (attr, &fileidx) != 0 || fileidx >= nfiles)
684 : 10656 : continue;
685 : :
686 : : Dwarf_Word lineno;
687 : 150984 : attr = dwarf_attr (die, DW_AT_decl_line, &attr_mem);
688 [ + - ][ + + ]: 150984 : if (dwarf_formudata (attr, &lineno) != 0 || lineno == 0)
689 : 10932 : continue;
690 : :
691 : : Dwarf_Addr lowpc;
692 : : Dwarf_Addr highpc;
693 [ + + ]: 140052 : if (tag == DW_TAG_subprogram)
694 : : {
695 [ + + ]: 110004 : if (dwarf_lowpc (die, &lowpc) != 0
696 [ - + ]: 36636 : || dwarf_highpc (die, &highpc) != 0)
697 : 73368 : continue;
698 : : }
699 : : else
700 : : {
701 [ + + ]: 30048 : if (get_var_range (die, &lowpc, &highpc) != 0)
702 : 23220 : continue;
703 : : }
704 : :
705 : : /* We have all the information. Create a record. */
706 : 43464 : struct local_name *newp
707 : : = (struct local_name *) xmalloc (sizeof (*newp));
708 : 43464 : newp->name = name;
709 : 43464 : newp->file = dwarf_filesrc (files, fileidx, NULL, NULL);
710 : 43464 : newp->lineno = lineno;
711 : 43464 : newp->lowpc = lowpc;
712 : 43464 : newp->highpc = highpc;
713 : :
714 : : /* Since we cannot deallocate individual memory we do not test
715 : : for duplicates in the tree. This should not happen anyway. */
716 [ - + ]: 43464 : if (tsearch (newp, &local_root, local_compare) == NULL)
717 : 0 : error (EXIT_FAILURE, errno,
718 : 0 : gettext ("cannot create search tree"));
719 : : }
720 [ + + ]: 1914240 : while (dwarf_siblingof (die, die) == 0);
721 : : }
722 : 288 : }
723 : :
724 : : /* Do elf_strptr, but return a backup string and never NULL. */
725 : : static const char *
726 : 316017 : sym_name (Elf *elf, GElf_Word strndx, GElf_Word st_name, char buf[], size_t n)
727 : : {
728 : 316017 : const char *symstr = elf_strptr (elf, strndx, st_name);
729 [ - + ]: 316017 : if (symstr == NULL)
730 : : {
731 : 0 : snprintf (buf, n, "[invalid st_name %#" PRIx32 "]", st_name);
732 : 0 : symstr = buf;
733 : : }
734 : 316017 : return symstr;
735 : : }
736 : :
737 : : /* Show symbols in SysV format. */
738 : : static void
739 : 288 : show_symbols_sysv (Ebl *ebl, GElf_Word strndx, const char *fullname,
740 : : GElf_SymX *syms, size_t nsyms, int longest_name,
741 : : int longest_where)
742 : : {
743 : : size_t shnum;
744 [ - + ]: 288 : if (elf_getshdrnum (ebl->elf, &shnum) < 0)
745 : 0 : INTERNAL_ERROR (fullname);
746 : :
747 : 288 : bool scnnames_malloced = shnum * sizeof (const char *) > 128 * 1024;
748 : : const char **scnnames;
749 [ - + ]: 288 : if (scnnames_malloced)
750 : 0 : scnnames = (const char **) xmalloc (sizeof (const char *) * shnum);
751 : : else
752 : 288 : scnnames = (const char **) alloca (sizeof (const char *) * shnum);
753 : : /* Get the section header string table index. */
754 : : size_t shstrndx;
755 [ + - ]: 288 : if (elf_getshdrstrndx (ebl->elf, &shstrndx) < 0)
756 : : error (EXIT_FAILURE, 0,
757 : 0 : gettext ("cannot get section header string table index"));
758 : :
759 : : /* Cache the section names. */
760 : : Elf_Scn *scn = NULL;
761 : : size_t cnt = 1;
762 [ + + ]: 10812 : while ((scn = elf_nextscn (ebl->elf, scn)) != NULL)
763 : : {
764 : : GElf_Shdr shdr_mem;
765 : :
766 [ - + ]: 10524 : assert (elf_ndxscn (scn) == cnt++);
767 : :
768 : 10524 : char *name = elf_strptr (ebl->elf, shstrndx,
769 : 10524 : gelf_getshdr (scn, &shdr_mem)->sh_name);
770 [ - + ]: 10524 : if (unlikely (name == NULL))
771 : : {
772 : 0 : name = alloca (sizeof "[invalid sh_name 0x12345678]");
773 : 0 : snprintf (name, sizeof name, "[invalid sh_name %#" PRIx32 "]",
774 : 0 : gelf_getshdr (scn, &shdr_mem)->sh_name);
775 : : }
776 : 10524 : scnnames[elf_ndxscn (scn)] = name;
777 : : }
778 : :
779 : 288 : int digits = length_map[gelf_getclass (ebl->elf) - 1][radix];
780 : :
781 : : /* We always print this prolog. */
782 : 288 : printf (gettext ("\n\nSymbols from %s:\n\n"), fullname);
783 : :
784 : : /* The header line. */
785 [ - + ]: 288 : printf (gettext ("%*s%-*s %-*s Class Type %-*s %*s Section\n\n"),
786 : 0 : print_file_name ? (int) strlen (fullname) + 1: 0, "",
787 [ + - ]: 288 : longest_name, sgettext ("sysv|Name"),
788 : : /* TRANS: the "sysv|" parts makes the string unique. */
789 [ + - ]: 288 : digits, sgettext ("sysv|Value"),
790 : : /* TRANS: the "sysv|" parts makes the string unique. */
791 [ + - ]: 288 : digits, sgettext ("sysv|Size"),
792 : : /* TRANS: the "sysv|" parts makes the string unique. */
793 [ + - ]: 288 : longest_where, sgettext ("sysv|Line"));
794 : :
795 : : /* Which format string to use (different radix for numbers). */
796 : : const char *number_fmtstr;
797 [ - + ]: 288 : if (radix == radix_hex)
798 : : number_fmtstr = "%0*" PRIx64;
799 [ # # ]: 0 : else if (radix == radix_decimal)
800 : : number_fmtstr = "%0*" PRId64;
801 : : else
802 : 0 : number_fmtstr = "%0*" PRIo64;
803 : :
804 : : #ifdef USE_DEMANGLE
805 : 288 : size_t demangle_buffer_len = 0;
806 : 288 : char *demangle_buffer = NULL;
807 : : #endif
808 : :
809 : : /* Iterate over all symbols. */
810 [ + + ]: 107046 : for (cnt = 1; cnt < nsyms; ++cnt)
811 : : {
812 : : /* In this format SECTION entries are not printed. */
813 [ + + ]: 106758 : if (GELF_ST_TYPE (syms[cnt].sym.st_info) == STT_SECTION)
814 : 4833 : continue;
815 : :
816 : : char symstrbuf[50];
817 : 101925 : const char *symstr = sym_name (ebl->elf, strndx, syms[cnt].sym.st_name,
818 : : symstrbuf, sizeof symstrbuf);
819 : :
820 : : #ifdef USE_DEMANGLE
821 : : /* Demangle if necessary. */
822 [ - + ]: 101925 : if (demangle)
823 : : {
824 : 0 : int status = -1;
825 : 0 : char *dmsymstr = __cxa_demangle (symstr, demangle_buffer,
826 : : &demangle_buffer_len, &status);
827 : :
828 [ # # ]: 0 : if (status == 0)
829 : 0 : symstr = dmsymstr;
830 : : }
831 : : #endif
832 : :
833 : : char symbindbuf[50];
834 : : char symtypebuf[50];
835 : : char secnamebuf[1024];
836 : : char addressbuf[(64 + 2) / 3 + 1];
837 : : char sizebuf[(64 + 2) / 3 + 1];
838 : :
839 : : /* If we have to precede the line with the file name. */
840 [ - + ]: 101925 : if (print_file_name)
841 : : {
842 : 0 : fputs_unlocked (fullname, stdout);
843 : 0 : putchar_unlocked (':');
844 : : }
845 : :
846 : : /* Covert the address. */
847 [ + + ]: 101925 : if (syms[cnt].sym.st_shndx == SHN_UNDEF)
848 : 11473 : addressbuf[0] = sizebuf[0] = '\0';
849 : : else
850 : : {
851 : 90452 : snprintf (addressbuf, sizeof (addressbuf), number_fmtstr,
852 : : digits, syms[cnt].sym.st_value);
853 : 90452 : snprintf (sizebuf, sizeof (sizebuf), number_fmtstr,
854 : : digits, syms[cnt].sym.st_size);
855 : : }
856 : :
857 : : /* Print the actual string. */
858 : 101925 : printf ("%-*s|%s|%-6s|%-8s|%s|%*s|%s\n",
859 : : longest_name, symstr, addressbuf,
860 : : ebl_symbol_binding_name (ebl,
861 : 101925 : GELF_ST_BIND (syms[cnt].sym.st_info),
862 : : symbindbuf, sizeof (symbindbuf)),
863 : 101925 : ebl_symbol_type_name (ebl, GELF_ST_TYPE (syms[cnt].sym.st_info),
864 : : symtypebuf, sizeof (symtypebuf)),
865 : : sizebuf, longest_where, syms[cnt].where,
866 : 203850 : ebl_section_name (ebl, syms[cnt].sym.st_shndx, syms[cnt].xndx,
867 : : secnamebuf, sizeof (secnamebuf), scnnames,
868 : : shnum));
869 : : }
870 : :
871 : : #ifdef USE_DEMANGLE
872 : : free (demangle_buffer);
873 : : #endif
874 : :
875 [ - + ]: 288 : if (scnnames_malloced)
876 : 0 : free (scnnames);
877 : 288 : }
878 : :
879 : :
880 : : static char
881 : 185247 : class_type_char (Elf *elf, const GElf_Ehdr *ehdr, GElf_Sym *sym)
882 : : {
883 : 185247 : int local_p = GELF_ST_BIND (sym->st_info) == STB_LOCAL;
884 : :
885 : : /* XXX Add support for architecture specific types and classes. */
886 [ + + ]: 185247 : if (sym->st_shndx == SHN_ABS)
887 [ + + ]: 9576 : return local_p ? 'a' : 'A';
888 : :
889 [ + + ]: 175671 : if (sym->st_shndx == SHN_UNDEF)
890 : : /* Undefined symbols must be global. */
891 : : return 'U';
892 : :
893 : 164178 : char result = "NDTSFBD "[GELF_ST_TYPE (sym->st_info)];
894 : :
895 [ + + ]: 164178 : if (result == 'D')
896 : : {
897 : : /* Special handling: unique data symbols. */
898 [ - + ]: 94314 : if (ehdr->e_ident[EI_OSABI] == ELFOSABI_LINUX
899 [ # # ]: 0 : && GELF_ST_BIND (sym->st_info) == STB_GNU_UNIQUE)
900 : : result = 'u';
901 : : else
902 : : {
903 : : GElf_Shdr shdr_mem;
904 : 94314 : GElf_Shdr *shdr = gelf_getshdr (elf_getscn (elf, sym->st_shndx),
905 : : &shdr_mem);
906 [ + - ]: 94314 : if (shdr != NULL)
907 : : {
908 [ + + ]: 94314 : if ((shdr->sh_flags & SHF_WRITE) == 0)
909 : : result = 'R';
910 [ + + ]: 84438 : else if (shdr->sh_type == SHT_NOBITS)
911 : 94314 : result = 'B';
912 : : }
913 : : }
914 : : }
915 : :
916 [ + + ]: 185247 : return local_p ? tolower (result) : result;
917 : : }
918 : :
919 : :
920 : : static void
921 : 288 : show_symbols_bsd (Elf *elf, const GElf_Ehdr *ehdr, GElf_Word strndx,
922 : : const char *prefix, const char *fname, const char *fullname,
923 : : GElf_SymX *syms, size_t nsyms)
924 : : {
925 : 288 : int digits = length_map[gelf_getclass (elf) - 1][radix];
926 : :
927 [ - + ][ # # ]: 288 : if (prefix != NULL && ! print_file_name)
928 : 0 : printf ("\n%s:\n", fname);
929 : :
930 : : static const char *const fmtstrs[] =
931 : : {
932 : : [radix_hex] = "%6$s%2$0*1$" PRIx64 "%8$s %7$s%3$c%4$s %5$s",
933 : : [radix_decimal] = "%6$s%*" PRId64 "%8$s %7$s%3$c%4$s %5$s",
934 : : [radix_octal] = "%6$s%2$0*1$" PRIo64 "%8$s %7$s%3$c%4$s %5$s"
935 : : };
936 : : static const char *const sfmtstrs[] =
937 : : {
938 : : [radix_hex] = "%6$s%2$0*1$" PRIx64 "%8$s %10$0*9$" PRIx64 " %7$s%3$c%4$s %5$s",
939 : : [radix_decimal] = "%6$s%2$*1$" PRId64 "%8$s %10$*9$" PRId64 " %7$s%3$c%4$s %5$s",
940 : : [radix_octal] = "%6$s%2$0*1$" PRIo64 "%8$s %10$0*9$" PRIo64 " %7$s%3$c%4$s %5$s"
941 : : };
942 : :
943 : : #ifdef USE_DEMANGLE
944 : 288 : size_t demangle_buffer_len = 0;
945 : 288 : char *demangle_buffer = NULL;
946 : : #endif
947 : :
948 : : /* Iterate over all symbols. */
949 [ + + ]: 107334 : for (size_t cnt = 0; cnt < nsyms; ++cnt)
950 : : {
951 : : char symstrbuf[50];
952 : 107046 : const char *symstr = sym_name (elf, strndx, syms[cnt].sym.st_name,
953 : : symstrbuf, sizeof symstrbuf);
954 : :
955 : : /* Printing entries with a zero-length name makes the output
956 : : not very well parseable. Since these entries don't carry
957 : : much information we leave them out. */
958 [ + + ]: 107046 : if (symstr[0] == '\0')
959 : 5025 : continue;
960 : :
961 : : /* We do not print the entries for files. */
962 [ + + ]: 102021 : if (GELF_ST_TYPE (syms[cnt].sym.st_info) == STT_FILE)
963 : 7302 : continue;
964 : :
965 : : #ifdef USE_DEMANGLE
966 : : /* Demangle if necessary. */
967 [ - + ]: 94719 : if (demangle)
968 : : {
969 : 0 : int status = -1;
970 : 0 : char *dmsymstr = __cxa_demangle (symstr, demangle_buffer,
971 : : &demangle_buffer_len, &status);
972 : :
973 [ # # ]: 0 : if (status == 0)
974 : 0 : symstr = dmsymstr;
975 : : }
976 : : #endif
977 : :
978 : : /* If we have to precede the line with the file name. */
979 [ - + ]: 94719 : if (print_file_name)
980 : : {
981 : 0 : fputs_unlocked (fullname, stdout);
982 : 0 : putchar_unlocked (':');
983 : : }
984 : :
985 : 94719 : bool is_tls = GELF_ST_TYPE (syms[cnt].sym.st_info) == STT_TLS;
986 : 94719 : bool is_weak = GELF_ST_BIND (syms[cnt].sym.st_info) == STB_WEAK;
987 : 94719 : const char *marker = (mark_special
988 [ - + ][ # # ]: 94719 : ? (is_tls ? "@" : (is_weak ? "*" : " ")) : "");
[ # # ]
989 : :
990 [ + + ]: 94719 : if (syms[cnt].sym.st_shndx == SHN_UNDEF)
991 : : {
992 : 11493 : const char *color = "";
993 [ - + ]: 11493 : if (color_mode)
994 : : {
995 [ # # ]: 0 : if (is_tls)
996 : 0 : color = color_undef_tls;
997 [ # # ]: 0 : else if (is_weak)
998 : 0 : color = color_undef_weak;
999 : : else
1000 : 0 : color = color_undef;
1001 : : }
1002 : :
1003 : 11493 : printf ("%*s %sU%s %s", digits, "", color, marker, symstr);
1004 : : }
1005 : : else
1006 : : {
1007 : 83226 : const char *color = "";
1008 [ - + ]: 83226 : if (color_mode)
1009 : : {
1010 [ # # ]: 0 : if (is_tls)
1011 : 0 : color = color_tls;
1012 [ # # ]: 0 : else if (is_weak)
1013 : 0 : color = color_weak;
1014 : : else
1015 : 0 : color = color_symbol;
1016 : : }
1017 : :
1018 [ + - ][ - + ]: 166452 : printf (print_size && syms[cnt].sym.st_size != 0
[ - + ][ # # ]
1019 : 83226 : ? sfmtstrs[radix] : fmtstrs[radix],
1020 : : digits, syms[cnt].sym.st_value,
1021 : 83226 : class_type_char (elf, ehdr, &syms[cnt].sym), marker,
1022 : : symstr,
1023 : : color_mode ? color_address : "",
1024 : : color,
1025 : : color_mode ? color_off : "",
1026 : : digits, (uint64_t) syms[cnt].sym.st_size);
1027 : : }
1028 : :
1029 [ - + ]: 94719 : if (color_mode)
1030 : 0 : fputs_unlocked (color_off, stdout);
1031 : 107046 : putchar_unlocked ('\n');
1032 : : }
1033 : :
1034 : : #ifdef USE_DEMANGLE
1035 : : free (demangle_buffer);
1036 : : #endif
1037 : 288 : }
1038 : :
1039 : :
1040 : : static void
1041 : 288 : show_symbols_posix (Elf *elf, const GElf_Ehdr *ehdr, GElf_Word strndx,
1042 : : const char *prefix, const char *fullname, GElf_SymX *syms,
1043 : : size_t nsyms)
1044 : : {
1045 [ - + ][ # # ]: 288 : if (prefix != NULL && ! print_file_name)
1046 : 0 : printf ("%s:\n", fullname);
1047 : :
1048 : : const char *fmtstr;
1049 [ - + ]: 288 : if (radix == radix_hex)
1050 : : fmtstr = "%s %c%s %0*" PRIx64 " %0*" PRIx64 "\n";
1051 [ # # ]: 0 : else if (radix == radix_decimal)
1052 : : fmtstr = "%s %c%s %*" PRId64 " %*" PRId64 "\n";
1053 : : else
1054 : 0 : fmtstr = "%s %c%s %0*" PRIo64 " %0*" PRIo64 "\n";
1055 : :
1056 : 288 : int digits = length_map[gelf_getclass (elf) - 1][radix];
1057 : :
1058 : : #ifdef USE_DEMANGLE
1059 : 288 : size_t demangle_buffer_len = 0;
1060 : 288 : char *demangle_buffer = NULL;
1061 : : #endif
1062 : :
1063 : : /* Iterate over all symbols. */
1064 [ + + ]: 107334 : for (size_t cnt = 0; cnt < nsyms; ++cnt)
1065 : : {
1066 : : char symstrbuf[50];
1067 : 107046 : const char *symstr = sym_name (elf, strndx, syms[cnt].sym.st_name,
1068 : : symstrbuf, sizeof symstrbuf);
1069 : :
1070 : : /* Printing entries with a zero-length name makes the output
1071 : : not very well parseable. Since these entries don't carry
1072 : : much information we leave them out. */
1073 [ + + ]: 107046 : if (symstr[0] == '\0')
1074 : 5025 : continue;
1075 : :
1076 : : #ifdef USE_DEMANGLE
1077 : : /* Demangle if necessary. */
1078 [ - + ]: 102021 : if (demangle)
1079 : : {
1080 : 0 : int status = -1;
1081 : 0 : char *dmsymstr = __cxa_demangle (symstr, demangle_buffer,
1082 : : &demangle_buffer_len, &status);
1083 : :
1084 [ # # ]: 0 : if (status == 0)
1085 : 0 : symstr = dmsymstr;
1086 : : }
1087 : : #endif
1088 : :
1089 : : /* If we have to precede the line with the file name. */
1090 [ - + ]: 102021 : if (print_file_name)
1091 : : {
1092 : 0 : fputs_unlocked (fullname, stdout);
1093 : 0 : putchar_unlocked (':');
1094 : 0 : putchar_unlocked (' ');
1095 : : }
1096 : :
1097 [ - + ]: 204042 : printf (fmtstr,
1098 : : symstr,
1099 : 102021 : class_type_char (elf, ehdr, &syms[cnt].sym),
1100 : : mark_special
1101 : 0 : ? (GELF_ST_TYPE (syms[cnt].sym.st_info) == STT_TLS
1102 : : ? "@"
1103 [ # # ]: 0 : : (GELF_ST_BIND (syms[cnt].sym.st_info) == STB_WEAK
1104 [ # # ]: 0 : ? "*" : " "))
1105 : : : "",
1106 : : digits, syms[cnt].sym.st_value,
1107 : : digits, syms[cnt].sym.st_size);
1108 : : }
1109 : :
1110 : : #ifdef USE_DEMANGLE
1111 : : free (demangle_buffer);
1112 : : #endif
1113 : 288 : }
1114 : :
1115 : :
1116 : : /* Maximum size of memory we allocate on the stack. */
1117 : : #define MAX_STACK_ALLOC 65536
1118 : :
1119 : : static int
1120 : 822738 : sort_by_address (const void *p1, const void *p2)
1121 : : {
1122 : 822738 : GElf_SymX *s1 = (GElf_SymX *) p1;
1123 : 822738 : GElf_SymX *s2 = (GElf_SymX *) p2;
1124 : :
1125 : 1645476 : int result = (s1->sym.st_value < s2->sym.st_value
1126 [ + + ]: 822738 : ? -1 : (s1->sym.st_value == s2->sym.st_value ? 0 : 1));
1127 : :
1128 [ - + ]: 822738 : return reverse_sort ? -result : result;
1129 : : }
1130 : :
1131 : : static Elf_Data *sort_by_name_strtab;
1132 : :
1133 : : static int
1134 : 856809 : sort_by_name (const void *p1, const void *p2)
1135 : : {
1136 : 856809 : GElf_SymX *s1 = (GElf_SymX *) p1;
1137 : 856809 : GElf_SymX *s2 = (GElf_SymX *) p2;
1138 : :
1139 : 856809 : const char *n1 = sort_by_name_strtab->d_buf + s1->sym.st_name;
1140 : 856809 : const char *n2 = sort_by_name_strtab->d_buf + s2->sym.st_name;
1141 : :
1142 : 856809 : int result = strcmp (n1, n2);
1143 : :
1144 [ + - ]: 856809 : return reverse_sort ? -result : result;
1145 : : }
1146 : :
1147 : : static void
1148 : 864 : show_symbols (Ebl *ebl, GElf_Ehdr *ehdr, Elf_Scn *scn, Elf_Scn *xndxscn,
1149 : : GElf_Shdr *shdr, const char *prefix, const char *fname,
1150 : : const char *fullname)
1151 : : {
1152 : : /* Get the section header string table index. */
1153 : : size_t shstrndx;
1154 [ - + ]: 864 : if (elf_getshdrstrndx (ebl->elf, &shstrndx) < 0)
1155 : : error (EXIT_FAILURE, 0,
1156 : 0 : gettext ("cannot get section header string table index"));
1157 : :
1158 : : /* The section is that large. */
1159 : 864 : size_t size = shdr->sh_size;
1160 : : /* One entry is this large. */
1161 : 864 : size_t entsize = shdr->sh_entsize;
1162 : :
1163 : : /* Consistency checks. */
1164 [ - + ]: 864 : if (entsize != gelf_fsize (ebl->elf, ELF_T_SYM, 1, ehdr->e_version))
1165 : 0 : error (0, 0,
1166 : 0 : gettext ("%s: entry size in section `%s' is not what we expect"),
1167 : 0 : fullname, elf_strptr (ebl->elf, shstrndx, shdr->sh_name));
1168 [ - + ]: 864 : else if (size % entsize != 0)
1169 : 0 : error (0, 0,
1170 : 0 : gettext ("%s: size of section `%s' is not multiple of entry size"),
1171 : 0 : fullname, elf_strptr (ebl->elf, shstrndx, shdr->sh_name));
1172 : :
1173 : : /* Compute number of entries. Handle buggy entsize values. */
1174 [ + - ]: 864 : size_t nentries = size / (entsize ?: 1);
1175 : :
1176 : :
1177 : : #define obstack_chunk_alloc xmalloc
1178 : : #define obstack_chunk_free free
1179 : : struct obstack whereob;
1180 : 864 : obstack_init (&whereob);
1181 : :
1182 : : /* Get a DWARF debugging descriptor. It's no problem if this isn't
1183 : : possible. We just won't print any line number information. */
1184 : 864 : Dwarf *dbg = NULL;
1185 [ + + ]: 864 : if (format == format_sysv)
1186 : : {
1187 : 288 : dbg = dwarf_begin_elf (ebl->elf, DWARF_C_READ, NULL);
1188 [ + - ]: 288 : if (dbg != NULL)
1189 : : {
1190 : 288 : (void) dwarf_getpubnames (dbg, get_global, NULL, 0);
1191 : :
1192 : 288 : get_local_names (dbg);
1193 : : }
1194 : : }
1195 : :
1196 : : /* Allocate the memory.
1197 : :
1198 : : XXX We can use a dirty trick here. Since GElf_Sym == Elf64_Sym we
1199 : : can use the data memory instead of copying again if what we read
1200 : : is a 64 bit file. */
1201 : : GElf_SymX *sym_mem;
1202 [ + + ]: 864 : if (nentries * sizeof (GElf_SymX) < MAX_STACK_ALLOC)
1203 : 783 : sym_mem = (GElf_SymX *) alloca (nentries * sizeof (GElf_SymX));
1204 : : else
1205 : 81 : sym_mem = (GElf_SymX *) xmalloc (nentries * sizeof (GElf_SymX));
1206 : :
1207 : : /* Get the data of the section. */
1208 : 864 : Elf_Data *data = elf_getdata (scn, NULL);
1209 : 864 : Elf_Data *xndxdata = elf_getdata (xndxscn, NULL);
1210 [ + - ][ - + ]: 864 : if (data == NULL || (xndxscn != NULL && xndxdata == NULL))
1211 : 0 : INTERNAL_ERROR (fullname);
1212 : :
1213 : : /* Iterate over all symbols. */
1214 : : #ifdef USE_DEMANGLE
1215 : 864 : size_t demangle_buffer_len = 0;
1216 : 864 : char *demangle_buffer = NULL;
1217 : : #endif
1218 : 864 : int longest_name = 4;
1219 : 864 : int longest_where = 4;
1220 : 864 : size_t nentries_used = 0;
1221 [ + + ]: 448452 : for (size_t cnt = 0; cnt < nentries; ++cnt)
1222 : : {
1223 : 447588 : GElf_Sym *sym = gelf_getsymshndx (data, xndxdata, cnt,
1224 : : &sym_mem[nentries_used].sym,
1225 : 447588 : &sym_mem[nentries_used].xndx);
1226 [ - + ]: 447588 : if (sym == NULL)
1227 : 0 : INTERNAL_ERROR (fullname);
1228 : :
1229 : : /* Filter out administrative symbols without a name and those
1230 : : deselected by the user with command line options. */
1231 [ + + ][ + + ]: 447588 : if ((hide_undefined && sym->st_shndx == SHN_UNDEF)
1232 [ - + ][ # # ]: 435870 : || (hide_defined && sym->st_shndx != SHN_UNDEF)
1233 [ + + ][ + + ]: 435870 : || (hide_local && GELF_ST_BIND (sym->st_info) == STB_LOCAL))
1234 : 126450 : continue;
1235 : :
1236 : 321138 : sym_mem[nentries_used].where = "";
1237 [ + + ]: 321138 : if (format == format_sysv)
1238 : : {
1239 : 107046 : const char *symstr = elf_strptr (ebl->elf, shdr->sh_link,
1240 : 107046 : sym->st_name);
1241 [ - + ]: 107046 : if (symstr == NULL)
1242 : 0 : continue;
1243 : :
1244 : : #ifdef USE_DEMANGLE
1245 : : /* Demangle if necessary. */
1246 [ - + ]: 107046 : if (demangle)
1247 : : {
1248 : 0 : int status = -1;
1249 : 0 : char *dmsymstr = __cxa_demangle (symstr, demangle_buffer,
1250 : : &demangle_buffer_len, &status);
1251 : :
1252 [ # # ]: 0 : if (status == 0)
1253 : 0 : symstr = dmsymstr;
1254 : : }
1255 : : #endif
1256 : :
1257 [ + + ]: 107046 : longest_name = MAX ((size_t) longest_name, strlen (symstr));
1258 : :
1259 [ + + ]: 107046 : if (sym->st_shndx != SHN_UNDEF
1260 [ + + ]: 95409 : && GELF_ST_BIND (sym->st_info) != STB_LOCAL
1261 [ - + ]: 19014 : && global_root != NULL)
1262 : : {
1263 : 0 : Dwarf_Global fake = { .name = symstr };
1264 : 0 : Dwarf_Global **found = tfind (&fake, &global_root,
1265 : : global_compare);
1266 [ # # ]: 0 : if (found != NULL)
1267 : : {
1268 : : Dwarf_Die die_mem;
1269 : 0 : Dwarf_Die *die = dwarf_offdie (dbg, (*found)->die_offset,
1270 : : &die_mem);
1271 : :
1272 : : Dwarf_Die cudie_mem;
1273 : 0 : Dwarf_Die *cudie = NULL;
1274 : :
1275 : : Dwarf_Addr lowpc;
1276 : : Dwarf_Addr highpc;
1277 [ # # ]: 0 : if (die != NULL
1278 [ # # ]: 0 : && dwarf_lowpc (die, &lowpc) == 0
1279 [ # # ]: 0 : && lowpc <= sym->st_value
1280 [ # # ]: 0 : && dwarf_highpc (die, &highpc) == 0
1281 [ # # ]: 0 : && highpc > sym->st_value)
1282 : 0 : cudie = dwarf_offdie (dbg, (*found)->cu_offset,
1283 : : &cudie_mem);
1284 [ # # ]: 0 : if (cudie != NULL)
1285 : : {
1286 : 0 : Dwarf_Line *line = dwarf_getsrc_die (cudie,
1287 : : sym->st_value);
1288 [ # # ]: 0 : if (line != NULL)
1289 : : {
1290 : : /* We found the line. */
1291 : : int lineno;
1292 : 0 : (void) dwarf_lineno (line, &lineno);
1293 : : int n;
1294 : 0 : n = obstack_printf (&whereob, "%s:%d%c",
1295 : : basename (dwarf_linesrc (line,
1296 : : NULL,
1297 : : NULL)),
1298 : : lineno, '\0');
1299 : : sym_mem[nentries_used].where
1300 [ # # ][ # # ]: 0 : = obstack_finish (&whereob);
1301 : :
1302 : : /* The return value of obstack_print included the
1303 : : NUL byte, so subtract one. */
1304 [ # # ]: 0 : if (--n > (int) longest_where)
1305 : 0 : longest_where = (size_t) n;
1306 : : }
1307 : : }
1308 : : }
1309 : : }
1310 : :
1311 : : /* Try to find the symbol among the local symbols. */
1312 [ + - ]: 107046 : if (sym_mem[nentries_used].where[0] == '\0')
1313 : : {
1314 : 214092 : struct local_name fake =
1315 : : {
1316 : : .name = symstr,
1317 : 107046 : .lowpc = sym->st_value,
1318 : : .highpc = sym->st_value,
1319 : : };
1320 : 107046 : struct local_name **found = tfind (&fake, &local_root,
1321 : : local_compare);
1322 [ + + ]: 107046 : if (found != NULL)
1323 : : {
1324 : : /* We found the line. */
1325 : 27657 : int n = obstack_printf (&whereob, "%s:%" PRIu64 "%c",
1326 : : basename ((*found)->file),
1327 : 27657 : (*found)->lineno,
1328 : : '\0');
1329 [ - + ][ - + ]: 27657 : sym_mem[nentries_used].where = obstack_finish (&whereob);
1330 : :
1331 : : /* The return value of obstack_print included the
1332 : : NUL byte, so subtract one. */
1333 [ + + ]: 27657 : if (--n > (int) longest_where)
1334 : 107046 : longest_where = (size_t) n;
1335 : : }
1336 : : }
1337 : : }
1338 : :
1339 : : /* We use this entry. */
1340 : 321138 : ++nentries_used;
1341 : : }
1342 : : #ifdef USE_DEMANGLE
1343 : : free (demangle_buffer);
1344 : : #endif
1345 : : /* Now we know the exact number. */
1346 : 864 : nentries = nentries_used;
1347 : :
1348 : : /* Sort the entries according to the users wishes. */
1349 [ + + ]: 864 : if (sort == sort_name)
1350 : : {
1351 : 288 : sort_by_name_strtab = elf_getdata (elf_getscn (ebl->elf, shdr->sh_link),
1352 : : NULL);
1353 : 288 : qsort (sym_mem, nentries, sizeof (GElf_SymX), sort_by_name);
1354 : : }
1355 [ + + ]: 576 : else if (sort == sort_numeric)
1356 : 288 : qsort (sym_mem, nentries, sizeof (GElf_SymX), sort_by_address);
1357 : :
1358 : : /* Finally print according to the users selection. */
1359 [ + + + ]: 864 : switch (format)
1360 : : {
1361 : : case format_sysv:
1362 : 288 : show_symbols_sysv (ebl, shdr->sh_link, fullname, sym_mem, nentries,
1363 : : longest_name, longest_where);
1364 : 288 : break;
1365 : :
1366 : : case format_bsd:
1367 : 288 : show_symbols_bsd (ebl->elf, ehdr, shdr->sh_link, prefix, fname, fullname,
1368 : : sym_mem, nentries);
1369 : 288 : break;
1370 : :
1371 : : case format_posix:
1372 : : default:
1373 [ - + ]: 288 : assert (format == format_posix);
1374 : 288 : show_symbols_posix (ebl->elf, ehdr, shdr->sh_link, prefix, fullname,
1375 : : sym_mem, nentries);
1376 : 288 : break;
1377 : : }
1378 : :
1379 : : /* Free all memory. */
1380 [ - + ]: 864 : if (nentries * sizeof (GElf_Sym) >= MAX_STACK_ALLOC)
1381 : 0 : free (sym_mem);
1382 : :
1383 : 864 : obstack_free (&whereob, NULL);
1384 : :
1385 [ + + ]: 864 : if (dbg != NULL)
1386 : : {
1387 : 288 : tdestroy (global_root, free);
1388 : 288 : global_root = NULL;
1389 : :
1390 : 288 : tdestroy (local_root, free);
1391 : 288 : local_root = NULL;
1392 : :
1393 : 288 : (void) dwarf_end (dbg);
1394 : : }
1395 : 864 : }
1396 : :
1397 : :
1398 : : static int
1399 : 864 : handle_elf (Elf *elf, const char *prefix, const char *fname,
1400 : : const char *suffix)
1401 : : {
1402 [ - + ]: 864 : size_t prefix_len = prefix == NULL ? 0 : strlen (prefix);
1403 [ - + ]: 864 : size_t suffix_len = suffix == NULL ? 0 : strlen (suffix);
1404 : 864 : size_t fname_len = strlen (fname) + 1;
1405 : 864 : char fullname[prefix_len + 1 + fname_len + suffix_len];
1406 : 864 : char *cp = fullname;
1407 : 864 : Elf_Scn *scn = NULL;
1408 : 864 : int any = 0;
1409 : 864 : int result = 0;
1410 : : GElf_Ehdr ehdr_mem;
1411 : : GElf_Ehdr *ehdr;
1412 : : Ebl *ebl;
1413 : :
1414 : : /* Get the backend for this object file type. */
1415 : 864 : ebl = ebl_openbackend (elf);
1416 : :
1417 : : /* We need the ELF header in a few places. */
1418 : 864 : ehdr = gelf_getehdr (elf, &ehdr_mem);
1419 [ - + ]: 864 : if (ehdr == NULL)
1420 : 0 : INTERNAL_ERROR (fullname);
1421 : :
1422 : : /* If we are asked to print the dynamic symbol table and this is
1423 : : executable or dynamic executable, fail. */
1424 [ + + ]: 864 : if (symsec_type == SHT_DYNSYM
1425 [ + + ][ - + ]: 216 : && ehdr->e_type != ET_EXEC && ehdr->e_type != ET_DYN)
1426 : : {
1427 : : /* XXX Add machine specific object file types. */
1428 [ # # ][ # # ]: 0 : error (0, 0, gettext ("%s%s%s%s: Invalid operation"),
[ # # ]
1429 : : prefix ?: "", prefix ? "(" : "", fname, prefix ? ")" : "");
1430 : 0 : result = 1;
1431 : 0 : goto out;
1432 : : }
1433 : :
1434 : : /* Create the full name of the file. */
1435 [ - + ]: 864 : if (prefix != NULL)
1436 : 0 : cp = mempcpy (cp, prefix, prefix_len);
1437 : 864 : cp = mempcpy (cp, fname, fname_len);
1438 [ - + ]: 864 : if (suffix != NULL)
1439 : 864 : memcpy (cp - 1, suffix, suffix_len + 1);
1440 : :
1441 : : /* Find the symbol table.
1442 : :
1443 : : XXX Can there be more than one? Do we print all? Currently we do. */
1444 [ + + ]: 32436 : while ((scn = elf_nextscn (elf, scn)) != NULL)
1445 : : {
1446 : : GElf_Shdr shdr_mem;
1447 : 31572 : GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
1448 : :
1449 [ - + ]: 31572 : if (shdr == NULL)
1450 : 0 : INTERNAL_ERROR (fullname);
1451 : :
1452 [ + + ]: 31572 : if (shdr->sh_type == symsec_type)
1453 : : {
1454 : 864 : Elf_Scn *xndxscn = NULL;
1455 : :
1456 : : /* We have a symbol table. First make sure we remember this. */
1457 : 864 : any = 1;
1458 : :
1459 : : /* Look for an extended section index table for this section. */
1460 [ + + ]: 864 : if (symsec_type == SHT_SYMTAB)
1461 : : {
1462 : 648 : size_t scnndx = elf_ndxscn (scn);
1463 : :
1464 [ + + ]: 24327 : while ((xndxscn = elf_nextscn (elf, xndxscn)) != NULL)
1465 : : {
1466 : : GElf_Shdr xndxshdr_mem;
1467 : 23679 : GElf_Shdr *xndxshdr = gelf_getshdr (xndxscn, &xndxshdr_mem);
1468 : :
1469 [ - + ]: 23679 : if (xndxshdr == NULL)
1470 : 0 : INTERNAL_ERROR (fullname);
1471 : :
1472 [ - + ]: 23679 : if (xndxshdr->sh_type == SHT_SYMTAB_SHNDX
1473 [ # # ]: 23679 : && xndxshdr->sh_link == scnndx)
1474 : : break;
1475 : : }
1476 : : }
1477 : :
1478 : 31572 : show_symbols (ebl, ehdr, scn, xndxscn, shdr, prefix, fname,
1479 : : fullname);
1480 : : }
1481 : : }
1482 : :
1483 [ - + ]: 864 : if (! any)
1484 : : {
1485 [ # # ][ # # ]: 0 : error (0, 0, gettext ("%s%s%s: no symbols"),
1486 : : prefix ?: "", prefix ? ":" : "", fname);
1487 : 0 : result = 1;
1488 : : }
1489 : :
1490 : : out:
1491 : : /* Close the ELF backend library descriptor. */
1492 : 864 : ebl_closebackend (ebl);
1493 : :
1494 : 864 : return result;
1495 : : }
1496 : :
1497 : :
1498 : 389289 : #include "debugpred.h"
|