1 /*-
2 * Copyright (c) 2007 Hyogeol Lee <[email protected]>
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer
10 * in this position and unchanged.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
16 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
17 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
18 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
19 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
20 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
24 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 */
26
27 #include <sys/queue.h>
28 #include <sys/types.h>
29 #include <sys/stat.h>
30 #include <ar.h>
31 #include <assert.h>
32 #include <ctype.h>
33 #include <dwarf.h>
34 #include <err.h>
35 #include <errno.h>
36 #include <fcntl.h>
37 #include <gelf.h>
38 #include <getopt.h>
39 #include <inttypes.h>
40 #include <libdwarf.h>
41 #include <libelftc.h>
42 #include <stdbool.h>
43 #include <stdio.h>
44 #include <stdlib.h>
45 #include <string.h>
46 #include <strings.h>
47 #include <unistd.h>
48
49 #include "_elftc.h"
50
51 ELFTC_VCSID("$Id: nm.c 3504 2016-12-17 15:33:16Z kaiwang27 $");
52
53 /* symbol information list */
54 STAILQ_HEAD(sym_head, sym_entry);
55
56 struct sym_entry {
57 char *name;
58 GElf_Sym *sym;
59 STAILQ_ENTRY(sym_entry) sym_entries;
60 };
61
62 typedef int (*fn_sort)(const void *, const void *);
63 typedef void (*fn_elem_print)(char, const char *, const GElf_Sym *, const char *);
64 typedef void (*fn_sym_print)(const GElf_Sym *);
65 typedef int (*fn_filter)(char, const GElf_Sym *, const char *);
66
67 /* output filter list */
68 static SLIST_HEAD(filter_head, filter_entry) nm_out_filter =
69 SLIST_HEAD_INITIALIZER(nm_out_filter);
70
71 struct filter_entry {
72 fn_filter fn;
73 SLIST_ENTRY(filter_entry) filter_entries;
74 };
75
76 struct sym_print_data {
77 struct sym_head *headp;
78 size_t sh_num, list_num;
79 const char *t_table, **s_table, *filename, *objname;
80 };
81
82 struct nm_prog_info {
83 const char *name;
84 const char *def_filename;
85 };
86
87 /* List for line number information. */
88 struct line_info_entry {
89 uint64_t addr; /* address */
90 uint64_t line; /* line number */
91 char *file; /* file name with path */
92 SLIST_ENTRY(line_info_entry) entries;
93 };
94 SLIST_HEAD(line_info_head, line_info_entry);
95
96 /* List for function line number information. */
97 struct func_info_entry {
98 char *name; /* function name */
99 char *file; /* file name with path */
100 uint64_t lowpc; /* low address */
101 uint64_t highpc; /* high address */
102 uint64_t line; /* line number */
103 SLIST_ENTRY(func_info_entry) entries;
104 };
105 SLIST_HEAD(func_info_head, func_info_entry);
106
107 /* List for variable line number information. */
108 struct var_info_entry {
109 char *name; /* variable name */
110 char *file; /* file name with path */
111 uint64_t addr; /* address */
112 uint64_t line; /* line number */
113 SLIST_ENTRY(var_info_entry) entries;
114 };
115 SLIST_HEAD(var_info_head, var_info_entry);
116
117 /* output numric type */
118 enum radix {
119 RADIX_OCT,
120 RADIX_HEX,
121 RADIX_DEC
122 };
123
124 /* output symbol type, PRINT_SYM_DYN for dynamic symbol only */
125 enum print_symbol {
126 PRINT_SYM_SYM,
127 PRINT_SYM_DYN
128 };
129
130 /* output name type */
131 enum print_name {
132 PRINT_NAME_NONE,
133 PRINT_NAME_FULL,
134 PRINT_NAME_MULTI
135 };
136
137 struct nm_prog_options {
138 enum print_symbol print_symbol;
139 enum print_name print_name;
140 enum radix t;
141 int demangle_type;
142 bool print_debug;
143 bool print_armap;
144 int print_size;
145 bool debug_line;
146 int def_only;
147 bool undef_only;
148 int sort_size;
149 bool sort_reverse;
150 int no_demangle;
151
152 /*
153 * function pointer to sort symbol list.
154 * possible function - cmp_name, cmp_none, cmp_size, cmp_value
155 */
156 fn_sort sort_fn;
157
158 /*
159 * function pointer to print symbol elem.
160 * possible function - sym_elem_print_all
161 * sym_elem_print_all_portable
162 * sym_elem_print_all_sysv
163 */
164 fn_elem_print elem_print_fn;
165
166 fn_sym_print value_print_fn;
167 fn_sym_print size_print_fn;
168 };
169
170 #define CHECK_SYM_PRINT_DATA(p) (p->headp == NULL || p->sh_num == 0 || \
171 p->t_table == NULL || p->s_table == NULL || p->filename == NULL)
172 #define IS_SYM_TYPE(t) ((t) == '?' || isalpha((t)) != 0)
173 #define IS_UNDEF_SYM_TYPE(t) ((t) == 'U' || (t) == 'v' || (t) == 'w')
174 #define UNUSED(p) ((void)p)
175
176 static int cmp_name(const void *, const void *);
177 static int cmp_none(const void *, const void *);
178 static int cmp_size(const void *, const void *);
179 static int cmp_value(const void *, const void *);
180 static void filter_dest(void);
181 static int filter_insert(fn_filter);
182 static void get_opt(int, char **);
183 static int get_sym(Elf *, struct sym_head *, int, size_t, size_t,
184 const char *, const char **, int);
185 static const char * get_sym_name(Elf *, const GElf_Sym *, size_t,
186 const char **, int);
187 static char get_sym_type(const GElf_Sym *, const char *);
188 static void global_dest(void);
189 static void global_init(void);
190 static bool is_sec_data(GElf_Shdr *);
191 static bool is_sec_debug(const char *);
192 static bool is_sec_nobits(GElf_Shdr *);
193 static bool is_sec_readonly(GElf_Shdr *);
194 static bool is_sec_text(GElf_Shdr *);
195 static void print_ar_index(int, Elf *);
196 static void print_header(const char *, const char *);
197 static void print_version(void);
198 static int read_elf(Elf *, const char *, Elf_Kind);
199 static int read_object(const char *);
200 static int read_files(int, char **);
201 static void set_opt_value_print_fn(enum radix);
202 static int sym_elem_def(char, const GElf_Sym *, const char *);
203 static int sym_elem_global(char, const GElf_Sym *, const char *);
204 static int sym_elem_global_static(char, const GElf_Sym *,
205 const char *);
206 static int sym_elem_nondebug(char, const GElf_Sym *, const char *);
207 static int sym_elem_nonzero_size(char, const GElf_Sym *,
208 const char *);
209 static void sym_elem_print_all(char, const char *,
210 const GElf_Sym *, const char *);
211 static void sym_elem_print_all_portable(char, const char *,
212 const GElf_Sym *, const char *);
213 static void sym_elem_print_all_sysv(char, const char *,
214 const GElf_Sym *, const char *);
215 static int sym_elem_undef(char, const GElf_Sym *, const char *);
216 static void sym_list_dest(struct sym_head *);
217 static int sym_list_insert(struct sym_head *, const char *,
218 const GElf_Sym *);
219 static void sym_list_print(struct sym_print_data *,
220 struct func_info_head *, struct var_info_head *,
221 struct line_info_head *);
222 static void sym_list_print_each(struct sym_entry *,
223 struct sym_print_data *, struct func_info_head *,
224 struct var_info_head *, struct line_info_head *);
225 static struct sym_entry *sym_list_sort(struct sym_print_data *);
226 static void sym_size_oct_print(const GElf_Sym *);
227 static void sym_size_hex_print(const GElf_Sym *);
228 static void sym_size_dec_print(const GElf_Sym *);
229 static void sym_value_oct_print(const GElf_Sym *);
230 static void sym_value_hex_print(const GElf_Sym *);
231 static void sym_value_dec_print(const GElf_Sym *);
232 static void usage(int);
233
234 static struct nm_prog_info nm_info;
235 static struct nm_prog_options nm_opts;
236 static int nm_elfclass;
237
238 /*
239 * Point to current sym_print_data to use portable qsort function.
240 * (e.g. There is no qsort_r function in NetBSD.)
241 *
242 * Using in sym_list_sort.
243 */
244 static struct sym_print_data *nm_print_data;
245
246 static const struct option nm_longopts[] = {
247 { "debug-syms", no_argument, NULL, 'a' },
248 { "defined-only", no_argument, &nm_opts.def_only, 1},
249 { "demangle", optional_argument, NULL, 'C' },
250 { "dynamic", no_argument, NULL, 'D' },
251 { "extern-only", no_argument, NULL, 'g' },
252 { "format", required_argument, NULL, 'F' },
253 { "help", no_argument, NULL, 'h' },
254 { "line-numbers", no_argument, NULL, 'l' },
255 { "no-demangle", no_argument, &nm_opts.no_demangle,
256 1},
257 { "no-sort", no_argument, NULL, 'p' },
258 { "numeric-sort", no_argument, NULL, 'v' },
259 { "print-armap", no_argument, NULL, 's' },
260 { "print-file-name", no_argument, NULL, 'A' },
261 { "print-size", no_argument, NULL, 'S' },
262 { "radix", required_argument, NULL, 't' },
263 { "reverse-sort", no_argument, NULL, 'r' },
264 { "size-sort", no_argument, &nm_opts.sort_size, 1},
265 { "undefined-only", no_argument, NULL, 'u' },
266 { "version", no_argument, NULL, 'V' },
267 { NULL, 0, NULL, 0 }
268 };
269
270 #if defined(ELFTC_NEED_BYTEORDER_EXTENSIONS)
271 static __inline uint32_t
be32dec(const void * pp)272 be32dec(const void *pp)
273 {
274 unsigned char const *p = (unsigned char const *)pp;
275
276 return ((p[0] << 24) | (p[1] << 16) | (p[2] << 8) | p[3]);
277 }
278
279 static __inline uint32_t
le32dec(const void * pp)280 le32dec(const void *pp)
281 {
282 unsigned char const *p = (unsigned char const *)pp;
283
284 return ((p[3] << 24) | (p[2] << 16) | (p[1] << 8) | p[0]);
285 }
286
287 static __inline uint64_t
be64dec(const void * pp)288 be64dec(const void *pp)
289 {
290 unsigned char const *p = (unsigned char const *)pp;
291
292 return (((uint64_t)be32dec(p) << 32) | be32dec(p + 4));
293 }
294
295 static __inline uint64_t
le64dec(const void * pp)296 le64dec(const void *pp)
297 {
298 unsigned char const *p = (unsigned char const *)pp;
299
300 return (((uint64_t)le32dec(p + 4) << 32) | le32dec(p));
301 }
302 #endif
303
304 static int
cmp_name(const void * l,const void * r)305 cmp_name(const void *l, const void *r)
306 {
307
308 assert(l != NULL);
309 assert(r != NULL);
310 assert(((const struct sym_entry *)l)->name != NULL);
311 assert(((const struct sym_entry *)r)->name != NULL);
312
313 return (strcmp(((const struct sym_entry *)l)->name,
314 ((const struct sym_entry *)r)->name));
315 }
316
317 static int
cmp_none(const void * l,const void * r)318 cmp_none(const void *l, const void *r)
319 {
320
321 UNUSED(l);
322 UNUSED(r);
323
324 return (0);
325 }
326
327 /* Size comparison. If l and r have same size, compare their name. */
328 static int
cmp_size(const void * lp,const void * rp)329 cmp_size(const void *lp, const void *rp)
330 {
331 const struct sym_entry *l, *r;
332
333 l = lp;
334 r = rp;
335
336 assert(l != NULL);
337 assert(l->name != NULL);
338 assert(l->sym != NULL);
339 assert(r != NULL);
340 assert(r->name != NULL);
341 assert(r->sym != NULL);
342
343 if (l->sym->st_size == r->sym->st_size)
344 return (strcmp(l->name, r->name));
345
346 return (l->sym->st_size - r->sym->st_size);
347 }
348
349 /* Value comparison. Undefined symbols come first. */
350 static int
cmp_value(const void * lp,const void * rp)351 cmp_value(const void *lp, const void *rp)
352 {
353 const struct sym_entry *l, *r;
354 const char *ttable;
355 int l_is_undef, r_is_undef;
356
357 l = lp;
358 r = rp;
359
360 assert(nm_print_data != NULL);
361 ttable = nm_print_data->t_table;
362
363 assert(l != NULL);
364 assert(l->name != NULL);
365 assert(l->sym != NULL);
366 assert(r != NULL);
367 assert(r->name != NULL);
368 assert(r->sym != NULL);
369 assert(ttable != NULL);
370
371 l_is_undef = IS_UNDEF_SYM_TYPE(get_sym_type(l->sym, ttable)) ? 1 : 0;
372 r_is_undef = IS_UNDEF_SYM_TYPE(get_sym_type(r->sym, ttable)) ? 1 : 0;
373
374 assert(l_is_undef + r_is_undef >= 0);
375 assert(l_is_undef + r_is_undef <= 2);
376
377 switch (l_is_undef + r_is_undef) {
378 case 0:
379 /* Both defined */
380 if (l->sym->st_value == r->sym->st_value)
381 return (strcmp(l->name, r->name));
382 return (l->sym->st_value > r->sym->st_value ? 1 : -1);
383 case 1:
384 /* One undefined */
385 return (l_is_undef == 0 ? 1 : -1);
386 case 2:
387 /* Both undefined */
388 return (strcmp(l->name, r->name));
389 }
390 /* NOTREACHED */
391
392 return (l->sym->st_value - r->sym->st_value);
393 }
394
395 static void
filter_dest(void)396 filter_dest(void)
397 {
398 struct filter_entry *e;
399
400 while (!SLIST_EMPTY(&nm_out_filter)) {
401 e = SLIST_FIRST(&nm_out_filter);
402 SLIST_REMOVE_HEAD(&nm_out_filter, filter_entries);
403 free(e);
404 }
405 }
406
407 static int
filter_insert(fn_filter filter_fn)408 filter_insert(fn_filter filter_fn)
409 {
410 struct filter_entry *e;
411
412 assert(filter_fn != NULL);
413
414 if ((e = malloc(sizeof(struct filter_entry))) == NULL) {
415 warn("malloc");
416 return (0);
417 }
418 e->fn = filter_fn;
419 SLIST_INSERT_HEAD(&nm_out_filter, e, filter_entries);
420
421 return (1);
422 }
423
424 static int
parse_demangle_option(const char * opt)425 parse_demangle_option(const char *opt)
426 {
427
428 if (opt == NULL)
429 return (ELFTC_DEM_UNKNOWN);
430 else if (!strncasecmp(opt, "gnu-v2", 6))
431 return (ELFTC_DEM_GNU2);
432 else if (!strncasecmp(opt, "gnu-v3", 6))
433 return (ELFTC_DEM_GNU3);
434 else if (!strncasecmp(opt, "arm", 3))
435 return (ELFTC_DEM_ARM);
436 else
437 errx(EXIT_FAILURE, "unknown demangling style '%s'", opt);
438
439 /* NOTREACHED */
440 return (0);
441 }
442
443 static void
get_opt(int argc,char ** argv)444 get_opt(int argc, char **argv)
445 {
446 int ch;
447 bool is_posix, oflag;
448
449 if (argc <= 0 || argv == NULL)
450 return;
451
452 oflag = is_posix = false;
453 nm_opts.t = RADIX_HEX;
454 while ((ch = getopt_long(argc, argv, "ABCDF:PSVaefghlnoprst:uvx",
455 nm_longopts, NULL)) != -1) {
456 switch (ch) {
457 case 'A':
458 nm_opts.print_name = PRINT_NAME_FULL;
459 break;
460 case 'B':
461 nm_opts.elem_print_fn = &sym_elem_print_all;
462 break;
463 case 'C':
464 nm_opts.demangle_type = parse_demangle_option(optarg);
465 break;
466 case 'D':
467 nm_opts.print_symbol = PRINT_SYM_DYN;
468 break;
469 case 'F':
470 /* sysv, bsd, posix */
471 switch (optarg[0]) {
472 case 'B':
473 case 'b':
474 nm_opts.elem_print_fn = &sym_elem_print_all;
475 break;
476 case 'P':
477 case 'p':
478 is_posix = true;
479 nm_opts.elem_print_fn =
480 &sym_elem_print_all_portable;
481 break;
482 case 'S':
483 case 's':
484 nm_opts.elem_print_fn =
485 &sym_elem_print_all_sysv;
486 break;
487 default:
488 warnx("%s: Invalid format", optarg);
489 usage(1);
490 }
491
492 break;
493 case 'P':
494 is_posix = true;
495 nm_opts.elem_print_fn = &sym_elem_print_all_portable;
496 break;
497 case 'S':
498 nm_opts.print_size = 1;
499 break;
500 case 'V':
501 print_version();
502 /* NOTREACHED */
503 case 'a':
504 nm_opts.print_debug = true;
505 break;
506 case 'e':
507 filter_insert(sym_elem_global_static);
508 break;
509 case 'f':
510 break;
511 case 'g':
512 filter_insert(sym_elem_global);
513 break;
514 case 'h':
515 usage(0);
516 break;
517 case 'l':
518 nm_opts.debug_line = true;
519 break;
520 case 'n':
521 case 'v':
522 nm_opts.sort_fn = &cmp_value;
523 break;
524 case 'o':
525 oflag = true;
526 break;
527 case 'p':
528 nm_opts.sort_fn = &cmp_none;
529 break;
530 case 'r':
531 nm_opts.sort_reverse = true;
532 break;
533 case 's':
534 nm_opts.print_armap = true;
535 break;
536 case 't':
537 /* t require always argument to getopt_long */
538 switch (optarg[0]) {
539 case 'd':
540 nm_opts.t = RADIX_DEC;
541 break;
542 case 'o':
543 nm_opts.t = RADIX_OCT;
544 break;
545 case 'x':
546 nm_opts.t = RADIX_HEX;
547 break;
548 default:
549 warnx("%s: Invalid radix", optarg);
550 usage(1);
551 }
552 break;
553 case 'u':
554 filter_insert(sym_elem_undef);
555 nm_opts.undef_only = true;
556 break;
557 /* case 'v': see case 'n' above. */
558 case 'x':
559 nm_opts.t = RADIX_HEX;
560 break;
561 case 0:
562 if (nm_opts.sort_size != 0) {
563 nm_opts.sort_fn = &cmp_size;
564 filter_insert(sym_elem_def);
565 filter_insert(sym_elem_nonzero_size);
566 }
567 if (nm_opts.def_only != 0)
568 filter_insert(sym_elem_def);
569 if (nm_opts.no_demangle != 0)
570 nm_opts.demangle_type = -1;
571 break;
572 default :
573 usage(1);
574 }
575 }
576
577 /*
578 * In POSIX mode, the '-o' option controls the output radix.
579 * In non-POSIX mode, the option is a synonym for the '-A' and
580 * '--print-file-name' options.
581 */
582 if (oflag) {
583 if (is_posix)
584 nm_opts.t = RADIX_OCT;
585 else
586 nm_opts.print_name = PRINT_NAME_FULL;
587 }
588
589 assert(nm_opts.sort_fn != NULL && "nm_opts.sort_fn is null");
590 assert(nm_opts.elem_print_fn != NULL &&
591 "nm_opts.elem_print_fn is null");
592 assert(nm_opts.value_print_fn != NULL &&
593 "nm_opts.value_print_fn is null");
594
595 set_opt_value_print_fn(nm_opts.t);
596
597 if (nm_opts.undef_only == true) {
598 if (nm_opts.sort_fn == &cmp_size)
599 errx(EXIT_FAILURE,
600 "--size-sort with -u is meaningless");
601 if (nm_opts.def_only != 0)
602 errx(EXIT_FAILURE,
603 "-u with --defined-only is meaningless");
604 }
605 if (nm_opts.print_debug == false)
606 filter_insert(sym_elem_nondebug);
607 if (nm_opts.sort_reverse == true && nm_opts.sort_fn == cmp_none)
608 nm_opts.sort_reverse = false;
609 }
610
611 /*
612 * Get symbol information from elf.
613 */
614 static int
get_sym(Elf * elf,struct sym_head * headp,int shnum,size_t dynndx,size_t strndx,const char * type_table,const char ** sec_table,int sec_table_size)615 get_sym(Elf *elf, struct sym_head *headp, int shnum, size_t dynndx,
616 size_t strndx, const char *type_table, const char **sec_table,
617 int sec_table_size)
618 {
619 Elf_Scn *scn;
620 Elf_Data *data;
621 GElf_Shdr shdr;
622 GElf_Sym sym;
623 struct filter_entry *fep;
624 size_t ndx;
625 int rtn;
626 const char *sym_name;
627 char type;
628 bool filter;
629 int i, j;
630
631 assert(elf != NULL);
632 assert(headp != NULL);
633
634 rtn = 0;
635 for (i = 1; i < shnum; i++) {
636 if ((scn = elf_getscn(elf, i)) == NULL) {
637 warnx("elf_getscn failed: %s", elf_errmsg(-1));
638 continue;
639 }
640 if (gelf_getshdr(scn, &shdr) != &shdr) {
641 warnx("gelf_getshdr failed: %s", elf_errmsg(-1));
642 continue;
643 }
644 if (shdr.sh_type == SHT_SYMTAB) {
645 if (nm_opts.print_symbol != PRINT_SYM_SYM)
646 continue;
647 } else if (shdr.sh_type == SHT_DYNSYM) {
648 if (nm_opts.print_symbol != PRINT_SYM_DYN)
649 continue;
650 } else
651 continue;
652
653 ndx = shdr.sh_type == SHT_DYNSYM ? dynndx : strndx;
654
655 data = NULL;
656 while ((data = elf_getdata(scn, data)) != NULL) {
657 j = 1;
658 while (gelf_getsym(data, j++, &sym) != NULL) {
659 sym_name = get_sym_name(elf, &sym, ndx,
660 sec_table, sec_table_size);
661 filter = false;
662 type = get_sym_type(&sym, type_table);
663 SLIST_FOREACH(fep, &nm_out_filter,
664 filter_entries) {
665 if (!fep->fn(type, &sym, sym_name)) {
666 filter = true;
667 break;
668 }
669 }
670 if (filter == false) {
671 if (sym_list_insert(headp, sym_name,
672 &sym) == 0)
673 return (0);
674 rtn++;
675 }
676 }
677 }
678 }
679
680 return (rtn);
681 }
682
683 static const char *
get_sym_name(Elf * elf,const GElf_Sym * sym,size_t ndx,const char ** sec_table,int sec_table_size)684 get_sym_name(Elf *elf, const GElf_Sym *sym, size_t ndx, const char **sec_table,
685 int sec_table_size)
686 {
687 const char *sym_name;
688
689 sym_name = NULL;
690
691 /* Show section name as symbol name for STT_SECTION symbols. */
692 if (GELF_ST_TYPE(sym->st_info) == STT_SECTION) {
693 if (sec_table != NULL && sym->st_shndx < sec_table_size)
694 sym_name = sec_table[sym->st_shndx];
695 } else
696 sym_name = elf_strptr(elf, ndx, sym->st_name);
697
698 if (sym_name == NULL)
699 sym_name = "(null)";
700
701 return (sym_name);
702 }
703
704 static char
get_sym_type(const GElf_Sym * sym,const char * type_table)705 get_sym_type(const GElf_Sym *sym, const char *type_table)
706 {
707 bool is_local;
708
709 if (sym == NULL || type_table == NULL)
710 return ('?');
711
712 is_local = sym->st_info >> 4 == STB_LOCAL;
713
714 if (sym->st_shndx == SHN_ABS) /* absolute */
715 return (is_local ? 'a' : 'A');
716
717 if (sym->st_shndx == SHN_COMMON) /* common */
718 return ('C');
719
720 if ((sym->st_info) >> 4 == STB_WEAK) { /* weak */
721 if ((sym->st_info & 0xf) == STT_OBJECT)
722 return (sym->st_shndx == SHN_UNDEF ? 'v' : 'V');
723
724 return (sym->st_shndx == SHN_UNDEF ? 'w' : 'W');
725 }
726
727 if (sym->st_shndx == SHN_UNDEF) /* undefined */
728 return ('U');
729
730 return (is_local == true && type_table[sym->st_shndx] != 'N' ?
731 tolower((unsigned char) type_table[sym->st_shndx]) :
732 type_table[sym->st_shndx]);
733 }
734
735 static void
global_dest(void)736 global_dest(void)
737 {
738
739 filter_dest();
740 }
741
742 static void
global_init(void)743 global_init(void)
744 {
745
746 if (elf_version(EV_CURRENT) == EV_NONE)
747 errx(EXIT_FAILURE, "elf_version error");
748
749 nm_info.name = ELFTC_GETPROGNAME();
750 nm_info.def_filename = "a.out";
751 nm_opts.print_symbol = PRINT_SYM_SYM;
752 nm_opts.print_name = PRINT_NAME_NONE;
753 nm_opts.demangle_type = -1;
754 nm_opts.print_debug = false;
755 nm_opts.print_armap = false;
756 nm_opts.print_size = 0;
757 nm_opts.debug_line = false;
758 nm_opts.def_only = 0;
759 nm_opts.undef_only = false;
760 nm_opts.sort_size = 0;
761 nm_opts.sort_reverse = false;
762 nm_opts.no_demangle = 0;
763 nm_opts.sort_fn = &cmp_name;
764 nm_opts.elem_print_fn = &sym_elem_print_all;
765 nm_opts.value_print_fn = &sym_value_dec_print;
766 nm_opts.size_print_fn = &sym_size_dec_print;
767 SLIST_INIT(&nm_out_filter);
768 }
769
770 static bool
is_sec_data(GElf_Shdr * s)771 is_sec_data(GElf_Shdr *s)
772 {
773
774 assert(s != NULL && "shdr is NULL");
775
776 return (((s->sh_flags & SHF_ALLOC) != 0) && s->sh_type != SHT_NOBITS);
777 }
778
779 static bool
is_sec_debug(const char * shname)780 is_sec_debug(const char *shname)
781 {
782 const char *dbg_sec[] = {
783 ".debug",
784 ".gnu.linkonce.wi.",
785 ".line",
786 ".rel.debug",
787 ".rela.debug",
788 ".stab",
789 NULL
790 };
791 const char **p;
792
793 if (shname == NULL)
794 return (false);
795
796 for (p = dbg_sec; *p; p++) {
797 if (!strncmp(shname, *p, strlen(*p)))
798 return (true);
799 }
800
801 return (false);
802 }
803
804 static bool
is_sec_nobits(GElf_Shdr * s)805 is_sec_nobits(GElf_Shdr *s)
806 {
807
808 assert(s != NULL && "shdr is NULL");
809
810 return (s->sh_type == SHT_NOBITS);
811 }
812
813 static bool
is_sec_readonly(GElf_Shdr * s)814 is_sec_readonly(GElf_Shdr *s)
815 {
816
817 assert(s != NULL && "shdr is NULL");
818
819 return ((s->sh_flags & SHF_WRITE) == 0);
820 }
821
822 static bool
is_sec_text(GElf_Shdr * s)823 is_sec_text(GElf_Shdr *s)
824 {
825
826 assert(s != NULL && "shdr is NULL");
827
828 return ((s->sh_flags & SHF_EXECINSTR) != 0);
829 }
830
831 static void
print_ar_index(int fd,Elf * arf)832 print_ar_index(int fd, Elf *arf)
833 {
834 Elf *elf;
835 Elf_Arhdr *arhdr;
836 Elf_Arsym *arsym;
837 Elf_Cmd cmd;
838 off_t start;
839 size_t arsym_size;
840
841 if (arf == NULL)
842 return;
843
844 if ((arsym = elf_getarsym(arf, &arsym_size)) == NULL)
845 return;
846
847 printf("\nArchive index:\n");
848
849 start = arsym->as_off;
850 cmd = ELF_C_READ;
851 while (arsym_size > 1) {
852 if (elf_rand(arf, arsym->as_off) == arsym->as_off &&
853 (elf = elf_begin(fd, cmd, arf)) != NULL) {
854 if ((arhdr = elf_getarhdr(elf)) != NULL)
855 printf("%s in %s\n", arsym->as_name,
856 arhdr->ar_name != NULL ?
857 arhdr->ar_name : arhdr->ar_rawname);
858 elf_end(elf);
859 }
860 ++arsym;
861 --arsym_size;
862 }
863
864 elf_rand(arf, start);
865 }
866
867 #define DEMANGLED_BUFFER_SIZE (8 * 1024)
868 #define PRINT_DEMANGLED_NAME(FORMAT, NAME) do { \
869 char _demangled[DEMANGLED_BUFFER_SIZE]; \
870 if (nm_opts.demangle_type < 0 || \
871 elftc_demangle((NAME), _demangled, sizeof(_demangled), \
872 nm_opts.demangle_type) < 0) \
873 printf((FORMAT), (NAME)); \
874 else \
875 printf((FORMAT), _demangled); \
876 } while (0)
877
878 static void
print_header(const char * file,const char * obj)879 print_header(const char *file, const char *obj)
880 {
881
882 if (file == NULL)
883 return;
884
885 if (nm_opts.elem_print_fn == &sym_elem_print_all_sysv) {
886 printf("\n\n%s from %s",
887 nm_opts.undef_only == false ? "Symbols" :
888 "Undefined symbols", file);
889 if (obj != NULL)
890 printf("[%s]", obj);
891 printf(":\n\n");
892
893 printf("\
894 Name Value Class Type Size Line Section\n\n");
895 } else {
896 /* archive file without -A option and POSIX */
897 if (nm_opts.print_name != PRINT_NAME_FULL && obj != NULL) {
898 if (nm_opts.elem_print_fn ==
899 sym_elem_print_all_portable)
900 printf("%s[%s]:\n", file, obj);
901 else if (nm_opts.elem_print_fn == sym_elem_print_all)
902 printf("\n%s:\n", obj);
903 /* multiple files(not archive) without -A option */
904 } else if (nm_opts.print_name == PRINT_NAME_MULTI) {
905 if (nm_opts.elem_print_fn == sym_elem_print_all)
906 printf("\n");
907 printf("%s:\n", file);
908 }
909 }
910 }
911
912 static void
print_version(void)913 print_version(void)
914 {
915
916 (void) printf("%s (%s)\n", nm_info.name, elftc_version());
917 exit(0);
918 }
919
920 static uint64_t
get_block_value(Dwarf_Debug dbg,Dwarf_Block * block)921 get_block_value(Dwarf_Debug dbg, Dwarf_Block *block)
922 {
923 Elf *elf;
924 GElf_Ehdr eh;
925 Dwarf_Error de;
926
927 if (dwarf_get_elf(dbg, &elf, &de) != DW_DLV_OK) {
928 warnx("dwarf_get_elf failed: %s", dwarf_errmsg(de));
929 return (0);
930 }
931
932 if (gelf_getehdr(elf, &eh) != &eh) {
933 warnx("gelf_getehdr failed: %s", elf_errmsg(-1));
934 return (0);
935 }
936
937 if (block->bl_len == 5) {
938 if (eh.e_ident[EI_DATA] == ELFDATA2LSB)
939 return (le32dec((uint8_t *) block->bl_data + 1));
940 else
941 return (be32dec((uint8_t *) block->bl_data + 1));
942 } else if (block->bl_len == 9) {
943 if (eh.e_ident[EI_DATA] == ELFDATA2LSB)
944 return (le64dec((uint8_t *) block->bl_data + 1));
945 else
946 return (be64dec((uint8_t *) block->bl_data + 1));
947 }
948
949 return (0);
950 }
951
952 static char *
find_object_name(Dwarf_Debug dbg,Dwarf_Die die)953 find_object_name(Dwarf_Debug dbg, Dwarf_Die die)
954 {
955 Dwarf_Die ret_die;
956 Dwarf_Attribute at;
957 Dwarf_Off off;
958 Dwarf_Error de;
959 const char *str;
960 char *name;
961
962 if (dwarf_attrval_string(die, DW_AT_name, &str, &de) == DW_DLV_OK) {
963 if ((name = strdup(str)) == NULL) {
964 warn("strdup");
965 return (NULL);
966 }
967 return (name);
968 }
969
970 if (dwarf_attr(die, DW_AT_specification, &at, &de) != DW_DLV_OK)
971 return (NULL);
972
973 if (dwarf_global_formref(at, &off, &de) != DW_DLV_OK)
974 return (NULL);
975
976 if (dwarf_offdie(dbg, off, &ret_die, &de) != DW_DLV_OK)
977 return (NULL);
978
979 return (find_object_name(dbg, ret_die));
980 }
981
982 static void
search_line_attr(Dwarf_Debug dbg,struct func_info_head * func_info,struct var_info_head * var_info,Dwarf_Die die,char ** src_files,Dwarf_Signed filecount)983 search_line_attr(Dwarf_Debug dbg, struct func_info_head *func_info,
984 struct var_info_head *var_info, Dwarf_Die die, char **src_files,
985 Dwarf_Signed filecount)
986 {
987 Dwarf_Attribute at;
988 Dwarf_Unsigned udata;
989 Dwarf_Half tag;
990 Dwarf_Block *block;
991 Dwarf_Bool flag;
992 Dwarf_Die ret_die;
993 Dwarf_Error de;
994 struct func_info_entry *func;
995 struct var_info_entry *var;
996 int ret;
997
998 if (dwarf_tag(die, &tag, &de) != DW_DLV_OK) {
999 warnx("dwarf_tag failed: %s", dwarf_errmsg(de));
1000 goto cont_search;
1001 }
1002
1003 /* We're interested in DIEs which define functions or variables. */
1004 if (tag != DW_TAG_subprogram && tag != DW_TAG_entry_point &&
1005 tag != DW_TAG_inlined_subroutine && tag != DW_TAG_variable)
1006 goto cont_search;
1007
1008 if (tag == DW_TAG_variable) {
1009
1010 /* Ignore "artificial" variable. */
1011 if (dwarf_attrval_flag(die, DW_AT_artificial, &flag, &de) ==
1012 DW_DLV_OK && flag)
1013 goto cont_search;
1014
1015 /* Ignore pure declaration. */
1016 if (dwarf_attrval_flag(die, DW_AT_declaration, &flag, &de) ==
1017 DW_DLV_OK && flag)
1018 goto cont_search;
1019
1020 /* Ignore stack varaibles. */
1021 if (dwarf_attrval_flag(die, DW_AT_external, &flag, &de) !=
1022 DW_DLV_OK || !flag)
1023 goto cont_search;
1024
1025 if ((var = calloc(1, sizeof(*var))) == NULL) {
1026 warn("calloc failed");
1027 goto cont_search;
1028 }
1029
1030 if (dwarf_attrval_unsigned(die, DW_AT_decl_file, &udata,
1031 &de) == DW_DLV_OK && udata > 0 &&
1032 (Dwarf_Signed) (udata - 1) < filecount) {
1033 var->file = strdup(src_files[udata - 1]);
1034 if (var->file == NULL) {
1035 warn("strdup");
1036 free(var);
1037 goto cont_search;
1038 }
1039 }
1040
1041 if (dwarf_attrval_unsigned(die, DW_AT_decl_line, &udata, &de) ==
1042 DW_DLV_OK)
1043 var->line = udata;
1044
1045 var->name = find_object_name(dbg, die);
1046 if (var->name == NULL) {
1047 if (var->file)
1048 free(var->file);
1049 free(var);
1050 goto cont_search;
1051 }
1052
1053 if (dwarf_attr(die, DW_AT_location, &at, &de) == DW_DLV_OK &&
1054 dwarf_formblock(at, &block, &de) == DW_DLV_OK) {
1055 /*
1056 * Since we ignored stack variables, the rest are the
1057 * external varaibles which should always use DW_OP_addr
1058 * operator for DW_AT_location value.
1059 */
1060 if (*((uint8_t *)block->bl_data) == DW_OP_addr)
1061 var->addr = get_block_value(dbg, block);
1062 }
1063
1064 SLIST_INSERT_HEAD(var_info, var, entries);
1065
1066 } else {
1067
1068 if ((func = calloc(1, sizeof(*func))) == NULL) {
1069 warn("calloc failed");
1070 goto cont_search;
1071 }
1072
1073 /*
1074 * Note that dwarf_attrval_unsigned() handles DW_AT_abstract_origin
1075 * internally, so it can retrieve DW_AT_decl_file/DW_AT_decl_line
1076 * attributes for inlined functions as well.
1077 */
1078 if (dwarf_attrval_unsigned(die, DW_AT_decl_file, &udata,
1079 &de) == DW_DLV_OK && udata > 0 &&
1080 (Dwarf_Signed) (udata - 1) < filecount) {
1081 func->file = strdup(src_files[udata - 1]);
1082 if (func->file == NULL) {
1083 warn("strdup");
1084 free(func);
1085 goto cont_search;
1086 }
1087 }
1088
1089 if (dwarf_attrval_unsigned(die, DW_AT_decl_line, &udata, &de) ==
1090 DW_DLV_OK)
1091 func->line = udata;
1092
1093 func->name = find_object_name(dbg, die);
1094 if (func->name == NULL) {
1095 if (func->file)
1096 free(func->file);
1097 free(func);
1098 goto cont_search;
1099 }
1100
1101 if (dwarf_attrval_unsigned(die, DW_AT_low_pc, &udata, &de) ==
1102 DW_DLV_OK)
1103 func->lowpc = udata;
1104 if (dwarf_attrval_unsigned(die, DW_AT_high_pc, &udata, &de) ==
1105 DW_DLV_OK)
1106 func->highpc = udata;
1107
1108 SLIST_INSERT_HEAD(func_info, func, entries);
1109 }
1110
1111 cont_search:
1112
1113 /* Search children. */
1114 ret = dwarf_child(die, &ret_die, &de);
1115 if (ret == DW_DLV_ERROR)
1116 warnx("dwarf_child: %s", dwarf_errmsg(de));
1117 else if (ret == DW_DLV_OK)
1118 search_line_attr(dbg, func_info, var_info, ret_die, src_files,
1119 filecount);
1120
1121 /* Search sibling. */
1122 ret = dwarf_siblingof(dbg, die, &ret_die, &de);
1123 if (ret == DW_DLV_ERROR)
1124 warnx("dwarf_siblingof: %s", dwarf_errmsg(de));
1125 else if (ret == DW_DLV_OK)
1126 search_line_attr(dbg, func_info, var_info, ret_die, src_files,
1127 filecount);
1128
1129 dwarf_dealloc(dbg, die, DW_DLA_DIE);
1130 }
1131
1132 /*
1133 * Read elf file and collect symbol information, sort them, print.
1134 * Return 1 at failed, 0 at success.
1135 */
1136 static int
read_elf(Elf * elf,const char * filename,Elf_Kind kind)1137 read_elf(Elf *elf, const char *filename, Elf_Kind kind)
1138 {
1139 Dwarf_Debug dbg;
1140 Dwarf_Die die;
1141 Dwarf_Error de;
1142 Dwarf_Half tag;
1143 Elf_Arhdr *arhdr;
1144 Elf_Scn *scn;
1145 GElf_Shdr shdr;
1146 GElf_Half i;
1147 Dwarf_Line *lbuf;
1148 Dwarf_Unsigned lineno;
1149 Dwarf_Signed lcount, filecount;
1150 Dwarf_Addr lineaddr;
1151 struct sym_print_data p_data;
1152 struct sym_head list_head;
1153 struct line_info_head *line_info;
1154 struct func_info_head *func_info;
1155 struct var_info_head *var_info;
1156 struct line_info_entry *lie;
1157 struct func_info_entry *func;
1158 struct var_info_entry *var;
1159 const char *shname, *objname;
1160 char *type_table, **sec_table, *sfile, **src_files;
1161 size_t shstrndx, shnum, dynndx, strndx;
1162 int ret, rtn, e_err;
1163
1164 #define OBJNAME (objname == NULL ? filename : objname)
1165
1166 assert(filename != NULL && "filename is null");
1167
1168 STAILQ_INIT(&list_head);
1169 type_table = NULL;
1170 sec_table = NULL;
1171 line_info = NULL;
1172 func_info = NULL;
1173 var_info = NULL;
1174 objname = NULL;
1175 dynndx = SHN_UNDEF;
1176 strndx = SHN_UNDEF;
1177 rtn = 0;
1178
1179 nm_elfclass = gelf_getclass(elf);
1180
1181 if (kind == ELF_K_AR) {
1182 if ((arhdr = elf_getarhdr(elf)) == NULL)
1183 goto next_cmd;
1184 objname = arhdr->ar_name != NULL ? arhdr->ar_name :
1185 arhdr->ar_rawname;
1186 }
1187 if (!elf_getshnum(elf, &shnum)) {
1188 if ((e_err = elf_errno()) != 0)
1189 warnx("%s: %s", OBJNAME, "File format not recognized");
1190 else
1191 warnx("%s: cannot get section number", OBJNAME);
1192 rtn = 1;
1193 goto next_cmd;
1194 }
1195 if (shnum == 0) {
1196 warnx("%s: has no section", OBJNAME);
1197 rtn = 1;
1198 goto next_cmd;
1199 }
1200 if (!elf_getshstrndx(elf, &shstrndx)) {
1201 warnx("%s: cannot get str index", OBJNAME);
1202 rtn = 1;
1203 goto next_cmd;
1204 }
1205 /* type_table for type determine */
1206 if ((type_table = malloc(sizeof(char) * shnum)) == NULL) {
1207 warn("%s: malloc", OBJNAME);
1208 rtn = 1;
1209 goto next_cmd;
1210 }
1211 /* sec_table for section name to display in sysv format */
1212 if ((sec_table = calloc(shnum, sizeof(char *))) == NULL) {
1213 warn("%s: calloc", OBJNAME);
1214 rtn = 1;
1215 goto next_cmd;
1216 }
1217
1218 type_table[0] = 'U';
1219 if ((sec_table[0] = strdup("*UND*")) == NULL) {
1220 warn("strdup");
1221 goto next_cmd;
1222 }
1223
1224 for (i = 1; i < shnum; ++i) {
1225 type_table[i] = 'U';
1226 if ((scn = elf_getscn(elf, i)) == NULL) {
1227 if ((e_err = elf_errno()) != 0)
1228 warnx("%s: %s", OBJNAME, elf_errmsg(e_err));
1229 else
1230 warnx("%s: cannot get section", OBJNAME);
1231 rtn = 1;
1232 goto next_cmd;
1233 }
1234 if (gelf_getshdr(scn, &shdr) == NULL)
1235 goto next_cmd;
1236
1237 /*
1238 * Cannot test by type and attribute for dynstr, strtab
1239 */
1240 shname = elf_strptr(elf, shstrndx, (size_t) shdr.sh_name);
1241 if (shname != NULL) {
1242 if ((sec_table[i] = strdup(shname)) == NULL) {
1243 warn("strdup");
1244 goto next_cmd;
1245 }
1246 if (!strncmp(shname, ".dynstr", 7)) {
1247 dynndx = elf_ndxscn(scn);
1248 if (dynndx == SHN_UNDEF) {
1249 warnx("%s: elf_ndxscn failed: %s",
1250 OBJNAME, elf_errmsg(-1));
1251 goto next_cmd;
1252 }
1253 }
1254 if (!strncmp(shname, ".strtab", 7)) {
1255 strndx = elf_ndxscn(scn);
1256 if (strndx == SHN_UNDEF) {
1257 warnx("%s: elf_ndxscn failed: %s",
1258 OBJNAME, elf_errmsg(-1));
1259 goto next_cmd;
1260 }
1261 }
1262 } else {
1263 sec_table[i] = strdup("*UND*");
1264 if (sec_table[i] == NULL) {
1265 warn("strdup");
1266 goto next_cmd;
1267 }
1268 }
1269
1270
1271 if (is_sec_text(&shdr))
1272 type_table[i] = 'T';
1273 else if (is_sec_data(&shdr)) {
1274 if (is_sec_readonly(&shdr))
1275 type_table[i] = 'R';
1276 else
1277 type_table[i] = 'D';
1278 } else if (is_sec_nobits(&shdr))
1279 type_table[i] = 'B';
1280 else if (is_sec_debug(shname))
1281 type_table[i] = 'N';
1282 else if (is_sec_readonly(&shdr) && !is_sec_nobits(&shdr))
1283 type_table[i] = 'n';
1284 }
1285
1286 print_header(filename, objname);
1287
1288 if ((dynndx == SHN_UNDEF && nm_opts.print_symbol == PRINT_SYM_DYN) ||
1289 (strndx == SHN_UNDEF && nm_opts.print_symbol == PRINT_SYM_SYM)) {
1290 warnx("%s: no symbols", OBJNAME);
1291 /* This is not an error case */
1292 goto next_cmd;
1293 }
1294
1295 STAILQ_INIT(&list_head);
1296
1297 if (!nm_opts.debug_line)
1298 goto process_sym;
1299
1300 /*
1301 * Collect dwarf line number information.
1302 */
1303
1304 if (dwarf_elf_init(elf, DW_DLC_READ, NULL, NULL, &dbg, &de) !=
1305 DW_DLV_OK) {
1306 warnx("dwarf_elf_init failed: %s", dwarf_errmsg(de));
1307 goto process_sym;
1308 }
1309
1310 line_info = malloc(sizeof(struct line_info_head));
1311 func_info = malloc(sizeof(struct func_info_head));
1312 var_info = malloc(sizeof(struct var_info_head));
1313 if (line_info != NULL)
1314 SLIST_INIT(line_info);
1315 if (func_info != NULL)
1316 SLIST_INIT(func_info);
1317 if (var_info != NULL)
1318 SLIST_INIT(var_info);
1319 if (line_info == NULL || func_info == NULL || var_info == NULL) {
1320 warn("malloc");
1321 (void) dwarf_finish(dbg, &de);
1322 goto process_sym;
1323 }
1324
1325 while ((ret = dwarf_next_cu_header(dbg, NULL, NULL, NULL, NULL, NULL,
1326 &de)) == DW_DLV_OK) {
1327 die = NULL;
1328 while (dwarf_siblingof(dbg, die, &die, &de) == DW_DLV_OK) {
1329 if (dwarf_tag(die, &tag, &de) != DW_DLV_OK) {
1330 warnx("dwarf_tag failed: %s",
1331 dwarf_errmsg(de));
1332 continue;
1333 }
1334 /* XXX: What about DW_TAG_partial_unit? */
1335 if (tag == DW_TAG_compile_unit)
1336 break;
1337 }
1338 if (die == NULL) {
1339 warnx("could not find DW_TAG_compile_unit die");
1340 continue;
1341 }
1342
1343 /* Retrieve source file list. */
1344 ret = dwarf_srcfiles(die, &src_files, &filecount, &de);
1345 if (ret == DW_DLV_ERROR)
1346 warnx("dwarf_srclines: %s", dwarf_errmsg(de));
1347 if (ret != DW_DLV_OK)
1348 continue;
1349
1350 /*
1351 * Retrieve line number information from .debug_line section.
1352 */
1353
1354 ret = dwarf_srclines(die, &lbuf, &lcount, &de);
1355 if (ret == DW_DLV_ERROR)
1356 warnx("dwarf_srclines: %s", dwarf_errmsg(de));
1357 if (ret != DW_DLV_OK)
1358 goto line_attr;
1359 for (i = 0; (Dwarf_Signed) i < lcount; i++) {
1360 if (dwarf_lineaddr(lbuf[i], &lineaddr, &de)) {
1361 warnx("dwarf_lineaddr: %s", dwarf_errmsg(de));
1362 continue;
1363 }
1364 if (dwarf_lineno(lbuf[i], &lineno, &de)) {
1365 warnx("dwarf_lineno: %s", dwarf_errmsg(de));
1366 continue;
1367 }
1368 if (dwarf_linesrc(lbuf[i], &sfile, &de)) {
1369 warnx("dwarf_linesrc: %s", dwarf_errmsg(de));
1370 continue;
1371 }
1372 if ((lie = malloc(sizeof(*lie))) == NULL) {
1373 warn("malloc");
1374 continue;
1375 }
1376 lie->addr = lineaddr;
1377 lie->line = lineno;
1378 lie->file = strdup(sfile);
1379 if (lie->file == NULL) {
1380 warn("strdup");
1381 free(lie);
1382 continue;
1383 }
1384 SLIST_INSERT_HEAD(line_info, lie, entries);
1385 }
1386
1387 line_attr:
1388 /* Retrieve line number information from DIEs. */
1389 search_line_attr(dbg, func_info, var_info, die, src_files, filecount);
1390 }
1391
1392 (void) dwarf_finish(dbg, &de);
1393
1394 process_sym:
1395
1396 p_data.list_num = get_sym(elf, &list_head, shnum, dynndx, strndx,
1397 type_table, (void *) sec_table, shnum);
1398
1399 if (p_data.list_num == 0)
1400 goto next_cmd;
1401
1402 p_data.headp = &list_head;
1403 p_data.sh_num = shnum;
1404 p_data.t_table = type_table;
1405 p_data.s_table = (void *) sec_table;
1406 p_data.filename = filename;
1407 p_data.objname = objname;
1408
1409 sym_list_print(&p_data, func_info, var_info, line_info);
1410
1411 next_cmd:
1412 if (nm_opts.debug_line) {
1413 if (func_info != NULL) {
1414 while (!SLIST_EMPTY(func_info)) {
1415 func = SLIST_FIRST(func_info);
1416 SLIST_REMOVE_HEAD(func_info, entries);
1417 free(func->file);
1418 free(func->name);
1419 free(func);
1420 }
1421 free(func_info);
1422 func_info = NULL;
1423 }
1424 if (var_info != NULL) {
1425 while (!SLIST_EMPTY(var_info)) {
1426 var = SLIST_FIRST(var_info);
1427 SLIST_REMOVE_HEAD(var_info, entries);
1428 free(var->file);
1429 free(var->name);
1430 free(var);
1431 }
1432 free(var_info);
1433 var_info = NULL;
1434 }
1435 if (line_info != NULL) {
1436 while (!SLIST_EMPTY(line_info)) {
1437 lie = SLIST_FIRST(line_info);
1438 SLIST_REMOVE_HEAD(line_info, entries);
1439 free(lie->file);
1440 free(lie);
1441 }
1442 free(line_info);
1443 line_info = NULL;
1444 }
1445 }
1446
1447 if (sec_table != NULL)
1448 for (i = 0; i < shnum; ++i)
1449 free(sec_table[i]);
1450 free(sec_table);
1451 free(type_table);
1452
1453 sym_list_dest(&list_head);
1454
1455 return (rtn);
1456
1457 #undef OBJNAME
1458 }
1459
1460 static int
read_object(const char * filename)1461 read_object(const char *filename)
1462 {
1463 Elf *elf, *arf;
1464 Elf_Cmd elf_cmd;
1465 Elf_Kind kind;
1466 int fd, rtn, e_err;
1467
1468 assert(filename != NULL && "filename is null");
1469
1470 if ((fd = open(filename, O_RDONLY)) == -1) {
1471 warn("'%s'", filename);
1472 return (1);
1473 }
1474
1475 elf_cmd = ELF_C_READ;
1476 if ((arf = elf_begin(fd, elf_cmd, (Elf *) NULL)) == NULL) {
1477 if ((e_err = elf_errno()) != 0)
1478 warnx("elf_begin error: %s", elf_errmsg(e_err));
1479 else
1480 warnx("elf_begin error");
1481 close(fd);
1482 return (1);
1483 }
1484
1485 assert(arf != NULL && "arf is null.");
1486
1487 rtn = 0;
1488 if ((kind = elf_kind(arf)) == ELF_K_NONE) {
1489 warnx("%s: File format not recognized", filename);
1490 elf_end(arf);
1491 close(fd);
1492 return (1);
1493 }
1494 if (kind == ELF_K_AR) {
1495 if (nm_opts.print_name == PRINT_NAME_MULTI &&
1496 nm_opts.elem_print_fn == sym_elem_print_all)
1497 printf("\n%s:\n", filename);
1498 if (nm_opts.print_armap == true)
1499 print_ar_index(fd, arf);
1500 }
1501
1502 while ((elf = elf_begin(fd, elf_cmd, arf)) != NULL) {
1503 rtn |= read_elf(elf, filename, kind);
1504
1505 /*
1506 * If file is not archive, elf_next return ELF_C_NULL and
1507 * stop the loop.
1508 */
1509 elf_cmd = elf_next(elf);
1510 elf_end(elf);
1511 }
1512
1513 elf_end(arf);
1514 close(fd);
1515
1516 return (rtn);
1517 }
1518
1519 static int
read_files(int argc,char ** argv)1520 read_files(int argc, char **argv)
1521 {
1522 int rtn = 0;
1523
1524 if (argc < 0 || argv == NULL)
1525 return (1);
1526
1527 if (argc == 0)
1528 rtn |= read_object(nm_info.def_filename);
1529 else {
1530 if (nm_opts.print_name == PRINT_NAME_NONE && argc > 1)
1531 nm_opts.print_name = PRINT_NAME_MULTI;
1532 while (argc > 0) {
1533 rtn |= read_object(*argv);
1534 --argc;
1535 ++argv;
1536 }
1537 }
1538
1539 return (rtn);
1540 }
1541
1542 static void
print_lineno(struct sym_entry * ep,struct func_info_head * func_info,struct var_info_head * var_info,struct line_info_head * line_info)1543 print_lineno(struct sym_entry *ep, struct func_info_head *func_info,
1544 struct var_info_head *var_info, struct line_info_head *line_info)
1545 {
1546 struct func_info_entry *func;
1547 struct var_info_entry *var;
1548 struct line_info_entry *lie;
1549
1550 /* For function symbol, search the function line information list. */
1551 if ((ep->sym->st_info & 0xf) == STT_FUNC && func_info != NULL) {
1552 SLIST_FOREACH(func, func_info, entries) {
1553 if (func->name != NULL &&
1554 !strcmp(ep->name, func->name) &&
1555 ep->sym->st_value >= func->lowpc &&
1556 ep->sym->st_value < func->highpc) {
1557 printf("\t%s:%" PRIu64, func->file, func->line);
1558 return;
1559 }
1560 }
1561 }
1562
1563 /* For variable symbol, search the variable line information list. */
1564 if ((ep->sym->st_info & 0xf) == STT_OBJECT && var_info != NULL) {
1565 SLIST_FOREACH(var, var_info, entries) {
1566 if (!strcmp(ep->name, var->name) &&
1567 ep->sym->st_value == var->addr) {
1568 printf("\t%s:%" PRIu64, var->file, var->line);
1569 return;
1570 }
1571 }
1572 }
1573
1574 /* Otherwise search line number information the .debug_line section. */
1575 if (line_info != NULL) {
1576 SLIST_FOREACH(lie, line_info, entries) {
1577 if (ep->sym->st_value == lie->addr) {
1578 printf("\t%s:%" PRIu64, lie->file, lie->line);
1579 return;
1580 }
1581 }
1582 }
1583 }
1584
1585 static void
set_opt_value_print_fn(enum radix t)1586 set_opt_value_print_fn(enum radix t)
1587 {
1588
1589 switch (t) {
1590 case RADIX_OCT:
1591 nm_opts.value_print_fn = &sym_value_oct_print;
1592 nm_opts.size_print_fn = &sym_size_oct_print;
1593
1594 break;
1595 case RADIX_DEC:
1596 nm_opts.value_print_fn = &sym_value_dec_print;
1597 nm_opts.size_print_fn = &sym_size_dec_print;
1598
1599 break;
1600 case RADIX_HEX:
1601 default :
1602 nm_opts.value_print_fn = &sym_value_hex_print;
1603 nm_opts.size_print_fn = &sym_size_hex_print;
1604 }
1605
1606 assert(nm_opts.value_print_fn != NULL &&
1607 "nm_opts.value_print_fn is null");
1608 }
1609
1610 static void
sym_elem_print_all(char type,const char * sec,const GElf_Sym * sym,const char * name)1611 sym_elem_print_all(char type, const char *sec, const GElf_Sym *sym,
1612 const char *name)
1613 {
1614
1615 if (sec == NULL || sym == NULL || name == NULL ||
1616 nm_opts.value_print_fn == NULL)
1617 return;
1618
1619 if (IS_UNDEF_SYM_TYPE(type)) {
1620 if (nm_opts.t == RADIX_HEX && nm_elfclass == ELFCLASS32)
1621 printf("%-8s", "");
1622 else
1623 printf("%-16s", "");
1624 } else {
1625 switch ((nm_opts.sort_fn == & cmp_size ? 2 : 0) +
1626 nm_opts.print_size) {
1627 case 3:
1628 if (sym->st_size != 0) {
1629 nm_opts.value_print_fn(sym);
1630 printf(" ");
1631 nm_opts.size_print_fn(sym);
1632 }
1633 break;
1634
1635 case 2:
1636 if (sym->st_size != 0)
1637 nm_opts.size_print_fn(sym);
1638 break;
1639
1640 case 1:
1641 nm_opts.value_print_fn(sym);
1642 if (sym->st_size != 0) {
1643 printf(" ");
1644 nm_opts.size_print_fn(sym);
1645 }
1646 break;
1647
1648 case 0:
1649 default:
1650 nm_opts.value_print_fn(sym);
1651 }
1652 }
1653
1654 printf(" %c ", type);
1655 PRINT_DEMANGLED_NAME("%s", name);
1656 }
1657
1658 static void
sym_elem_print_all_portable(char type,const char * sec,const GElf_Sym * sym,const char * name)1659 sym_elem_print_all_portable(char type, const char *sec, const GElf_Sym *sym,
1660 const char *name)
1661 {
1662
1663 if (sec == NULL || sym == NULL || name == NULL ||
1664 nm_opts.value_print_fn == NULL)
1665 return;
1666
1667 PRINT_DEMANGLED_NAME("%s", name);
1668 printf(" %c ", type);
1669 if (!IS_UNDEF_SYM_TYPE(type)) {
1670 nm_opts.value_print_fn(sym);
1671 printf(" ");
1672 if (sym->st_size != 0)
1673 nm_opts.size_print_fn(sym);
1674 } else
1675 printf(" ");
1676 }
1677
1678 static void
sym_elem_print_all_sysv(char type,const char * sec,const GElf_Sym * sym,const char * name)1679 sym_elem_print_all_sysv(char type, const char *sec, const GElf_Sym *sym,
1680 const char *name)
1681 {
1682
1683 if (sec == NULL || sym == NULL || name == NULL ||
1684 nm_opts.value_print_fn == NULL)
1685 return;
1686
1687 PRINT_DEMANGLED_NAME("%-20s|", name);
1688 if (IS_UNDEF_SYM_TYPE(type))
1689 printf(" ");
1690 else
1691 nm_opts.value_print_fn(sym);
1692
1693 printf("| %c |", type);
1694
1695 switch (sym->st_info & 0xf) {
1696 case STT_OBJECT:
1697 printf("%18s|", "OBJECT");
1698 break;
1699
1700 case STT_FUNC:
1701 printf("%18s|", "FUNC");
1702 break;
1703
1704 case STT_SECTION:
1705 printf("%18s|", "SECTION");
1706 break;
1707
1708 case STT_FILE:
1709 printf("%18s|", "FILE");
1710 break;
1711
1712 case STT_LOPROC:
1713 printf("%18s|", "LOPROC");
1714 break;
1715
1716 case STT_HIPROC:
1717 printf("%18s|", "HIPROC");
1718 break;
1719
1720 case STT_NOTYPE:
1721 default:
1722 printf("%18s|", "NOTYPE");
1723 }
1724
1725 if (sym->st_size != 0)
1726 nm_opts.size_print_fn(sym);
1727 else
1728 printf(" ");
1729
1730 printf("| |%s", sec);
1731 }
1732
1733 static int
sym_elem_def(char type,const GElf_Sym * sym,const char * name)1734 sym_elem_def(char type, const GElf_Sym *sym, const char *name)
1735 {
1736
1737 assert(IS_SYM_TYPE((unsigned char) type));
1738
1739 UNUSED(sym);
1740 UNUSED(name);
1741
1742 return (!IS_UNDEF_SYM_TYPE((unsigned char) type));
1743 }
1744
1745 static int
sym_elem_global(char type,const GElf_Sym * sym,const char * name)1746 sym_elem_global(char type, const GElf_Sym *sym, const char *name)
1747 {
1748
1749 assert(IS_SYM_TYPE((unsigned char) type));
1750
1751 UNUSED(sym);
1752 UNUSED(name);
1753
1754 /* weak symbols resemble global. */
1755 return (isupper((unsigned char) type) || type == 'w');
1756 }
1757
1758 static int
sym_elem_global_static(char type,const GElf_Sym * sym,const char * name)1759 sym_elem_global_static(char type, const GElf_Sym *sym, const char *name)
1760 {
1761 unsigned char info;
1762
1763 assert(sym != NULL);
1764
1765 UNUSED(type);
1766 UNUSED(name);
1767
1768 info = sym->st_info >> 4;
1769
1770 return (info == STB_LOCAL ||
1771 info == STB_GLOBAL ||
1772 info == STB_WEAK);
1773 }
1774
1775 static int
sym_elem_nondebug(char type,const GElf_Sym * sym,const char * name)1776 sym_elem_nondebug(char type, const GElf_Sym *sym, const char *name)
1777 {
1778
1779 assert(sym != NULL);
1780
1781 UNUSED(type);
1782 UNUSED(name);
1783
1784 if (sym->st_value == 0 && (sym->st_info & 0xf) == STT_FILE)
1785 return (0);
1786 if (sym->st_name == 0)
1787 return (0);
1788
1789 return (1);
1790 }
1791
1792 static int
sym_elem_nonzero_size(char type,const GElf_Sym * sym,const char * name)1793 sym_elem_nonzero_size(char type, const GElf_Sym *sym, const char *name)
1794 {
1795
1796 assert(sym != NULL);
1797
1798 UNUSED(type);
1799 UNUSED(name);
1800
1801 return (sym->st_size > 0);
1802 }
1803
1804 static int
sym_elem_undef(char type,const GElf_Sym * sym,const char * name)1805 sym_elem_undef(char type, const GElf_Sym *sym, const char *name)
1806 {
1807
1808 assert(IS_SYM_TYPE((unsigned char) type));
1809
1810 UNUSED(sym);
1811 UNUSED(name);
1812
1813 return (IS_UNDEF_SYM_TYPE((unsigned char) type));
1814 }
1815
1816 static void
sym_list_dest(struct sym_head * headp)1817 sym_list_dest(struct sym_head *headp)
1818 {
1819 struct sym_entry *ep, *ep_n;
1820
1821 if (headp == NULL)
1822 return;
1823
1824 ep = STAILQ_FIRST(headp);
1825 while (ep != NULL) {
1826 ep_n = STAILQ_NEXT(ep, sym_entries);
1827 free(ep->sym);
1828 free(ep->name);
1829 free(ep);
1830 ep = ep_n;
1831 }
1832 }
1833
1834 static int
sym_list_insert(struct sym_head * headp,const char * name,const GElf_Sym * sym)1835 sym_list_insert(struct sym_head *headp, const char *name, const GElf_Sym *sym)
1836 {
1837 struct sym_entry *e;
1838
1839 if (headp == NULL || name == NULL || sym == NULL)
1840 return (0);
1841 if ((e = malloc(sizeof(struct sym_entry))) == NULL) {
1842 warn("malloc");
1843 return (0);
1844 }
1845 if ((e->name = strdup(name)) == NULL) {
1846 warn("strdup");
1847 free(e);
1848 return (0);
1849 }
1850 if ((e->sym = malloc(sizeof(GElf_Sym))) == NULL) {
1851 warn("malloc");
1852 free(e->name);
1853 free(e);
1854 return (0);
1855 }
1856
1857 memcpy(e->sym, sym, sizeof(GElf_Sym));
1858
1859 /* Display size instead of value for common symbol. */
1860 if (sym->st_shndx == SHN_COMMON)
1861 e->sym->st_value = sym->st_size;
1862
1863 STAILQ_INSERT_TAIL(headp, e, sym_entries);
1864
1865 return (1);
1866 }
1867
1868 /* If file has not .debug_info, line_info will be NULL */
1869 static void
sym_list_print(struct sym_print_data * p,struct func_info_head * func_info,struct var_info_head * var_info,struct line_info_head * line_info)1870 sym_list_print(struct sym_print_data *p, struct func_info_head *func_info,
1871 struct var_info_head *var_info, struct line_info_head *line_info)
1872 {
1873 struct sym_entry *e_v;
1874 size_t si;
1875 int i;
1876
1877 if (p == NULL || CHECK_SYM_PRINT_DATA(p))
1878 return;
1879 if ((e_v = sym_list_sort(p)) == NULL)
1880 return;
1881 if (nm_opts.sort_reverse == false)
1882 for (si = 0; si != p->list_num; ++si)
1883 sym_list_print_each(&e_v[si], p, func_info, var_info,
1884 line_info);
1885 else
1886 for (i = p->list_num - 1; i != -1; --i)
1887 sym_list_print_each(&e_v[i], p, func_info, var_info,
1888 line_info);
1889
1890 free(e_v);
1891 }
1892
1893 /* If file has not .debug_info, line_info will be NULL */
1894 static void
sym_list_print_each(struct sym_entry * ep,struct sym_print_data * p,struct func_info_head * func_info,struct var_info_head * var_info,struct line_info_head * line_info)1895 sym_list_print_each(struct sym_entry *ep, struct sym_print_data *p,
1896 struct func_info_head *func_info, struct var_info_head *var_info,
1897 struct line_info_head *line_info)
1898 {
1899 const char *sec;
1900 char type;
1901
1902 if (ep == NULL || CHECK_SYM_PRINT_DATA(p))
1903 return;
1904
1905 assert(ep->name != NULL);
1906 assert(ep->sym != NULL);
1907
1908 type = get_sym_type(ep->sym, p->t_table);
1909
1910 if (nm_opts.print_name == PRINT_NAME_FULL) {
1911 printf("%s", p->filename);
1912 if (nm_opts.elem_print_fn == &sym_elem_print_all_portable) {
1913 if (p->objname != NULL)
1914 printf("[%s]", p->objname);
1915 printf(": ");
1916 } else {
1917 if (p->objname != NULL)
1918 printf(":%s", p->objname);
1919 printf(":");
1920 }
1921 }
1922
1923 switch (ep->sym->st_shndx) {
1924 case SHN_LOPROC:
1925 /* LOPROC or LORESERVE */
1926 sec = "*LOPROC*";
1927 break;
1928 case SHN_HIPROC:
1929 sec = "*HIPROC*";
1930 break;
1931 case SHN_LOOS:
1932 sec = "*LOOS*";
1933 break;
1934 case SHN_HIOS:
1935 sec = "*HIOS*";
1936 break;
1937 case SHN_ABS:
1938 sec = "*ABS*";
1939 break;
1940 case SHN_COMMON:
1941 sec = "*COM*";
1942 break;
1943 case SHN_HIRESERVE:
1944 /* HIRESERVE or XINDEX */
1945 sec = "*HIRESERVE*";
1946 break;
1947 default:
1948 if (ep->sym->st_shndx > p->sh_num)
1949 return;
1950 sec = p->s_table[ep->sym->st_shndx];
1951 break;
1952 }
1953
1954 nm_opts.elem_print_fn(type, sec, ep->sym, ep->name);
1955
1956 if (nm_opts.debug_line == true && !IS_UNDEF_SYM_TYPE(type))
1957 print_lineno(ep, func_info, var_info, line_info);
1958
1959 printf("\n");
1960 }
1961
1962 static struct sym_entry *
sym_list_sort(struct sym_print_data * p)1963 sym_list_sort(struct sym_print_data *p)
1964 {
1965 struct sym_entry *ep, *e_v;
1966 int idx;
1967
1968 if (p == NULL || CHECK_SYM_PRINT_DATA(p))
1969 return (NULL);
1970
1971 if ((e_v = malloc(sizeof(struct sym_entry) * p->list_num)) == NULL) {
1972 warn("malloc");
1973 return (NULL);
1974 }
1975
1976 idx = 0;
1977 STAILQ_FOREACH(ep, p->headp, sym_entries) {
1978 if (ep->name != NULL && ep->sym != NULL) {
1979 e_v[idx].name = ep->name;
1980 e_v[idx].sym = ep->sym;
1981 ++idx;
1982 }
1983 }
1984
1985 assert((size_t)idx == p->list_num);
1986
1987 if (nm_opts.sort_fn != &cmp_none) {
1988 nm_print_data = p;
1989 assert(nm_print_data != NULL);
1990 qsort(e_v, p->list_num, sizeof(struct sym_entry),
1991 nm_opts.sort_fn);
1992 }
1993
1994 return (e_v);
1995 }
1996
1997 static void
sym_size_oct_print(const GElf_Sym * sym)1998 sym_size_oct_print(const GElf_Sym *sym)
1999 {
2000
2001 assert(sym != NULL && "sym is null");
2002 printf("%016" PRIo64, sym->st_size);
2003 }
2004
2005 static void
sym_size_hex_print(const GElf_Sym * sym)2006 sym_size_hex_print(const GElf_Sym *sym)
2007 {
2008
2009 assert(sym != NULL && "sym is null");
2010 if (nm_elfclass == ELFCLASS32)
2011 printf("%08" PRIx64, sym->st_size);
2012 else
2013 printf("%016" PRIx64, sym->st_size);
2014 }
2015
2016 static void
sym_size_dec_print(const GElf_Sym * sym)2017 sym_size_dec_print(const GElf_Sym *sym)
2018 {
2019
2020 assert(sym != NULL && "sym is null");
2021 printf("%016" PRId64, sym->st_size);
2022 }
2023
2024 static void
sym_value_oct_print(const GElf_Sym * sym)2025 sym_value_oct_print(const GElf_Sym *sym)
2026 {
2027
2028 assert(sym != NULL && "sym is null");
2029 printf("%016" PRIo64, sym->st_value);
2030 }
2031
2032 static void
sym_value_hex_print(const GElf_Sym * sym)2033 sym_value_hex_print(const GElf_Sym *sym)
2034 {
2035
2036 assert(sym != NULL && "sym is null");
2037 if (nm_elfclass == ELFCLASS32)
2038 printf("%08" PRIx64, sym->st_value);
2039 else
2040 printf("%016" PRIx64, sym->st_value);
2041 }
2042
2043 static void
sym_value_dec_print(const GElf_Sym * sym)2044 sym_value_dec_print(const GElf_Sym *sym)
2045 {
2046
2047 assert(sym != NULL && "sym is null");
2048 printf("%016" PRId64, sym->st_value);
2049 }
2050
2051 static void
usage(int exitcode)2052 usage(int exitcode)
2053 {
2054
2055 printf("Usage: %s [options] file ...\
2056 \n Display symbolic information in file.\n\
2057 \n Options: \
2058 \n -A, --print-file-name Write the full pathname or library name of an\
2059 \n object on each line.\
2060 \n -a, --debug-syms Display all symbols include debugger-only\
2061 \n symbols.", nm_info.name);
2062 printf("\
2063 \n -B Equivalent to specifying \"--format=bsd\".\
2064 \n -C, --demangle[=style] Decode low-level symbol names.\
2065 \n --no-demangle Do not demangle low-level symbol names.\
2066 \n -D, --dynamic Display only dynamic symbols.\
2067 \n -e Display only global and static symbols.");
2068 printf("\
2069 \n -f Produce full output (default).\
2070 \n --format=format Display output in specific format. Allowed\
2071 \n formats are: \"bsd\", \"posix\" and \"sysv\".\
2072 \n -g, --extern-only Display only global symbol information.\
2073 \n -h, --help Show this help message.\
2074 \n -l, --line-numbers Display filename and linenumber using\
2075 \n debugging information.\
2076 \n -n, --numeric-sort Sort symbols numerically by value.");
2077 printf("\
2078 \n -o Write numeric values in octal. Equivalent to\
2079 \n specifying \"-t o\".\
2080 \n -p, --no-sort Do not sort symbols.\
2081 \n -P Write information in a portable output format.\
2082 \n Equivalent to specifying \"--format=posix\".\
2083 \n -r, --reverse-sort Reverse the order of the sort.\
2084 \n -S, --print-size Print symbol sizes instead values.\
2085 \n -s, --print-armap Include an index of archive members.\
2086 \n --size-sort Sort symbols by size.");
2087 printf("\
2088 \n -t, --radix=format Write each numeric value in the specified\
2089 \n format:\
2090 \n d In decimal,\
2091 \n o In octal,\
2092 \n x In hexadecimal.");
2093 printf("\
2094 \n -u, --undefined-only Display only undefined symbols.\
2095 \n --defined-only Display only defined symbols.\
2096 \n -V, --version Show the version identifier for %s.\
2097 \n -v Sort output by value.\
2098 \n -x Write numeric values in hexadecimal.\
2099 \n Equivalent to specifying \"-t x\".",
2100 nm_info.name);
2101 printf("\n\
2102 \n The default options are: output in bsd format, use a hexadecimal radix,\
2103 \n sort by symbol name, do not demangle names.\n");
2104
2105 exit(exitcode);
2106 }
2107
2108 /*
2109 * Display symbolic information in file.
2110 * Return 0 at success, >0 at failed.
2111 */
2112 int
main(int argc,char ** argv)2113 main(int argc, char **argv)
2114 {
2115 int rtn;
2116
2117 global_init();
2118 get_opt(argc, argv);
2119 rtn = read_files(argc - optind, argv + optind);
2120 global_dest();
2121
2122 exit(rtn);
2123 }
2124