2005-09-30 H.J. Lu <hongjiu.lu@intel.com>
[deliverable/binutils-gdb.git] / binutils / nm.c
CommitLineData
252b5132 1/* nm.c -- Describe symbol table of a rel file.
8c2bc687 2 Copyright 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000,
8615f3f2 3 2001, 2002, 2003, 2004, 2005
252b5132
RH
4 Free Software Foundation, Inc.
5
6 This file is part of GNU Binutils.
7
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
b43b5d5f
NC
20 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
21 02110-1301, USA. */
252b5132
RH
22
23#include "bfd.h"
24#include "progress.h"
25#include "bucomm.h"
a6637ec0 26#include "budemang.h"
252b5132
RH
27#include "getopt.h"
28#include "aout/stab_gnu.h"
29#include "aout/ranlib.h"
30#include "demangle.h"
31#include "libiberty.h"
6ab6b380 32#include "elf-bfd.h"
33f5f537 33#include "elf/common.h"
252b5132
RH
34
35/* When sorting by size, we use this structure to hold the size and a
36 pointer to the minisymbol. */
37
38struct size_sym
39{
2da42df6 40 const void *minisym;
252b5132
RH
41 bfd_vma size;
42};
43
44/* When fetching relocs, we use this structure to pass information to
45 get_relocs. */
46
47struct get_relocs_info
48{
49 asection **secs;
50 arelent ***relocs;
51 long *relcount;
52 asymbol **syms;
53};
54
9710509e 55struct extended_symbol_info
977f7911
NC
56{
57 symbol_info *sinfo;
58 bfd_vma ssize;
33f5f537 59 elf_symbol_type *elfinfo;
977f7911
NC
60 /* FIXME: We should add more fields for Type, Line, Section. */
61};
62#define SYM_NAME(sym) (sym->sinfo->name)
63#define SYM_VALUE(sym) (sym->sinfo->value)
64#define SYM_TYPE(sym) (sym->sinfo->type)
65#define SYM_STAB_NAME(sym) (sym->sinfo->stab_name)
66#define SYM_STAB_DESC(sym) (sym->sinfo->stab_desc)
67#define SYM_STAB_OTHER(sym) (sym->sinfo->stab_other)
33f5f537
L
68#define SYM_SIZE(sym) \
69 (sym->elfinfo ? sym->elfinfo->internal_elf_sym.st_size: sym->ssize)
977f7911 70
252b5132 71/* The output formatting functions. */
2da42df6
AJ
72static void print_object_filename_bsd (char *);
73static void print_object_filename_sysv (char *);
74static void print_object_filename_posix (char *);
75static void print_archive_filename_bsd (char *);
76static void print_archive_filename_sysv (char *);
77static void print_archive_filename_posix (char *);
78static void print_archive_member_bsd (char *, const char *);
79static void print_archive_member_sysv (char *, const char *);
80static void print_archive_member_posix (char *, const char *);
81static void print_symbol_filename_bsd (bfd *, bfd *);
82static void print_symbol_filename_sysv (bfd *, bfd *);
83static void print_symbol_filename_posix (bfd *, bfd *);
84static void print_value (bfd *, bfd_vma);
85static void print_symbol_info_bsd (struct extended_symbol_info *, bfd *);
86static void print_symbol_info_sysv (struct extended_symbol_info *, bfd *);
87static void print_symbol_info_posix (struct extended_symbol_info *, bfd *);
252b5132
RH
88
89/* Support for different output formats. */
90struct output_fns
91 {
92 /* Print the name of an object file given on the command line. */
2da42df6 93 void (*print_object_filename) (char *);
252b5132
RH
94
95 /* Print the name of an archive file given on the command line. */
2da42df6 96 void (*print_archive_filename) (char *);
252b5132
RH
97
98 /* Print the name of an archive member file. */
2da42df6 99 void (*print_archive_member) (char *, const char *);
252b5132
RH
100
101 /* Print the name of the file (and archive, if there is one)
102 containing a symbol. */
2da42df6 103 void (*print_symbol_filename) (bfd *, bfd *);
252b5132
RH
104
105 /* Print a line of information about a symbol. */
2da42df6 106 void (*print_symbol_info) (struct extended_symbol_info *, bfd *);
252b5132 107 };
977f7911 108
252b5132
RH
109static struct output_fns formats[] =
110{
111 {print_object_filename_bsd,
112 print_archive_filename_bsd,
113 print_archive_member_bsd,
114 print_symbol_filename_bsd,
115 print_symbol_info_bsd},
116 {print_object_filename_sysv,
117 print_archive_filename_sysv,
118 print_archive_member_sysv,
119 print_symbol_filename_sysv,
120 print_symbol_info_sysv},
121 {print_object_filename_posix,
122 print_archive_filename_posix,
123 print_archive_member_posix,
124 print_symbol_filename_posix,
125 print_symbol_info_posix}
126};
127
128/* Indices in `formats'. */
129#define FORMAT_BSD 0
130#define FORMAT_SYSV 1
131#define FORMAT_POSIX 2
132#define FORMAT_DEFAULT FORMAT_BSD
133
134/* The output format to use. */
135static struct output_fns *format = &formats[FORMAT_DEFAULT];
136
252b5132
RH
137/* Command options. */
138
139static int do_demangle = 0; /* Pretty print C++ symbol names. */
977f7911
NC
140static int external_only = 0; /* Print external symbols only. */
141static int defined_only = 0; /* Print defined symbols only. */
142static int no_sort = 0; /* Don't sort; print syms in order found. */
143static int print_debug_syms = 0;/* Print debugger-only symbols too. */
144static int print_armap = 0; /* Describe __.SYMDEF data in archive files. */
72797995 145static int print_size = 0; /* Print size of defined symbols. */
977f7911
NC
146static int reverse_sort = 0; /* Sort in downward(alpha or numeric) order. */
147static int sort_numerically = 0;/* Sort in numeric rather than alpha order. */
148static int sort_by_size = 0; /* Sort by size of symbol. */
149static int undefined_only = 0; /* Print undefined symbols only. */
150static int dynamic = 0; /* Print dynamic symbols. */
151static int show_version = 0; /* Show the version number. */
152static int show_stats = 0; /* Show statistics. */
0873df2a 153static int show_synthetic = 0; /* Display synthesized symbols too. */
977f7911 154static int line_numbers = 0; /* Print line numbers for symbols. */
3c9458e9 155static int allow_special_symbols = 0; /* Allow special symbols. */
252b5132
RH
156
157/* When to print the names of files. Not mutually exclusive in SYSV format. */
158static int filename_per_file = 0; /* Once per file, on its own line. */
159static int filename_per_symbol = 0; /* Once per symbol, at start of line. */
160
161/* Print formats for printing a symbol value. */
162#ifndef BFD64
163static char value_format[] = "%08lx";
164#else
165#if BFD_HOST_64BIT_LONG
166static char value_format[] = "%016lx";
167#else
168/* We don't use value_format for this case. */
169#endif
170#endif
62a5a82d
L
171#ifdef BFD64
172static int print_width = 16;
173#else
174static int print_width = 8;
175#endif
252b5132
RH
176static int print_radix = 16;
177/* Print formats for printing stab info. */
178static char other_format[] = "%02x";
179static char desc_format[] = "%04x";
180
181static char *target = NULL;
182
183/* Used to cache the line numbers for a BFD. */
184static bfd *lineno_cache_bfd;
185static bfd *lineno_cache_rel_bfd;
186
c20f4f8c
AM
187#define OPTION_TARGET 200
188
252b5132
RH
189static struct option long_options[] =
190{
191 {"debug-syms", no_argument, &print_debug_syms, 1},
28c309a2 192 {"demangle", optional_argument, 0, 'C'},
252b5132
RH
193 {"dynamic", no_argument, &dynamic, 1},
194 {"extern-only", no_argument, &external_only, 1},
195 {"format", required_argument, 0, 'f'},
196 {"help", no_argument, 0, 'h'},
197 {"line-numbers", no_argument, 0, 'l'},
198 {"no-cplus", no_argument, &do_demangle, 0}, /* Linux compatibility. */
199 {"no-demangle", no_argument, &do_demangle, 0},
200 {"no-sort", no_argument, &no_sort, 1},
201 {"numeric-sort", no_argument, &sort_numerically, 1},
202 {"portability", no_argument, 0, 'P'},
203 {"print-armap", no_argument, &print_armap, 1},
204 {"print-file-name", no_argument, 0, 'o'},
72797995 205 {"print-size", no_argument, 0, 'S'},
252b5132
RH
206 {"radix", required_argument, 0, 't'},
207 {"reverse-sort", no_argument, &reverse_sort, 1},
208 {"size-sort", no_argument, &sort_by_size, 1},
3c9458e9 209 {"special-syms", no_argument, &allow_special_symbols, 1},
252b5132 210 {"stats", no_argument, &show_stats, 1},
0873df2a 211 {"synthetic", no_argument, &show_synthetic, 1},
c20f4f8c 212 {"target", required_argument, 0, OPTION_TARGET},
252b5132
RH
213 {"defined-only", no_argument, &defined_only, 1},
214 {"undefined-only", no_argument, &undefined_only, 1},
215 {"version", no_argument, &show_version, 1},
216 {0, no_argument, 0, 0}
217};
218\f
977f7911 219/* Some error-reporting functions. */
252b5132
RH
220
221static void
2da42df6 222usage (FILE *stream, int status)
252b5132 223{
8b53311e
NC
224 fprintf (stream, _("Usage: %s [option(s)] [file(s)]\n"), program_name);
225 fprintf (stream, _(" List symbols in [file(s)] (a.out by default).\n"));
226 fprintf (stream, _(" The options are:\n\
b56f55ce
NC
227 -a, --debug-syms Display debugger-only symbols\n\
228 -A, --print-file-name Print name of the input file before every symbol\n\
229 -B Same as --format=bsd\n\
28c309a2
NC
230 -C, --demangle[=STYLE] Decode low-level symbol names into user-level names\n\
231 The STYLE, if specified, can be `auto' (the default),\n\
f0c8c24a
NC
232 `gnu', `lucid', `arm', `hp', `edg', `gnu-v3', `java'\n\
233 or `gnat'\n\
b56f55ce
NC
234 --no-demangle Do not demangle low-level symbol names\n\
235 -D, --dynamic Display dynamic symbols instead of normal symbols\n\
236 --defined-only Display only defined symbols\n\
237 -e (ignored)\n\
238 -f, --format=FORMAT Use the output format FORMAT. FORMAT can be `bsd',\n\
239 `sysv' or `posix'. The default is `bsd'\n\
240 -g, --extern-only Display only external symbols\n\
b56f55ce
NC
241 -l, --line-numbers Use debugging information to find a filename and\n\
242 line number for each symbol\n\
243 -n, --numeric-sort Sort symbols numerically by address\n\
244 -o Same as -A\n\
245 -p, --no-sort Do not sort the symbols\n\
246 -P, --portability Same as --format=posix\n\
247 -r, --reverse-sort Reverse the sense of the sort\n\
72797995 248 -S, --print-size Print size of defined symbols\n\
b56f55ce
NC
249 -s, --print-armap Include index for symbols from archive members\n\
250 --size-sort Sort symbols by size\n\
61bbd35b 251 --special-syms Include special symbols in the output\n\
0873df2a 252 --synthetic Display synthetic symbols as well\n\
b56f55ce
NC
253 -t, --radix=RADIX Use RADIX for printing symbol values\n\
254 --target=BFDNAME Specify the target object format as BFDNAME\n\
255 -u, --undefined-only Display only undefined symbols\n\
6e800839 256 -X 32_64 (ignored)\n\
8b53311e
NC
257 -h, --help Display this information\n\
258 -V, --version Display this program's version number\n\
b56f55ce 259\n"));
252b5132
RH
260 list_supported_targets (program_name, stream);
261 if (status == 0)
b56f55ce 262 fprintf (stream, _("Report bugs to %s.\n"), REPORT_BUGS_TO);
252b5132
RH
263 exit (status);
264}
265
266/* Set the radix for the symbol value and size according to RADIX. */
267
268static void
2da42df6 269set_print_radix (char *radix)
252b5132
RH
270{
271 switch (*radix)
272 {
273 case 'x':
274 break;
275 case 'd':
276 case 'o':
277 if (*radix == 'd')
278 print_radix = 10;
279 else
280 print_radix = 8;
281#ifndef BFD64
282 value_format[4] = *radix;
283#else
284#if BFD_HOST_64BIT_LONG
285 value_format[5] = *radix;
286#else
287 /* This case requires special handling for octal and decimal
288 printing. */
289#endif
290#endif
291 other_format[3] = desc_format[3] = *radix;
292 break;
293 default:
37cc8ec1 294 fatal (_("%s: invalid radix"), radix);
252b5132
RH
295 }
296}
297
298static void
2da42df6 299set_output_format (char *f)
252b5132
RH
300{
301 int i;
302
303 switch (*f)
304 {
305 case 'b':
306 case 'B':
307 i = FORMAT_BSD;
308 break;
309 case 'p':
310 case 'P':
311 i = FORMAT_POSIX;
312 break;
313 case 's':
314 case 'S':
315 i = FORMAT_SYSV;
316 break;
317 default:
37cc8ec1 318 fatal (_("%s: invalid output format"), f);
252b5132
RH
319 }
320 format = &formats[i];
321}
322\f
33f5f537 323static const char *
2da42df6 324get_symbol_type (unsigned int type)
33f5f537
L
325{
326 static char buff [32];
327
328 switch (type)
329 {
330 case STT_NOTYPE: return "NOTYPE";
331 case STT_OBJECT: return "OBJECT";
332 case STT_FUNC: return "FUNC";
333 case STT_SECTION: return "SECTION";
334 case STT_FILE: return "FILE";
335 case STT_COMMON: return "COMMON";
336 case STT_TLS: return "TLS";
337 default:
338 if (type >= STT_LOPROC && type <= STT_HIPROC)
339 sprintf (buff, _("<processor specific>: %d"), type);
340 else if (type >= STT_LOOS && type <= STT_HIOS)
341 sprintf (buff, _("<OS specific>: %d"), type);
342 else
343 sprintf (buff, _("<unknown>: %d"), type);
344 return buff;
345 }
346}
382c1116
NC
347\f
348/* Print symbol name NAME, read from ABFD, with printf format FORMAT,
349 demangling it if requested. */
33f5f537 350
252b5132 351static void
382c1116 352print_symname (const char *format, const char *name, bfd *abfd)
252b5132 353{
382c1116 354 if (do_demangle && *name)
252b5132 355 {
382c1116 356 char *res = demangle (abfd, name);
252b5132 357
382c1116
NC
358 printf (format, res);
359 free (res);
360 return;
361 }
252b5132 362
382c1116
NC
363 printf (format, name);
364}
252b5132 365
382c1116
NC
366static void
367print_symdef_entry (bfd *abfd)
368{
369 symindex idx = BFD_NO_MORE_SYMBOLS;
370 carsym *thesym;
371 bfd_boolean everprinted = FALSE;
33f5f537 372
382c1116
NC
373 for (idx = bfd_get_next_mapent (abfd, idx, &thesym);
374 idx != BFD_NO_MORE_SYMBOLS;
375 idx = bfd_get_next_mapent (abfd, idx, &thesym))
376 {
377 bfd *elt;
378 if (!everprinted)
252b5132 379 {
382c1116
NC
380 printf (_("\nArchive index:\n"));
381 everprinted = TRUE;
252b5132 382 }
382c1116
NC
383 elt = bfd_get_elt_at_index (abfd, idx);
384 if (elt == NULL)
385 bfd_fatal ("bfd_get_elt_at_index");
386 if (thesym->name != (char *) NULL)
252b5132 387 {
382c1116
NC
388 print_symname ("%s", thesym->name, abfd);
389 printf (" in %s\n", bfd_get_filename (elt));
252b5132 390 }
252b5132
RH
391 }
392}
382c1116
NC
393\f
394/* Choose which symbol entries to print;
395 compact them downward to get rid of the rest.
396 Return the number of symbols to be printed. */
252b5132 397
382c1116
NC
398static long
399filter_symbols (bfd *abfd, bfd_boolean dynamic, void *minisyms,
400 long symcount, unsigned int size)
252b5132 401{
382c1116
NC
402 bfd_byte *from, *fromend, *to;
403 asymbol *store;
252b5132 404
382c1116
NC
405 store = bfd_make_empty_symbol (abfd);
406 if (store == NULL)
407 bfd_fatal (bfd_get_filename (abfd));
f24ddbdd 408
382c1116
NC
409 from = (bfd_byte *) minisyms;
410 fromend = from + symcount * size;
411 to = (bfd_byte *) minisyms;
252b5132 412
382c1116 413 for (; from < fromend; from += size)
252b5132 414 {
382c1116
NC
415 int keep = 0;
416 asymbol *sym;
33f5f537 417
382c1116
NC
418 PROGRESS (1);
419
420 sym = bfd_minisymbol_to_symbol (abfd, dynamic, (const void *) from, store);
421 if (sym == NULL)
422 bfd_fatal (bfd_get_filename (abfd));
423
424 if (undefined_only)
425 keep = bfd_is_und_section (sym->section);
426 else if (external_only)
427 keep = ((sym->flags & BSF_GLOBAL) != 0
428 || (sym->flags & BSF_WEAK) != 0
429 || bfd_is_und_section (sym->section)
430 || bfd_is_com_section (sym->section));
431 else
432 keep = 1;
433
434 if (keep
435 && ! print_debug_syms
436 && (sym->flags & BSF_DEBUGGING) != 0)
437 keep = 0;
438
439 if (keep
440 && sort_by_size
441 && (bfd_is_abs_section (sym->section)
442 || bfd_is_und_section (sym->section)))
443 keep = 0;
444
445 if (keep
446 && defined_only)
252b5132 447 {
382c1116
NC
448 if (bfd_is_und_section (sym->section))
449 keep = 0;
252b5132 450 }
252b5132 451
3c9458e9
NC
452 if (keep
453 && bfd_is_target_special_symbol (abfd, sym)
454 && ! allow_special_symbols)
455 keep = 0;
456
382c1116
NC
457 if (keep)
458 {
459 memcpy (to, from, size);
460 to += size;
461 }
462 }
252b5132 463
382c1116 464 return (to - (bfd_byte *) minisyms) / size;
252b5132
RH
465}
466\f
467/* These globals are used to pass information into the sorting
468 routines. */
469static bfd *sort_bfd;
b34976b6 470static bfd_boolean sort_dynamic;
252b5132
RH
471static asymbol *sort_x;
472static asymbol *sort_y;
473
474/* Symbol-sorting predicates */
475#define valueof(x) ((x)->section->vma + (x)->value)
476
477/* Numeric sorts. Undefined symbols are always considered "less than"
478 defined symbols with zero values. Common symbols are not treated
479 specially -- i.e., their sizes are used as their "values". */
480
252b5132 481static int
2da42df6 482non_numeric_forward (const void *P_x, const void *P_y)
252b5132
RH
483{
484 asymbol *x, *y;
485 const char *xn, *yn;
486
487 x = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_x, sort_x);
488 y = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_y, sort_y);
489 if (x == NULL || y == NULL)
490 bfd_fatal (bfd_get_filename (sort_bfd));
491
492 xn = bfd_asymbol_name (x);
493 yn = bfd_asymbol_name (y);
494
9710509e
AM
495 if (yn == NULL)
496 return xn != NULL;
497 if (xn == NULL)
498 return -1;
499
500#ifdef HAVE_STRCOLL
501 /* Solaris 2.5 has a bug in strcoll.
502 strcoll returns invalid values when confronted with empty strings. */
503 if (*yn == '\0')
504 return *xn != '\0';
505 if (*xn == '\0')
506 return -1;
507
508 return strcoll (xn, yn);
509#else
510 return strcmp (xn, yn);
511#endif
252b5132
RH
512}
513
514static int
2da42df6 515non_numeric_reverse (const void *x, const void *y)
252b5132
RH
516{
517 return - non_numeric_forward (x, y);
518}
519
382c1116
NC
520static int
521numeric_forward (const void *P_x, const void *P_y)
522{
523 asymbol *x, *y;
524 asection *xs, *ys;
525
526 x = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_x, sort_x);
527 y = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_y, sort_y);
528 if (x == NULL || y == NULL)
529 bfd_fatal (bfd_get_filename (sort_bfd));
530
531 xs = bfd_get_section (x);
532 ys = bfd_get_section (y);
533
534 if (bfd_is_und_section (xs))
535 {
536 if (! bfd_is_und_section (ys))
537 return -1;
538 }
539 else if (bfd_is_und_section (ys))
540 return 1;
541 else if (valueof (x) != valueof (y))
542 return valueof (x) < valueof (y) ? -1 : 1;
543
544 return non_numeric_forward (P_x, P_y);
545}
546
547static int
548numeric_reverse (const void *x, const void *y)
549{
550 return - numeric_forward (x, y);
551}
552
2da42df6 553static int (*(sorters[2][2])) (const void *, const void *) =
252b5132
RH
554{
555 { non_numeric_forward, non_numeric_reverse },
556 { numeric_forward, numeric_reverse }
557};
558
559/* This sort routine is used by sort_symbols_by_size. It is similar
560 to numeric_forward, but when symbols have the same value it sorts
561 by section VMA. This simplifies the sort_symbols_by_size code
562 which handles symbols at the end of sections. Also, this routine
563 tries to sort file names before other symbols with the same value.
564 That will make the file name have a zero size, which will make
565 sort_symbols_by_size choose the non file name symbol, leading to
566 more meaningful output. For similar reasons, this code sorts
567 gnu_compiled_* and gcc2_compiled before other symbols with the same
568 value. */
569
570static int
2da42df6 571size_forward1 (const void *P_x, const void *P_y)
252b5132
RH
572{
573 asymbol *x, *y;
574 asection *xs, *ys;
575 const char *xn, *yn;
576 size_t xnl, ynl;
577 int xf, yf;
578
579 x = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_x, sort_x);
580 y = bfd_minisymbol_to_symbol (sort_bfd, sort_dynamic, P_y, sort_y);
581 if (x == NULL || y == NULL)
582 bfd_fatal (bfd_get_filename (sort_bfd));
583
584 xs = bfd_get_section (x);
585 ys = bfd_get_section (y);
586
587 if (bfd_is_und_section (xs))
588 abort ();
589 if (bfd_is_und_section (ys))
590 abort ();
591
592 if (valueof (x) != valueof (y))
593 return valueof (x) < valueof (y) ? -1 : 1;
594
595 if (xs->vma != ys->vma)
596 return xs->vma < ys->vma ? -1 : 1;
597
598 xn = bfd_asymbol_name (x);
599 yn = bfd_asymbol_name (y);
600 xnl = strlen (xn);
601 ynl = strlen (yn);
602
603 /* The symbols gnu_compiled and gcc2_compiled convey even less
604 information than the file name, so sort them out first. */
605
606 xf = (strstr (xn, "gnu_compiled") != NULL
607 || strstr (xn, "gcc2_compiled") != NULL);
608 yf = (strstr (yn, "gnu_compiled") != NULL
609 || strstr (yn, "gcc2_compiled") != NULL);
610
611 if (xf && ! yf)
612 return -1;
613 if (! xf && yf)
614 return 1;
615
616 /* We use a heuristic for the file name. It may not work on non
617 Unix systems, but it doesn't really matter; the only difference
618 is precisely which symbol names get printed. */
619
620#define file_symbol(s, sn, snl) \
621 (((s)->flags & BSF_FILE) != 0 \
622 || ((sn)[(snl) - 2] == '.' \
623 && ((sn)[(snl) - 1] == 'o' \
624 || (sn)[(snl) - 1] == 'a')))
625
626 xf = file_symbol (x, xn, xnl);
627 yf = file_symbol (y, yn, ynl);
628
629 if (xf && ! yf)
630 return -1;
631 if (! xf && yf)
632 return 1;
633
634 return non_numeric_forward (P_x, P_y);
635}
636
637/* This sort routine is used by sort_symbols_by_size. It is sorting
638 an array of size_sym structures into size order. */
639
640static int
2da42df6 641size_forward2 (const void *P_x, const void *P_y)
252b5132
RH
642{
643 const struct size_sym *x = (const struct size_sym *) P_x;
644 const struct size_sym *y = (const struct size_sym *) P_y;
645
646 if (x->size < y->size)
647 return reverse_sort ? 1 : -1;
648 else if (x->size > y->size)
649 return reverse_sort ? -1 : 1;
650 else
651 return sorters[0][reverse_sort] (x->minisym, y->minisym);
652}
653
6ab6b380
NC
654/* Sort the symbols by size. ELF provides a size but for other formats
655 we have to make a guess by assuming that the difference between the
656 address of a symbol and the address of the next higher symbol is the
657 size. */
252b5132
RH
658
659static long
2da42df6
AJ
660sort_symbols_by_size (bfd *abfd, bfd_boolean dynamic, void *minisyms,
661 long symcount, unsigned int size,
662 struct size_sym **symsizesp)
252b5132
RH
663{
664 struct size_sym *symsizes;
665 bfd_byte *from, *fromend;
666 asymbol *sym = NULL;
667 asymbol *store_sym, *store_next;
668
669 qsort (minisyms, symcount, size, size_forward1);
670
671 /* We are going to return a special set of symbols and sizes to
672 print. */
382c1116 673 symsizes = xmalloc (symcount * sizeof (struct size_sym));
252b5132
RH
674 *symsizesp = symsizes;
675
676 /* Note that filter_symbols has already removed all absolute and
677 undefined symbols. Here we remove all symbols whose size winds
678 up as zero. */
252b5132
RH
679 from = (bfd_byte *) minisyms;
680 fromend = from + symcount * size;
681
682 store_sym = sort_x;
683 store_next = sort_y;
684
685 if (from < fromend)
686 {
2da42df6 687 sym = bfd_minisymbol_to_symbol (abfd, dynamic, (const void *) from,
252b5132
RH
688 store_sym);
689 if (sym == NULL)
690 bfd_fatal (bfd_get_filename (abfd));
691 }
692
693 for (; from < fromend; from += size)
694 {
695 asymbol *next;
696 asection *sec;
697 bfd_vma sz;
698 asymbol *temp;
699
700 if (from + size < fromend)
701 {
702 next = bfd_minisymbol_to_symbol (abfd,
703 dynamic,
2da42df6 704 (const void *) (from + size),
252b5132
RH
705 store_next);
706 if (next == NULL)
707 bfd_fatal (bfd_get_filename (abfd));
708 }
709 else
710 next = NULL;
711
712 sec = bfd_get_section (sym);
713
9710509e 714 if (bfd_get_flavour (abfd) == bfd_target_elf_flavour)
6ab6b380
NC
715 sz = ((elf_symbol_type *) sym)->internal_elf_sym.st_size;
716 else if (bfd_is_com_section (sec))
252b5132
RH
717 sz = sym->value;
718 else
719 {
720 if (from + size < fromend
721 && sec == bfd_get_section (next))
722 sz = valueof (next) - valueof (sym);
723 else
724 sz = (bfd_get_section_vma (abfd, sec)
725 + bfd_section_size (abfd, sec)
726 - valueof (sym));
727 }
728
729 if (sz != 0)
730 {
2da42df6 731 symsizes->minisym = (const void *) from;
252b5132
RH
732 symsizes->size = sz;
733 ++symsizes;
734 }
735
736 sym = next;
737
738 temp = store_sym;
739 store_sym = store_next;
740 store_next = temp;
741 }
742
743 symcount = symsizes - *symsizesp;
744
745 /* We must now sort again by size. */
2da42df6 746 qsort ((void *) *symsizesp, symcount, sizeof (struct size_sym), size_forward2);
252b5132
RH
747
748 return symcount;
749}
382c1116
NC
750
751/* This function is used to get the relocs for a particular section.
752 It is called via bfd_map_over_sections. */
252b5132
RH
753
754static void
382c1116 755get_relocs (bfd *abfd, asection *sec, void *dataarg)
252b5132 756{
382c1116 757 struct get_relocs_info *data = (struct get_relocs_info *) dataarg;
252b5132 758
382c1116
NC
759 *data->secs = sec;
760
761 if ((sec->flags & SEC_RELOC) == 0)
252b5132 762 {
382c1116
NC
763 *data->relocs = NULL;
764 *data->relcount = 0;
252b5132 765 }
382c1116
NC
766 else
767 {
768 long relsize;
252b5132 769
382c1116
NC
770 relsize = bfd_get_reloc_upper_bound (abfd, sec);
771 if (relsize < 0)
772 bfd_fatal (bfd_get_filename (abfd));
252b5132 773
382c1116
NC
774 *data->relocs = xmalloc (relsize);
775 *data->relcount = bfd_canonicalize_reloc (abfd, sec, *data->relocs,
776 data->syms);
777 if (*data->relcount < 0)
778 bfd_fatal (bfd_get_filename (abfd));
68a4c073 779 }
252b5132 780
382c1116
NC
781 ++data->secs;
782 ++data->relocs;
783 ++data->relcount;
784}
785
786/* Print a single symbol. */
787
788static void
789print_symbol (bfd *abfd, asymbol *sym, bfd_vma ssize, bfd *archive_bfd)
790{
791 symbol_info syminfo;
792 struct extended_symbol_info info;
793
794 PROGRESS (1);
795
796 format->print_symbol_filename (archive_bfd, abfd);
797
798 bfd_get_symbol_info (abfd, sym, &syminfo);
799 info.sinfo = &syminfo;
800 info.ssize = ssize;
801 if (bfd_get_flavour (abfd) == bfd_target_elf_flavour)
802 info.elfinfo = (elf_symbol_type *) sym;
803 else
804 info.elfinfo = NULL;
805 format->print_symbol_info (&info, abfd);
806
807 if (line_numbers)
0873df2a 808 {
382c1116
NC
809 static asymbol **syms;
810 static long symcount;
811 const char *filename, *functionname;
812 unsigned int lineno;
0873df2a 813
382c1116
NC
814 /* We need to get the canonical symbols in order to call
815 bfd_find_nearest_line. This is inefficient, but, then, you
816 don't have to use --line-numbers. */
817 if (abfd != lineno_cache_bfd && syms != NULL)
0873df2a 818 {
382c1116
NC
819 free (syms);
820 syms = NULL;
0873df2a 821 }
382c1116 822 if (syms == NULL)
0873df2a 823 {
382c1116
NC
824 long symsize;
825
826 symsize = bfd_get_symtab_upper_bound (abfd);
827 if (symsize < 0)
828 bfd_fatal (bfd_get_filename (abfd));
829 syms = xmalloc (symsize);
830 symcount = bfd_canonicalize_symtab (abfd, syms);
831 if (symcount < 0)
832 bfd_fatal (bfd_get_filename (abfd));
833 lineno_cache_bfd = abfd;
0873df2a 834 }
0873df2a 835
382c1116
NC
836 if (bfd_is_und_section (bfd_get_section (sym)))
837 {
838 static asection **secs;
839 static arelent ***relocs;
840 static long *relcount;
841 static unsigned int seccount;
842 unsigned int i;
843 const char *symname;
0873df2a 844
382c1116
NC
845 /* For an undefined symbol, we try to find a reloc for the
846 symbol, and print the line number of the reloc. */
847 if (abfd != lineno_cache_rel_bfd && relocs != NULL)
848 {
849 for (i = 0; i < seccount; i++)
850 if (relocs[i] != NULL)
851 free (relocs[i]);
852 free (secs);
853 free (relocs);
854 free (relcount);
855 secs = NULL;
856 relocs = NULL;
857 relcount = NULL;
858 }
252b5132 859
382c1116
NC
860 if (relocs == NULL)
861 {
862 struct get_relocs_info info;
252b5132 863
382c1116 864 seccount = bfd_count_sections (abfd);
252b5132 865
382c1116
NC
866 secs = xmalloc (seccount * sizeof *secs);
867 relocs = xmalloc (seccount * sizeof *relocs);
868 relcount = xmalloc (seccount * sizeof *relcount);
252b5132 869
382c1116
NC
870 info.secs = secs;
871 info.relocs = relocs;
872 info.relcount = relcount;
873 info.syms = syms;
874 bfd_map_over_sections (abfd, get_relocs, (void *) &info);
875 lineno_cache_rel_bfd = abfd;
876 }
252b5132 877
382c1116
NC
878 symname = bfd_asymbol_name (sym);
879 for (i = 0; i < seccount; i++)
880 {
881 long j;
882
883 for (j = 0; j < relcount[i]; j++)
884 {
885 arelent *r;
886
887 r = relocs[i][j];
888 if (r->sym_ptr_ptr != NULL
889 && (*r->sym_ptr_ptr)->section == sym->section
890 && (*r->sym_ptr_ptr)->value == sym->value
891 && strcmp (symname,
892 bfd_asymbol_name (*r->sym_ptr_ptr)) == 0
893 && bfd_find_nearest_line (abfd, secs[i], syms,
894 r->address, &filename,
895 &functionname, &lineno)
896 && filename != NULL)
897 {
898 /* We only print the first one we find. */
899 printf ("\t%s:%u", filename, lineno);
900 i = seccount;
901 break;
902 }
903 }
904 }
905 }
906 else if (bfd_get_section (sym)->owner == abfd)
907 {
5420f73d
L
908 if ((bfd_find_line (abfd, syms, sym, &filename, &lineno)
909 || bfd_find_nearest_line (abfd, bfd_get_section (sym),
910 syms, sym->value, &filename,
911 &functionname, &lineno))
382c1116
NC
912 && filename != NULL
913 && lineno != 0)
914 printf ("\t%s:%u", filename, lineno);
915 }
916 }
917
918 putchar ('\n');
252b5132
RH
919}
920\f
382c1116 921/* Print the symbols when sorting by size. */
252b5132 922
382c1116
NC
923static void
924print_size_symbols (bfd *abfd, bfd_boolean dynamic,
925 struct size_sym *symsizes, long symcount,
926 bfd *archive_bfd)
252b5132 927{
252b5132 928 asymbol *store;
382c1116 929 struct size_sym *from, *fromend;
252b5132
RH
930
931 store = bfd_make_empty_symbol (abfd);
932 if (store == NULL)
933 bfd_fatal (bfd_get_filename (abfd));
934
382c1116
NC
935 from = symsizes;
936 fromend = from + symcount;
937 for (; from < fromend; from++)
252b5132 938 {
252b5132 939 asymbol *sym;
382c1116 940 bfd_vma ssize;
252b5132 941
382c1116 942 sym = bfd_minisymbol_to_symbol (abfd, dynamic, from->minisym, store);
252b5132
RH
943 if (sym == NULL)
944 bfd_fatal (bfd_get_filename (abfd));
945
382c1116
NC
946 /* For elf we have already computed the correct symbol size. */
947 if (bfd_get_flavour (abfd) == bfd_target_elf_flavour)
948 ssize = from->size;
252b5132 949 else
382c1116 950 ssize = from->size - bfd_section_vma (abfd, bfd_get_section (sym));
eb3f2f5c 951
382c1116 952 print_symbol (abfd, sym, ssize, archive_bfd);
252b5132 953 }
252b5132
RH
954}
955
382c1116 956\f
252b5132
RH
957/* Print the symbols. If ARCHIVE_BFD is non-NULL, it is the archive
958 containing ABFD. */
959
960static void
2da42df6
AJ
961print_symbols (bfd *abfd, bfd_boolean dynamic, void *minisyms, long symcount,
962 unsigned int size, bfd *archive_bfd)
252b5132
RH
963{
964 asymbol *store;
965 bfd_byte *from, *fromend;
966
967 store = bfd_make_empty_symbol (abfd);
968 if (store == NULL)
969 bfd_fatal (bfd_get_filename (abfd));
970
971 from = (bfd_byte *) minisyms;
972 fromend = from + symcount * size;
973 for (; from < fromend; from += size)
974 {
975 asymbol *sym;
976
977 sym = bfd_minisymbol_to_symbol (abfd, dynamic, from, store);
978 if (sym == NULL)
979 bfd_fatal (bfd_get_filename (abfd));
980
33f5f537 981 print_symbol (abfd, sym, (bfd_vma) 0, archive_bfd);
252b5132
RH
982 }
983}
984
382c1116 985/* If ARCHIVE_BFD is non-NULL, it is the archive containing ABFD. */
252b5132 986
0af11b59 987static void
382c1116 988display_rel_file (bfd *abfd, bfd *archive_bfd)
252b5132 989{
382c1116
NC
990 long symcount;
991 void *minisyms;
992 unsigned int size;
993 struct size_sym *symsizes;
252b5132 994
382c1116
NC
995 if (! dynamic)
996 {
997 if (!(bfd_get_file_flags (abfd) & HAS_SYMS))
998 {
999 non_fatal (_("%s: no symbols"), bfd_get_filename (abfd));
1000 return;
1001 }
1002 }
1003
1004 symcount = bfd_read_minisymbols (abfd, dynamic, &minisyms, &size);
1005 if (symcount < 0)
252b5132
RH
1006 bfd_fatal (bfd_get_filename (abfd));
1007
382c1116 1008 if (symcount == 0)
252b5132 1009 {
382c1116
NC
1010 non_fatal (_("%s: no symbols"), bfd_get_filename (abfd));
1011 return;
1012 }
252b5132 1013
382c1116
NC
1014 if (show_synthetic && size == sizeof (asymbol *))
1015 {
1016 asymbol *synthsyms;
1017 long synth_count;
1018 asymbol **static_syms = NULL;
1019 asymbol **dyn_syms = NULL;
1020 long static_count = 0;
1021 long dyn_count = 0;
252b5132 1022
382c1116
NC
1023 if (dynamic)
1024 {
1025 dyn_count = symcount;
1026 dyn_syms = minisyms;
1027 }
977f7911 1028 else
382c1116 1029 {
8615f3f2
AM
1030 long storage = bfd_get_dynamic_symtab_upper_bound (abfd);
1031
382c1116
NC
1032 static_count = symcount;
1033 static_syms = minisyms;
8615f3f2
AM
1034
1035 if (storage > 0)
1036 {
1037 dyn_syms = xmalloc (storage);
1038 dyn_count = bfd_canonicalize_dynamic_symtab (abfd, dyn_syms);
1039 if (dyn_count < 0)
1040 bfd_fatal (bfd_get_filename (abfd));
1041 }
382c1116
NC
1042 }
1043 synth_count = bfd_get_synthetic_symtab (abfd, static_count, static_syms,
1044 dyn_count, dyn_syms, &synthsyms);
1045 if (synth_count > 0)
1046 {
1047 asymbol **symp;
1048 void *new_mini;
1049 long i;
977f7911 1050
382c1116
NC
1051 new_mini = xmalloc ((symcount + synth_count + 1) * sizeof (*symp));
1052 symp = new_mini;
1053 memcpy (symp, minisyms, symcount * sizeof (*symp));
1054 symp += symcount;
1055 for (i = 0; i < synth_count; i++)
1056 *symp++ = synthsyms + i;
1057 *symp = 0;
1058 minisyms = new_mini;
1059 symcount += synth_count;
1060 }
252b5132 1061 }
252b5132 1062
382c1116
NC
1063 /* Discard the symbols we don't want to print.
1064 It's OK to do this in place; we'll free the storage anyway
1065 (after printing). */
252b5132 1066
382c1116 1067 symcount = filter_symbols (abfd, dynamic, minisyms, symcount, size);
2da42df6 1068
382c1116
NC
1069 symsizes = NULL;
1070 if (! no_sort)
1071 {
1072 sort_bfd = abfd;
1073 sort_dynamic = dynamic;
1074 sort_x = bfd_make_empty_symbol (abfd);
1075 sort_y = bfd_make_empty_symbol (abfd);
1076 if (sort_x == NULL || sort_y == NULL)
1077 bfd_fatal (bfd_get_filename (abfd));
252b5132 1078
382c1116
NC
1079 if (! sort_by_size)
1080 qsort (minisyms, symcount, size,
1081 sorters[sort_numerically][reverse_sort]);
1082 else
1083 symcount = sort_symbols_by_size (abfd, dynamic, minisyms, symcount,
1084 size, &symsizes);
1085 }
252b5132 1086
382c1116
NC
1087 if (! sort_by_size)
1088 print_symbols (abfd, dynamic, minisyms, symcount, size, archive_bfd);
252b5132 1089 else
382c1116 1090 print_size_symbols (abfd, dynamic, symsizes, symcount, archive_bfd);
252b5132 1091
382c1116
NC
1092 free (minisyms);
1093}
1094
1095static void
1096display_archive (bfd *file)
1097{
1098 bfd *arfile = NULL;
1099 bfd *last_arfile = NULL;
1100 char **matching;
1101
1102 format->print_archive_filename (bfd_get_filename (file));
1103
1104 if (print_armap)
1105 print_symdef_entry (file);
1106
1107 for (;;)
252b5132 1108 {
382c1116 1109 PROGRESS (1);
252b5132 1110
382c1116
NC
1111 arfile = bfd_openr_next_archived_file (file, arfile);
1112
1113 if (arfile == NULL)
252b5132 1114 {
382c1116
NC
1115 if (bfd_get_error () != bfd_error_no_more_archived_files)
1116 bfd_fatal (bfd_get_filename (file));
1117 break;
252b5132 1118 }
382c1116
NC
1119
1120 if (bfd_check_format_matches (arfile, bfd_object, &matching))
252b5132 1121 {
382c1116 1122 char buf[30];
252b5132 1123
382c1116
NC
1124 bfd_sprintf_vma (arfile, buf, (bfd_vma) -1);
1125 print_width = strlen (buf);
1126 format->print_archive_member (bfd_get_filename (file),
1127 bfd_get_filename (arfile));
1128 display_rel_file (arfile, file);
252b5132 1129 }
382c1116 1130 else
252b5132 1131 {
382c1116
NC
1132 bfd_nonfatal (bfd_get_filename (arfile));
1133 if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
252b5132 1134 {
382c1116
NC
1135 list_matching_formats (matching);
1136 free (matching);
252b5132 1137 }
382c1116 1138 }
252b5132 1139
382c1116
NC
1140 if (last_arfile != NULL)
1141 {
1142 bfd_close (last_arfile);
1143 lineno_cache_bfd = NULL;
1144 lineno_cache_rel_bfd = NULL;
1145 }
1146 last_arfile = arfile;
1147 }
252b5132 1148
382c1116
NC
1149 if (last_arfile != NULL)
1150 {
1151 bfd_close (last_arfile);
1152 lineno_cache_bfd = NULL;
1153 lineno_cache_rel_bfd = NULL;
1154 }
1155}
252b5132 1156
382c1116
NC
1157static bfd_boolean
1158display_file (char *filename)
1159{
1160 bfd_boolean retval = TRUE;
1161 bfd *file;
1162 char **matching;
252b5132 1163
382c1116
NC
1164 if (get_file_size (filename) < 1)
1165 return FALSE;
252b5132 1166
382c1116
NC
1167 file = bfd_openr (filename, target);
1168 if (file == NULL)
1169 {
1170 bfd_nonfatal (filename);
1171 return FALSE;
1172 }
252b5132 1173
382c1116
NC
1174 if (bfd_check_format (file, bfd_archive))
1175 {
1176 display_archive (file);
1177 }
1178 else if (bfd_check_format_matches (file, bfd_object, &matching))
1179 {
1180 char buf[30];
1181
1182 bfd_sprintf_vma (file, buf, (bfd_vma) -1);
1183 print_width = strlen (buf);
1184 format->print_object_filename (filename);
1185 display_rel_file (file, NULL);
1186 }
1187 else
1188 {
1189 bfd_nonfatal (filename);
1190 if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
252b5132 1191 {
382c1116
NC
1192 list_matching_formats (matching);
1193 free (matching);
252b5132 1194 }
382c1116 1195 retval = FALSE;
252b5132
RH
1196 }
1197
382c1116
NC
1198 if (!bfd_close (file))
1199 bfd_fatal (filename);
1200
1201 lineno_cache_bfd = NULL;
1202 lineno_cache_rel_bfd = NULL;
1203
1204 return retval;
252b5132
RH
1205}
1206\f
1207/* The following 3 groups of functions are called unconditionally,
1208 once at the start of processing each file of the appropriate type.
1209 They should check `filename_per_file' and `filename_per_symbol',
1210 as appropriate for their output format, to determine whether to
1211 print anything. */
1212\f
1213/* Print the name of an object file given on the command line. */
1214
1215static void
2da42df6 1216print_object_filename_bsd (char *filename)
252b5132
RH
1217{
1218 if (filename_per_file && !filename_per_symbol)
1219 printf ("\n%s:\n", filename);
1220}
1221
1222static void
2da42df6 1223print_object_filename_sysv (char *filename)
252b5132
RH
1224{
1225 if (undefined_only)
1226 printf (_("\n\nUndefined symbols from %s:\n\n"), filename);
1227 else
1228 printf (_("\n\nSymbols from %s:\n\n"), filename);
33f5f537
L
1229 if (print_width == 8)
1230 printf (_("\
1231Name Value Class Type Size Line Section\n\n"));
1232 else
1233 printf (_("\
1234Name Value Class Type Size Line Section\n\n"));
252b5132
RH
1235}
1236
1237static void
2da42df6 1238print_object_filename_posix (char *filename)
252b5132
RH
1239{
1240 if (filename_per_file && !filename_per_symbol)
1241 printf ("%s:\n", filename);
1242}
1243\f
1244/* Print the name of an archive file given on the command line. */
1245
1246static void
2da42df6 1247print_archive_filename_bsd (char *filename)
252b5132
RH
1248{
1249 if (filename_per_file)
1250 printf ("\n%s:\n", filename);
1251}
1252
1253static void
2da42df6 1254print_archive_filename_sysv (char *filename ATTRIBUTE_UNUSED)
252b5132
RH
1255{
1256}
1257
1258static void
2da42df6 1259print_archive_filename_posix (char *filename ATTRIBUTE_UNUSED)
252b5132
RH
1260{
1261}
1262\f
1263/* Print the name of an archive member file. */
1264
1265static void
2da42df6
AJ
1266print_archive_member_bsd (char *archive ATTRIBUTE_UNUSED,
1267 const char *filename)
252b5132
RH
1268{
1269 if (!filename_per_symbol)
1270 printf ("\n%s:\n", filename);
1271}
1272
1273static void
2da42df6 1274print_archive_member_sysv (char *archive, const char *filename)
252b5132
RH
1275{
1276 if (undefined_only)
1277 printf (_("\n\nUndefined symbols from %s[%s]:\n\n"), archive, filename);
1278 else
1279 printf (_("\n\nSymbols from %s[%s]:\n\n"), archive, filename);
33f5f537
L
1280 if (print_width == 8)
1281 printf (_("\
1282Name Value Class Type Size Line Section\n\n"));
1283 else
1284 printf (_("\
1285Name Value Class Type Size Line Section\n\n"));
252b5132
RH
1286}
1287
1288static void
2da42df6 1289print_archive_member_posix (char *archive, const char *filename)
252b5132
RH
1290{
1291 if (!filename_per_symbol)
1292 printf ("%s[%s]:\n", archive, filename);
1293}
1294\f
1295/* Print the name of the file (and archive, if there is one)
1296 containing a symbol. */
1297
1298static void
2da42df6 1299print_symbol_filename_bsd (bfd *archive_bfd, bfd *abfd)
252b5132
RH
1300{
1301 if (filename_per_symbol)
1302 {
1303 if (archive_bfd)
1304 printf ("%s:", bfd_get_filename (archive_bfd));
1305 printf ("%s:", bfd_get_filename (abfd));
1306 }
1307}
1308
1309static void
2da42df6 1310print_symbol_filename_sysv (bfd *archive_bfd, bfd *abfd)
252b5132
RH
1311{
1312 if (filename_per_symbol)
1313 {
1314 if (archive_bfd)
1315 printf ("%s:", bfd_get_filename (archive_bfd));
1316 printf ("%s:", bfd_get_filename (abfd));
1317 }
1318}
1319
1320static void
2da42df6 1321print_symbol_filename_posix (bfd *archive_bfd, bfd *abfd)
252b5132
RH
1322{
1323 if (filename_per_symbol)
1324 {
1325 if (archive_bfd)
1326 printf ("%s[%s]: ", bfd_get_filename (archive_bfd),
1327 bfd_get_filename (abfd));
1328 else
1329 printf ("%s: ", bfd_get_filename (abfd));
1330 }
1331}
1332\f
1333/* Print a symbol value. */
1334
1335static void
2da42df6 1336print_value (bfd *abfd ATTRIBUTE_UNUSED, bfd_vma val)
252b5132
RH
1337{
1338#if ! defined (BFD64) || BFD_HOST_64BIT_LONG
1339 printf (value_format, val);
1340#else
1341 /* We have a 64 bit value to print, but the host is only 32 bit. */
1342 if (print_radix == 16)
d8180c76 1343 bfd_fprintf_vma (abfd, stdout, val);
252b5132
RH
1344 else
1345 {
1346 char buf[30];
1347 char *s;
1348
1349 s = buf + sizeof buf;
1350 *--s = '\0';
1351 while (val > 0)
1352 {
1353 *--s = (val % print_radix) + '0';
1354 val /= print_radix;
1355 }
1356 while ((buf + sizeof buf - 1) - s < 16)
1357 *--s = '0';
1358 printf ("%s", s);
1359 }
1360#endif
1361}
1362
1363/* Print a line of information about a symbol. */
1364
1365static void
2da42df6 1366print_symbol_info_bsd (struct extended_symbol_info *info, bfd *abfd)
252b5132 1367{
977f7911 1368 if (bfd_is_undefined_symclass (SYM_TYPE (info)))
252b5132 1369 {
62a5a82d
L
1370 if (print_width == 16)
1371 printf (" ");
21211521 1372 printf (" ");
252b5132
RH
1373 }
1374 else
977f7911 1375 {
06a30c77 1376 /* Normally we print the value of the symbol. If we are printing the
50c2245b 1377 size or sorting by size then we print its size, except for the
06a30c77
NC
1378 (weird) special case where both flags are defined, in which case we
1379 print both values. This conforms to documented behaviour. */
1380 if (sort_by_size && !print_size)
1381 print_value (abfd, SYM_SIZE (info));
1382 else
1383 print_value (abfd, SYM_VALUE (info));
977f7911 1384
72797995 1385 if (print_size && SYM_SIZE (info))
977f7911 1386 {
06a30c77 1387 printf (" ");
977f7911
NC
1388 print_value (abfd, SYM_SIZE (info));
1389 }
1390 }
1391
1392 printf (" %c", SYM_TYPE (info));
1393
1394 if (SYM_TYPE (info) == '-')
252b5132
RH
1395 {
1396 /* A stab. */
1397 printf (" ");
977f7911 1398 printf (other_format, SYM_STAB_OTHER (info));
252b5132 1399 printf (" ");
977f7911
NC
1400 printf (desc_format, SYM_STAB_DESC (info));
1401 printf (" %5s", SYM_STAB_NAME (info));
252b5132 1402 }
977f7911 1403 print_symname (" %s", SYM_NAME (info), abfd);
252b5132
RH
1404}
1405
1406static void
2da42df6 1407print_symbol_info_sysv (struct extended_symbol_info *info, bfd *abfd)
252b5132 1408{
977f7911
NC
1409 print_symname ("%-20s|", SYM_NAME (info), abfd);
1410
1411 if (bfd_is_undefined_symclass (SYM_TYPE (info)))
33f5f537
L
1412 {
1413 if (print_width == 8)
1414 printf (" ");
1415 else
1416 printf (" ");
1417 }
252b5132 1418 else
977f7911
NC
1419 print_value (abfd, SYM_VALUE (info));
1420
1421 printf ("| %c |", SYM_TYPE (info));
1422
1423 if (SYM_TYPE (info) == '-')
252b5132
RH
1424 {
1425 /* A stab. */
e3b83c8f
NC
1426 printf ("%18s| ", SYM_STAB_NAME (info)); /* (C) Type. */
1427 printf (desc_format, SYM_STAB_DESC (info)); /* Size. */
1428 printf ("| |"); /* Line, Section. */
252b5132
RH
1429 }
1430 else
9710509e 1431 {
977f7911 1432 /* Type, Size, Line, Section */
33f5f537
L
1433 if (info->elfinfo)
1434 printf ("%18s|",
1435 get_symbol_type (ELF_ST_TYPE (info->elfinfo->internal_elf_sym.st_info)));
1436 else
1437 printf (" |");
977f7911
NC
1438
1439 if (SYM_SIZE (info))
1440 print_value (abfd, SYM_SIZE (info));
1441 else
33f5f537
L
1442 {
1443 if (print_width == 8)
1444 printf (" ");
1445 else
1446 printf (" ");
1447 }
977f7911 1448
33f5f537
L
1449 if (info->elfinfo)
1450 printf("| |%s", info->elfinfo->symbol.section->name);
1451 else
1452 printf("| |");
977f7911 1453 }
252b5132
RH
1454}
1455
1456static void
2da42df6 1457print_symbol_info_posix (struct extended_symbol_info *info, bfd *abfd)
252b5132 1458{
977f7911
NC
1459 print_symname ("%s ", SYM_NAME (info), abfd);
1460 printf ("%c ", SYM_TYPE (info));
1461
1462 if (bfd_is_undefined_symclass (SYM_TYPE (info)))
252b5132
RH
1463 printf (" ");
1464 else
977f7911
NC
1465 {
1466 print_value (abfd, SYM_VALUE (info));
1467 printf (" ");
1468 if (SYM_SIZE (info))
1469 print_value (abfd, SYM_SIZE (info));
1470 }
252b5132
RH
1471}
1472\f
382c1116
NC
1473int
1474main (int argc, char **argv)
252b5132 1475{
382c1116
NC
1476 int c;
1477 int retval;
252b5132 1478
382c1116
NC
1479#if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
1480 setlocale (LC_MESSAGES, "");
1481#endif
1482#if defined (HAVE_SETLOCALE)
1483 setlocale (LC_CTYPE, "");
1484 setlocale (LC_COLLATE, "");
1485#endif
1486 bindtextdomain (PACKAGE, LOCALEDIR);
1487 textdomain (PACKAGE);
1488
1489 program_name = *argv;
1490 xmalloc_set_program_name (program_name);
1491
1492 START_PROGRESS (program_name, 0);
1493
1494 bfd_init ();
1495 set_default_bfd_target ();
1496
1497 while ((c = getopt_long (argc, argv, "aABCDef:gHhlnopPrSst:uvVvX:",
1498 long_options, (int *) 0)) != EOF)
252b5132 1499 {
382c1116 1500 switch (c)
252b5132 1501 {
382c1116
NC
1502 case 'a':
1503 print_debug_syms = 1;
1504 break;
1505 case 'A':
1506 case 'o':
1507 filename_per_symbol = 1;
1508 break;
1509 case 'B': /* For MIPS compatibility. */
1510 set_output_format ("bsd");
1511 break;
1512 case 'C':
1513 do_demangle = 1;
1514 if (optarg != NULL)
1515 {
1516 enum demangling_styles style;
1517
1518 style = cplus_demangle_name_to_style (optarg);
1519 if (style == unknown_demangling)
1520 fatal (_("unknown demangling style `%s'"),
1521 optarg);
1522
1523 cplus_demangle_set_style (style);
1524 }
1525 break;
1526 case 'D':
1527 dynamic = 1;
1528 break;
1529 case 'e':
1530 /* Ignored for HP/UX compatibility. */
1531 break;
1532 case 'f':
1533 set_output_format (optarg);
1534 break;
1535 case 'g':
1536 external_only = 1;
1537 break;
1538 case 'H':
1539 case 'h':
1540 usage (stdout, 0);
1541 case 'l':
1542 line_numbers = 1;
1543 break;
1544 case 'n':
1545 case 'v':
1546 sort_numerically = 1;
1547 break;
1548 case 'p':
1549 no_sort = 1;
1550 break;
1551 case 'P':
1552 set_output_format ("posix");
1553 break;
1554 case 'r':
1555 reverse_sort = 1;
1556 break;
1557 case 's':
1558 print_armap = 1;
1559 break;
1560 case 'S':
1561 print_size = 1;
1562 break;
1563 case 't':
1564 set_print_radix (optarg);
1565 break;
1566 case 'u':
1567 undefined_only = 1;
1568 break;
1569 case 'V':
1570 show_version = 1;
1571 break;
1572 case 'X':
1573 /* Ignored for (partial) AIX compatibility. On AIX, the
1574 argument has values 32, 64, or 32_64, and specifies that
1575 only 32-bit, only 64-bit, or both kinds of objects should
1576 be examined. The default is 32. So plain AIX nm on a
1577 library archive with both kinds of objects will ignore
1578 the 64-bit ones. For GNU nm, the default is and always
1579 has been -X 32_64, and other options are not supported. */
1580 if (strcmp (optarg, "32_64") != 0)
1581 fatal (_("Only -X 32_64 is supported"));
1582 break;
1583
1584 case OPTION_TARGET: /* --target */
1585 target = optarg;
1586 break;
1587
1588 case 0: /* A long option that just sets a flag. */
1589 break;
1590
1591 default:
1592 usage (stderr, 1);
252b5132
RH
1593 }
1594 }
252b5132 1595
382c1116
NC
1596 if (show_version)
1597 print_version ("nm");
252b5132 1598
382c1116 1599 if (sort_by_size && undefined_only)
252b5132 1600 {
382c1116
NC
1601 non_fatal (_("Using the --size-sort and --undefined-only options together"));
1602 non_fatal (_("will produce no output, since undefined symbols have no size."));
1603 return 0;
252b5132 1604 }
382c1116
NC
1605
1606 /* OK, all options now parsed. If no filename specified, do a.out. */
1607 if (optind == argc)
1608 return !display_file ("a.out");
1609
1610 retval = 0;
1611
1612 if (argc - optind > 1)
1613 filename_per_file = 1;
1614
1615 /* We were given several filenames to do. */
1616 while (optind < argc)
252b5132 1617 {
382c1116
NC
1618 PROGRESS (1);
1619 if (!display_file (argv[optind++]))
1620 retval++;
1621 }
252b5132 1622
382c1116 1623 END_PROGRESS (program_name);
252b5132 1624
382c1116
NC
1625#ifdef HAVE_SBRK
1626 if (show_stats)
1627 {
1628 char *lim = (char *) sbrk (0);
1629
1630 non_fatal (_("data size %ld"), (long) (lim - (char *) &environ));
252b5132 1631 }
382c1116 1632#endif
252b5132 1633
382c1116
NC
1634 exit (retval);
1635 return retval;
252b5132 1636}
This page took 0.324694 seconds and 4 git commands to generate.