ref: bd41c3ee99a819ae92887538049cdef0ad482763
dir: /nm/coff32.c/
static char sccsid[] = "@(#) ./nm/coff.c"; #include <assert.h> #include <ctype.h> #include <errno.h> #include <stdint.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include "../inc/coff32/filehdr.h" #include "../inc/coff32/scnhdr.h" #include "../inc/coff32/syms.h" #include "../inc/scc.h" #include "nm.h" static int (*unpack)(unsigned char *, char *, ...); static long strtbl, symtbl, sectbl; static SCNHDR *sections; static struct symbol *syms; static size_t nsect, nsyms; static char typeof(SYMENT *ent) { SCNHDR *sec; int c; long flags; switch (ent->n_scnum) { case N_DEBUG: c = 'n'; break; case N_ABS: c = 'a'; break; case N_UNDEF: c = (ent->n_value != 0) ? 'C' : 'U'; break; default: if (ent->n_scnum > nsect) die("nm: incorrect section index"); sec = §ions[ent->n_scnum-1]; flags = sec->s_flags; if (flags & STYP_TEXT) c = 't'; else if (flags & STYP_DATA) c = 'd'; else if (flags & STYP_BSS) c = 'b'; else c = '?'; break; } if (ent->n_sclass == C_EXT) c = toupper(c); return c; } static char * getsname(char *fname, FILE *fp, SYMENT *ent) { int c; size_t len; char *s, *err; fpos_t pos; if (ent->n_zeroes != 0) { for (len = 0; len < E_SYMNMLEN && ent->n_name[len]; ++len) ; s = xmalloc(len+1); s[len] = '\0'; return memcpy(s, ent->n_name, len); } /* TODO: read the string table in memory before reading symbols */ fgetpos(fp, &pos); fseek(fp, strtbl, SEEK_SET); fseek(fp, ent->n_offset, SEEK_CUR); if (ferror(fp)) goto error; s = NULL; for (len = 1; (c = getc(fp)) != EOF; len++) { s = xrealloc(s, len); if ((s[len-1] = c) == '\0') break; } if (c == EOF) goto error; fsetpos(fp, &pos); return s; error: err = (!ferror(fp)) ? "EOF before reading strings" : strerror(errno); die("nm: %s: %s", fname, err); } static void getfsym(unsigned char *buff, SYMENT *ent) { int n; n = (*unpack)(buff, "'8lsscc", &ent->n_name, &ent->n_value, &ent->n_scnum, &ent->n_type, &ent->n_sclass, &ent->n_numaux); assert(n == SYMESZ); } static void getsymbol(char *fname, FILE *fp, unsigned char *buff, SYMENT *ent, struct symbol *sym) { char *nam; getfsym(buff, ent); nam = ent->n_name; if (nam[0] == 0 && nam[1] == 0 && nam[2] == 0 && nam[3] == 0) { long zero, offset; (*unpack)(nam, "ll", &zero, &offset); ent->n_zeroes = zero; ent->n_offset = offset; } sym->name = getsname(fname, fp, ent); sym->type = typeof(ent); sym->value = ent->n_value; sym->size = (sym->type == 'C') ? ent->n_value : 0; } static void getsyms(char *fname, char *member, FILE *fp, FILHDR *hdr) { size_t n, i; unsigned aux; unsigned char buff[SYMESZ]; SYMENT ent; if (hdr->f_nsyms > SIZE_MAX) die("nm: %s:Too many symbols\n", member); n = hdr->f_nsyms; syms = xcalloc(sizeof(*syms), n); if (fseek(fp, symtbl, SEEK_SET) == EOF) die("nm: %s:%s", fname, strerror(errno)); aux = nsyms = 0; for (i = 0; i < n; i++) { if (fread(buff, SYMESZ, 1, fp) != 1) break; if (aux > 0) { aux--; continue; } getsymbol(member, fp, buff, &ent, &syms[nsyms++]); aux = ent.n_numaux; } if (n != i) { char *err; err = (!ferror(fp)) ? "EOF before reading symbols" : strerror(errno); die("nm: %s: %s", fname, err); } } static void getfsec(unsigned char *buff, SCNHDR *sec) { int n; n = (*unpack)(buff, "'8llllllssl", sec->s_name, &sec->s_paddr, &sec->s_vaddr, &sec->s_size, &sec->s_scnptr, &sec->s_relptr, &sec->s_lnnoptr, &sec->s_nrelloc, &sec->s_nlnno, &sec->s_flags); assert(n == SCNHSZ); } static void getsects(char *fname, char *member, FILE *fp, FILHDR *hdr) { size_t i; char buff[SCNHSZ]; nsect = hdr->f_nscns; if (nsect == 0) return; if (nsect > SIZE_MAX) die("nm: %s:Too many sections\n", member); if (fseek(fp, sectbl, SEEK_SET) == EOF) die("nm: %s:%s", member, strerror(errno)); sections = xcalloc(sizeof(*sections), nsect); for (i = 0; i < nsect; i++) { if (fread(buff, SCNHSZ, 1, fp) != 1) break; getfsec(buff, §ions[i]); } if (i != nsect) { char *err; err = (!ferror(fp)) ? "EOF before reading sections" : strerror(errno); die("nm: %s: %s", fname, err); } } static void getfhdr(unsigned char *buff, FILHDR *hdr) { int n; n = (*unpack)(buff, "sslllss", &hdr->f_magic, &hdr->f_nscns, &hdr->f_timdat, &hdr->f_symptr, &hdr->f_nsyms, &hdr->f_opthdr, &hdr->f_flags); assert(n == FILHSZ); } static int nm(char *fname, char *member, FILE *fp) { unsigned char buff[FILHSZ]; FILHDR hdr; long pos = ftell(fp); if (fread(buff, FILHSZ, 1, fp) != 1) { if (!ferror(fp)) return 0; die("nm: %s: %s", fname, strerror(errno)); } getfhdr(buff, &hdr); if ((hdr.f_flags & F_SYMS) != 0 || hdr.f_nsyms == 0) { fprintf(stderr, "nm: %s: no symbols\n", member); return 1; } /* TODO: Check overflow */ strtbl = pos + hdr.f_symptr + hdr.f_nsyms* SYMESZ; symtbl = pos + hdr.f_symptr; sectbl = pos + FILHSZ + hdr.f_opthdr; getsects(fname, member, fp, &hdr); getsyms(fname, member, fp, &hdr); printsyms(fname, member, syms, nsyms); free(sections); free(syms); return 1; } static int probe(char *fname, char *member, FILE *fp) { int c; int c1, c2; fpos_t pos; unsigned short magic; fgetpos(fp, &pos); c1 = getc(fp); c2 = getc(fp); fsetpos(fp, &pos); if (ferror(fp)) die("nm: %s: %s", fname, strerror(errno)); if (c1 == EOF || c2 == EOF) return 0; magic = c1 | c2 << 8; switch (magic) { case COFF_I386MAGIC: case COFF_Z80MAGIC: unpack = lunpack; return 1; default: unpack = NULL; return 0; } } struct objfile coff32 = { .probe = probe, .nm = nm, };