123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872 |
- /* Describe symbol table of a rel file.
- Copyright (C) 1986 Richard M. Stallman
- NO WARRANTY
- BECAUSE THIS PROGRAM IS LICENSED FREE OF CHARGE, WE PROVIDE ABSOLUTELY
- NO WARRANTY, TO THE EXTENT PERMITTED BY APPLICABLE STATE LAW. EXCEPT
- WHEN OTHERWISE STATED IN WRITING, FREE SOFTWARE FOUNDATION, INC,
- RICHARD M. STALLMAN AND/OR OTHER PARTIES PROVIDE THIS PROGRAM "AS IS"
- WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
- BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
- FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY
- AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE
- DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR
- CORRECTION.
- IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW WILL RICHARD M.
- STALLMAN, THE FREE SOFTWARE FOUNDATION, INC., AND/OR ANY OTHER PARTY
- WHO MAY MODIFY AND REDISTRIBUTE THIS PROGRAM AS PERMITTED BELOW, BE
- LIABLE TO YOU FOR DAMAGES, INCLUDING ANY LOST PROFITS, LOST MONIES, OR
- OTHER SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
- USE OR INABILITY TO USE (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR
- DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY THIRD PARTIES OR
- A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS) THIS
- PROGRAM, EVEN IF YOU HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH
- DAMAGES, OR FOR ANY CLAIM BY ANY OTHER PARTY.
- GENERAL PUBLIC LICENSE TO COPY
- 1. You may copy and distribute verbatim copies of this source file
- as you receive it, in any medium, provided that you conspicuously
- and appropriately publish on each copy a valid copyright notice
- "Copyright (C) 1986 Richard M. Stallman"; and include following the
- copyright notice a verbatim copy of the above disclaimer of warranty
- and of this License.
- 2. You may modify your copy or copies of this source file or
- any portion of it, and copy and distribute such modifications under
- the terms of Paragraph 1 above, provided that you also do the following:
- a) cause the modified files to carry prominent notices stating
- that you changed the files and the date of any change; and
- b) cause the whole of any work that you distribute or publish,
- that in whole or in part contains or is a derivative of this
- program or any part thereof, to be freely distributed
- and licensed to all third parties on terms identical to those
- contained in this License Agreement (except that you may choose
- to grant more extensive warranty protection to third parties,
- at your option).
- 3. You may copy and distribute this program or any portion of it in
- compiled, executable or object code form under the terms of Paragraphs
- 1 and 2 above provided that you do the following:
- a) cause each such copy to be accompanied by the
- corresponding machine-readable source code, which must
- be distributed under the terms of Paragraphs 1 and 2 above; or,
- b) cause each such copy to be accompanied by a
- written offer, with no time limit, to give any third party
- free (except for a nominal shipping charge) a machine readable
- copy of the corresponding source code, to be distributed
- under the terms of Paragraphs 1 and 2 above; or,
- c) in the case of a recipient of this program in compiled, executable
- or object code form (without the corresponding source code) you
- shall cause copies you distribute to be accompanied by a copy
- of the written offer of source code which you received along
- with the copy you received.
- 4. You may not copy, sublicense, distribute or transfer this program
- except as expressly provided under this License Agreement. Any attempt
- otherwise to copy, sublicense, distribute or transfer this program is void and
- your rights to use the program under this License agreement shall be
- automatically terminated. However, parties who have received computer
- software programs from you with this License Agreement will not have
- their licenses terminated so long as such parties remain in full compliance.
- In other words, you are welcome to use, share and improve this program.
- You are forbidden to forbid anyone else to use, share and improve
- what you give them. Help stamp out software-hoarding! */
- /* define this for running on the TI NU machine. */
- /* #define numachine 1 */
- #ifdef numachine
- #define O_RDONLY 0
- #endif
- #include <a.out.h>
- #include <stab.h>
- #include <stdio.h>
- #include <strings.h>
- #include <ar.h>
- #ifndef numachine
- #include <sys/file.h>
- #endif
- int number_of_files;
- /* current file's name */
- char *input_name;
- /* current member's name, or 0 if processing a non-library file. */
- char *input_member;
- /* offset within archive of the current member, if we are processing an archive. */
- int member_offset;
- /* command switches */
- int external_only; /* nonzero means print external symbols only. */
- int sort_numerically; /* sort in numerical order rather than alphabetic */
- int reverse_sort; /* sort in decreasing (alphabetical or numerical) order. */
- int no_sort; /* don't sort; print symbols in the order they appear in the file. */
- int undefined_only; /* print undefined symbols only. */
- int file_on_each_line; /* print file name on each line. */
- int debugger_syms; /* print the debugger-only symbols. */
- int print_symdefs; /* describe the __.SYMDEF data in any archive file specified. */
- /* The __.SYMDEF member of an archive has the following format:
- 1) A longword saying the size of the symdef data that follows
- 2) Zero or more struct symdef filling that many bytes
- 3) A longword saying how many bytes of strings follow
- 4) That many bytes of string data.
- */
- struct symdef
- {
- long stringoffset; /* Offset of this symbol's name in the string data */
- long offset; /* Offset in the archive of the header-data for the member that
- defines this symbol. */
- };
- void decode_switch ();
- int decode_arg ();
- void do_one_file (), do_one_rel_file (), do_symdef_member ();
- char *concat ();
- main(argc, argv)
- char **argv;
- int argc;
- {
- int i;
- number_of_files = 0;
- external_only = 0;
- sort_numerically = 0;
- reverse_sort = 0;
- no_sort = 0;
- undefined_only = 0;
- file_on_each_line = 0;
- debugger_syms = 0;
- print_symdefs = 0;
- /* First process the switches. */
- for (i = 1; i < argc; i++)
- {
- int code = decode_arg (argv[i]);
- if (code)
- {
- decode_switch (argv[i]);
- i += code - 1;
- }
- else
- number_of_files++;
- }
- if (!number_of_files)
- fatal ("no files specified", 0);
- /* Now scan again and print the files. */
- for (i = 1; i < argc; i++)
- {
- int code = decode_arg (argv[i]);
- if (code)
- i += code - 1;
- else
- do_one_file (argv[i]);
- }
- }
- /* analyze a command line argument.
- Returns 0 if the argument is a filename.
- Returns 1 if the argument is a switch complete in itself.
- Returns 2 if the argument is a switch which gobbles the following argument.
- (Except that `nm' doesn't have any switches that need arguments) */
- int
- decode_arg(arg)
- char *arg;
- {
- if (*arg != '-') return 0;
- return 1;
- }
- void
- decode_switch (swt)
- char *swt;
- {
- char *tail = swt + 1;
- char c;
- while (1)
- {
- switch ((c = *tail++))
- {
- case 0:
- return;
- case 'a':
- debugger_syms = 1;
- break;
- case 'g':
- external_only = 1;
- break;
- case 'n':
- sort_numerically = 1;
- break;
- case 'o':
- file_on_each_line = 1;
- break;
- case 'p':
- no_sort = 1;
- break;
- case 'r':
- reverse_sort = 1;
- break;
- case 's':
- print_symdefs = 1;
- break;
- case 'u':
- undefined_only = 1;
- break;
- default:
- error ("invalid command switch character -%c", c);
- }
- }
- }
- /* open the file specified by `name', and return a descriptor. */
- /* Print the filename of the current file on 'outfile' (a stdio stream). */
- print_file_name (outfile)
- FILE *outfile;
- {
- fprintf (outfile, "%s", input_name);
- if (input_member)
- fprintf (outfile, "(%s)", input_member);
- }
- /* process one input file */
- void scan_library ();
- void
- do_one_file (name)
- char *name;
- {
- int len, magicnum, desc;
- desc = open (name, O_RDONLY, 0);
- if (desc < 0)
- {
- perror_name (name);
- return;
- }
- input_name = name;
- input_member = 0;
- len = read (desc, &magicnum, sizeof magicnum);
- if (len != sizeof magicnum)
- error_with_file ("failure reading header of ");
- else if (!N_BADMAG(*((struct exec *)&magicnum)))
- do_one_rel_file (desc, 0);
- else
- {
- char armag[SARMAG];
- lseek (desc, 0, 0);
- if (SARMAG != read (desc, armag, SARMAG) || strncmp (armag, ARMAG, SARMAG))
- error_with_file ("malformed input file (not rel or archive) ");
- else
- scan_library (desc);
- }
- close (desc);
- }
- /* read in the archive data about one member.
- subfile_offset is the address within the archive of the start of that data.
- The value returned is the length of the member's contents, which does
- not include the archive data about the member.
- If there are no more valid members, zero is returned. */
- int
- decode_library_subfile (desc, subfile_offset, member_name_ptr)
- int desc;
- int subfile_offset;
- char **member_name_ptr;
- {
- int bytes_read;
- int namelen;
- int member_length;
- char *name;
- struct ar_hdr hdr1;
- lseek (desc, subfile_offset, 0);
- bytes_read = read (desc, &hdr1, sizeof hdr1);
- if (!bytes_read)
- ; /* end of archive */
- else if (sizeof hdr1 != bytes_read)
- error_with_file ("malformed library archive ");
- else if (sscanf (hdr1.ar_size, "%d", &member_length) != 1)
- error_with_file ("malformatted header of archive member in ");
- else
- {
- namelen = index (hdr1.ar_name, ' ') - hdr1.ar_name;
- name = (char *) xmalloc (namelen+1);
- strncpy (name, hdr1.ar_name, namelen);
- *(name + namelen) = 0;
- *member_name_ptr = name;
- return member_length;
- }
- return 0; /* tell caller to exit loop */
- }
- /* Scan a library and describe each member. */
- void
- scan_library (desc)
- int desc;
- {
- int this_subfile_offset = SARMAG;
- int member_length;
- if (!file_on_each_line)
- printf ("\n%s:\n", input_name);
-
- while (member_length = decode_library_subfile (desc, this_subfile_offset, &input_member))
- {
- /* describe every member except the ranlib data if any */
- if (strcmp (input_member, "__.SYMDEF"))
- do_one_rel_file (desc, this_subfile_offset + sizeof (struct ar_hdr));
- else if (print_symdefs)
- do_symdef_member (desc, this_subfile_offset + sizeof (struct ar_hdr), member_length);
- this_subfile_offset += ((member_length + sizeof (struct ar_hdr)) + 1) & -2;
- }
- }
- void print_symbols (), print_one_symbol ();
- void read_header ();
- int alphacompare (), valuecompare ();
- int filter_symbols ();
- void
- do_one_rel_file (desc, offset)
- int desc;
- int offset;
- {
- struct exec header; /* file header read in here */
- int string_size;
- struct nlist *symbols_and_strings;
- int symcount;
- int totalsize;
- char *strings;
- header.a_magic = 0;
- read_header (desc, &header, offset);
- if (N_BADMAG(header))
- {
- error_with_file ("bad magic number in ");
- return;
- }
- /* read the string-table-length out of the file */
- lseek (desc, N_STROFF(header) + offset, 0);
- if (sizeof string_size != read (desc, &string_size, sizeof string_size))
- {
- error_with_file ("bad string table in ");
- return;
- }
- /* number of symbol entries in the file */
- symcount = header.a_syms / sizeof (struct nlist);
- totalsize = string_size + header.a_syms;
- /* allocate space for symbol entries and string table */
- symbols_and_strings = (struct nlist *) xmalloc (totalsize);
- strings = (char *) symbols_and_strings + header.a_syms;
- /* read them both in all at once */
- lseek (desc, N_SYMOFF(header) + offset, 0);
- if (totalsize != read (desc, symbols_and_strings, totalsize))
- {
- error_with_file ("premature end of file in symbols/strings of ");
- return;
- }
- /* identify this file, if desired */
- if (!file_on_each_line && (number_of_files > 1 || input_member))
- printf ("\n%s:\n", input_member ? input_member : input_name);
- /* discard the symbols we don't want to print; compact the rest down */
- symcount = filter_symbols (symbols_and_strings, symcount, strings);
-
- /* modify each symbol entry to point directly at the symbol name.
- This is so the sort routine does not need to be passed
- the value of `strings' separately. */
- {
- struct nlist *p = symbols_and_strings;
- struct nlist *end = symbols_and_strings + symcount;
- for (; p < end; p++)
- if (p->n_un.n_strx)
- p->n_un.n_name = strings + p->n_un.n_strx;
- }
- /* sort the symbols if desired */
- if (!no_sort)
- qsort (symbols_and_strings, symcount, sizeof (struct nlist),
- sort_numerically ? valuecompare : alphacompare);
- /* print the symbols in the order they are now in. */
- print_symbols (symbols_and_strings, symcount);
- free (symbols_and_strings);
- }
- /* read a file's header */
- void
- read_header (desc, loc, offset)
- int desc;
- struct exec *loc;
- int offset;
- {
- int len;
- lseek (desc, offset, 0);
- len = read (desc, loc, sizeof (struct exec));
- if (len != sizeof (struct exec))
- error_with_file ("failure reading header of ");
- }
- /* Choose which symbol entries to print;
- compact them downward to get rid of the rest.
- Return the number of symbols to be printed. */
- int
- filter_symbols (syms, symcount, strings)
- struct nlist *syms;
- int symcount;
- char *strings;
- {
- struct nlist *from = syms, *to = syms;
- struct nlist *end = syms + symcount;
- while (from < end)
- {
- int keep = 0;
- /* undefined sym or common sym */
- if (from->n_type == N_EXT) keep = !undefined_only || !from->n_value;
- /* global defined sym */
- else if (from->n_type & N_EXT) keep = !undefined_only;
- /* debugger sym: normally don't print */
- else if (from->n_type & ~(N_TYPE | N_EXT)) keep = debugger_syms;
- /* local sym */
- else keep = !external_only && !undefined_only;
- if (keep)
- *to++ = *from;
- from++;
- }
- return to - syms;
- }
- /* comparison functions for sorting symbols */
- int
- alphacompare (sym1, sym2)
- struct nlist *sym1, *sym2;
- {
- if (reverse_sort)
- {
- if (!sym2)
- {
- if (sym1) return -1;
- else return 0;
- }
- if (!sym1) return 1;
- return strcmp (sym2->n_un.n_name, sym1->n_un.n_name);
- }
- else
- {
- if (!sym1)
- {
- if (sym2) return -1;
- else return 0;
- }
- if (!sym2) return 1;
- return strcmp (sym1->n_un.n_name, sym2->n_un.n_name);
- }
- }
- int
- valuecompare (sym1, sym2)
- struct nlist *sym1, *sym2;
- {
- if (reverse_sort)
- return sym2->n_value - sym1->n_value;
- else
- return sym1->n_value - sym2->n_value;
- }
- void
- print_symbols (syms, symcount)
- struct nlist *syms;
- int symcount;
- {
- int i;
- for (i = 0; i < symcount; i++)
- print_one_symbol (&syms[i]);
- }
- void
- print_one_symbol (sym)
- struct nlist *sym;
- {
- if (file_on_each_line)
- {
- print_file_name(stdout);
- printf (":");
- }
- if (undefined_only)
- {
- if (sym->n_type == N_EXT && !sym->n_value)
- printf ("%s\n", sym->n_un.n_name);
- return;
- }
- if (sym->n_type & ~N_EXT || sym->n_value)
- printf ("%08x ", sym->n_value);
- else printf (" ");
- switch (sym->n_type)
- {
- case N_EXT:
- if (sym->n_value) printf ("C");
- else printf ("U");
- break;
- case 0:
- if (sym->n_value) printf ("c");
- else printf ("u");
- break;
- case N_ABS | N_EXT:
- printf ("A");
- break;
- case N_ABS:
- printf ("a");
- break;
- case N_TEXT | N_EXT:
- printf ("T");
- break;
- case N_TEXT:
- printf ("t");
- break;
- case N_DATA | N_EXT:
- printf ("D");
- break;
- case N_DATA:
- printf ("d");
- break;
- case N_BSS | N_EXT:
- printf ("B");
- break;
- case N_BSS:
- printf ("b");
- break;
- default:
- {
- char *s;
- switch (sym->n_type)
- {
- case N_GSYM:
- s = "GSYM";
- break;
- case N_FNAME:
- s = "FNAME";
- break;
- case N_FUN:
- s = "FUN";
- break;
- case N_STSYM:
- s = "STSYM";
- break;
- case N_LCSYM:
- s = "LCSYM";
- break;
- case N_RSYM:
- s = "RSYM";
- break;
- case N_SLINE:
- s = "SLINE";
- break;
- case N_SSYM:
- s = "SSYM";
- break;
- case N_SO:
- s = "SO";
- break;
- case N_LSYM:
- s = "LSYM";
- break;
- case N_SOL:
- s = "SOL";
- break;
- case N_PSYM:
- s = "PSYM";
- break;
- case N_ENTRY:
- s = "ENTRY";
- break;
- case N_LBRAC:
- s = "LBRAC";
- break;
- case N_RBRAC:
- s = "RBRAC";
- break;
- case N_BCOMM:
- s = "BCOMM";
- break;
- case N_ECOMM:
- s = "ECOMM";
- break;
- case N_ECOML:
- s = "ECOML";
- break;
- case N_LENG:
- s = "LENG";
- break;
- default:
- s = "";
- }
- printf ("- %02x %04x %5s", sym->n_other, sym->n_desc, s);
- }
- }
- if (sym->n_un.n_name)
- printf (" %s\n", sym->n_un.n_name);
- else
- printf (" \n");
- }
- void
- do_symdef_member (desc, offset, member_length)
- int desc;
- int offset;
- int member_length;
- {
- int symdef_size;
- int nsymdefs;
- struct symdef *symdefs;
- int stringsize;
- char *strings;
- int i;
- char *member_name;
- int member_offset;
- /* read the string-table-length out of the file */
- lseek (desc, offset, 0);
- if (sizeof symdef_size != read (desc, &symdef_size, sizeof symdef_size))
- {
- error_with_file ("premature eof in ");
- return;
- }
- if (symdef_size < 0)
- {
- error_with_file ("invalid size value in ");
- return;
- }
- nsymdefs = symdef_size / sizeof (struct symdef);
- symdefs = (struct symdef *) alloca (symdef_size);
- if (symdef_size != read (desc, symdefs, symdef_size))
- {
- error_with_file ("premature eof in ");
- return;
- }
- if (stringsize < 0)
- {
- error_with_file ("invalid size value in ");
- return;
- }
- if (sizeof stringsize != read (desc, &stringsize, sizeof stringsize))
- {
- error_with_file ("premature eof in ");
- return;
- }
- strings = (char *) alloca (stringsize);
- if (stringsize != read (desc, strings, stringsize))
- {
- error_with_file ("premature eof in ");
- return;
- }
- if (stringsize + symdef_size + sizeof stringsize + sizeof symdef_size != member_length)
- {
- error_with_file ("size of data isn't what the data calls for in ");
- return;
- }
- if (!file_on_each_line && (number_of_files > 1 || input_member))
- printf ("\n%s:\n", input_member ? input_member : input_name);
-
- member_offset = -1;
- for (i = 0; i < nsymdefs; i++)
- {
- if (symdefs[i].stringoffset < 0 || symdefs[i].stringoffset >= stringsize)
- {
- error_with_file ("invalid entry in ");
- return;
- }
- if (member_offset != symdefs[i].offset)
- {
- member_offset = symdefs[i].offset;
- decode_library_subfile (desc, member_offset, &member_name);
- }
- if (file_on_each_line)
- {
- print_file_name(stdout);
- printf (":");
- }
- printf ("%s in %s\n", symdefs[i].stringoffset + strings, member_name);
- }
- }
- /* Report a fatal error. 'string' is a printf format string and 'arg' is one arg for it. */
- fatal (string, arg)
- char *string, *arg;
- {
- fprintf (stderr, "nm: ");
- fprintf (stderr, string, arg);
- fprintf (stderr, "\n");
- exit (1);
- }
- /* Report a nonfatal error. 'string' is a printf format string and 'arg' is one arg for it. */
- error (string, arg)
- char *string, *arg;
- {
- fprintf (stderr, "nm: ");
- fprintf (stderr, string, arg);
- fprintf (stderr, "\n");
- }
- /* Report a nonfatal error. 'string' is printed, followed by the current file name. */
- error_with_file (string)
- char *string;
- {
- fprintf (stderr, "nm: ");
- fprintf (stderr, string);
- print_file_name (stderr);
- fprintf (stderr, "\n");
- }
- /* Report a fatal error using the message for the last failed system call,
- followed by the string 'name'. */
- perror_name (name)
- char *name;
- {
- extern int errno, sys_nerr;
- extern char *sys_errlist[];
- char *s;
- if (errno < sys_nerr)
- s = concat ("", sys_errlist[errno], " for %s");
- else
- s = "cannot open %s";
- error (s, name);
- }
- /* Like malloc but get fatal error if memory is exhausted. */
- int
- xmalloc (size)
- int size;
- {
- int result = malloc (size);
- if (!result)
- fatal ("virtual memory exhausted", 0);
- return result;
- }
- /* Return a newly-allocated string whose contents concatenate those of s1, s2, s3. */
- char *
- concat (s1, s2, s3)
- char *s1, *s2, *s3;
- {
- int len1 = strlen (s1), len2 = strlen (s2), len3 = strlen (s3);
- char *result = (char *) xmalloc (len1 + len2 + len3 + 1);
- strcpy (result, s1);
- strcpy (result + len1, s2);
- strcpy (result + len1 + len2, s3);
- *(result + len1 + len2 + len3) = 0;
- return result;
- }
|