/* * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands. * See the copyright notice in the ACK home directory, in the file "Copyright". */ /* $Id$ */ /* ** print symbol tables for ** ACK object files ** ** anm [-gopruns] [name ...] */ #include #include #include #include "out.h" #include "arch.h" #include "ranlib.h" int numsort_flg; int sectsort_flg; int undef_flg; int revsort_flg = 1; int globl_flg; int nosort_flg; int arch_flg; int prep_flg; int read_error; struct outhead hbuf; struct outsect sbuf; long off; long s_base[S_MAX]; /* for specially encoded bases */ char *filename; int narg; main(argc, argv) char **argv; { if (--argc>0 && argv[1][0]=='-' && argv[1][1]!=0) { argv++; while (*++*argv) switch (**argv) { case 'n': /* sort numerically */ numsort_flg++; continue; case 's': /* sort in section order */ sectsort_flg++; continue; case 'g': /* globl symbols only */ globl_flg++; continue; case 'u': /* undefined symbols only */ undef_flg++; continue; case 'r': /* sort in reverse order */ revsort_flg = -1; continue; case 'p': /* don't sort -- symbol table order */ nosort_flg++; continue; case 'o': /* prepend a name to each line */ prep_flg++; continue; default: /* oops */ fprintf(stderr, "anm: invalid argument -%c\n", *argv[0]); exit(1); } argc--; } if (argc == 0) { argc = 1; argv[1] = "a.out"; } narg = argc; while(argc--) { int fd; filename = *++argv; if ((fd = open(filename, 0)) < 0) { fprintf(stderr, "anm: cannot open %s\n", filename); continue; } process(fd); close(fd); } exit(0); } extern int rd_unsigned2(); process(fd) int fd; { unsigned int magic; long nextpos; struct ar_hdr archive_header; static char buf[sizeof(archive_header.ar_name)+1]; if (narg > 1) printf("\n%s:\n", filename); magic = rd_unsigned2(fd); switch(magic) { case O_MAGIC: lseek(fd, 0L, 0); do_file(fd); break; case ARMAG: case AALMAG: while (rd_arhdr(fd, &archive_header)) { nextpos = lseek(fd, 0L, 1) + archive_header.ar_size; if (nextpos & 1) nextpos++; strncpy(buf,archive_header.ar_name,sizeof(archive_header.ar_name)); filename = buf; if ( strcmp(filename, SYMDEF)) { printf("\n%s:\n", filename); do_file(fd); } lseek(fd, nextpos, 0); } break; default: fprintf(stderr, "anm: %s -- bad format\n", filename); break; } } do_file(fd) int fd; { struct outname *nbufp = NULL; struct outname nbuf; char *cbufp; long fi_to_co; long n; unsigned readcount; int i,j; int compare(); read_error = 0; rd_fdopen(fd); rd_ohead(&hbuf); if (read_error) { return; } if (BADMAGIC(hbuf)) { return; } n = hbuf.oh_nname; if (n == 0) { fprintf(stderr, "anm: %s -- no name list\n", filename); return; } if (hbuf.oh_nchar == 0) { fprintf(stderr, "anm: %s -- no names\n", filename); return; } if ((readcount = hbuf.oh_nchar) != hbuf.oh_nchar) { fprintf(stderr, "anm: string area too big in %s\n", filename); exit(2); } /* store special section bases ??? */ if (hbuf.oh_flags & HF_8086) { rd_sect(&sbuf, hbuf.oh_nsect); if (read_error) { return; } for (i=0; i>12) & 03777760; } } if ((cbufp = (char *)malloc(readcount)) == NULL) { fprintf(stderr, "anm: out of memory on %s\n", filename); exit(2); } rd_string(cbufp, hbuf.oh_nchar); if (read_error) { free(cbufp); return; } fi_to_co = (long) (cbufp - OFF_CHAR(hbuf)); i = 0; while (--n >= 0) { rd_name(&nbuf, 1); if (read_error) { break; } if (globl_flg && (nbuf.on_type&S_EXT)==0) continue; if (undef_flg && ((nbuf.on_type&S_TYP)!=S_UND || (nbuf.on_type&S_ETC)!=0)) continue; if (nbuf.on_foff == 0) nbuf.on_mptr = 0; else nbuf.on_mptr = (char *) (nbuf.on_foff + fi_to_co); /* adjust value for specially encoded bases */ if (hbuf.oh_flags & HF_8086) { if (((nbuf.on_type&S_ETC) == 0) || ((nbuf.on_type&S_ETC) == S_SCT)) { j = nbuf.on_type&S_TYP; if ((j>=S_MIN) && (j<=S_MAX)) nbuf.on_valu += s_base[j]; } } if (nbufp == NULL) nbufp = (struct outname *)malloc(sizeof(struct outname)); else nbufp = (struct outname *)realloc(nbufp, (i+1)*sizeof(struct outname)); if (nbufp == NULL) { fprintf(stderr, "anm: out of memory on %s\n", filename); exit(2); } nbufp[i++] = nbuf; } if (nbufp && nosort_flg==0) qsort(nbufp, i, sizeof(struct outname), compare); for (n=0; non_type&S_TYP) > (p2->on_type&S_TYP)) return(revsort_flg); if ((p1->on_type&S_TYP) < (p2->on_type&S_TYP)) return(-revsort_flg); } if (numsort_flg) { if (p1->on_valu > p2->on_valu) return(revsort_flg); if (p1->on_valu < p2->on_valu) return(-revsort_flg); } if (! p1->on_mptr) { if (! p2->on_mptr) return 0; return -revsort_flg; } if (! p2->on_mptr) return revsort_flg; i = strcmp(p1->on_mptr, p2->on_mptr); if (i > 0) return(revsort_flg); if (i < 0) return(-revsort_flg); return(0); } rd_fatal() { fprintf(stderr,"read error on %s\n", filename); read_error = 1; }