/* $Id$ */ /* a.out file reading ... */ #include "rd.h" #include "misc.h" #include #include #if defined(__sun) #if ! defined(sun) #define sun #endif #endif #if defined(__i386) #define i386 #endif #if defined(__mc68020) #define mc68020 #endif #if defined(__sparc) #if ! defined(sparc) #define sparc #endif #endif #if defined(__vax) #define vax #endif #if defined(__solaris) || defined(__solaris__) #define solaris #endif #if ! defined(solaris) #if defined(sun) || defined(vax) struct exec { #ifdef sun short a_x; unsigned short a_magic; #else unsigned long a_magic; #endif unsigned long a_text; unsigned long a_data; unsigned long a_bss; unsigned long a_syms; unsigned long a_entry; unsigned long a_trsize; unsigned long a_drsize; }; #define OMAGIC 0407 #define NMAGIC 0410 #define ZMAGIC 0413 #define N_BADMAG(x) \ ((x).a_magic!=OMAGIC && (x).a_magic!=NMAGIC && (x).a_magic!=ZMAGIC) #ifdef sun #define N_TXTOFF(x) ((x).a_magic == ZMAGIC ? 0 : sizeof(struct exec)) #else #define N_TXTOFF(x) (sizeof(struct exec)) #endif #define N_STROFF(x) (N_TXTOFF(x)+(x).a_text+(x).a_data+(x).a_trsize+(x).a_drsize+(x).a_syms) #ifdef sparc #define RELOC_SIZE 12 #else #define RELOC_SIZE 8 #endif struct nlist { union { char *n_name; long n_strx; } n_un; unsigned char n_type; char n_other; short n_desc; unsigned long n_value; }; #define N_UNDF 0 #define N_ABS 2 #define N_TEXT 4 #define N_DATA 6 #define N_BSS 8 #define N_FN 0x1e #define N_EXT 01 #define N_STAB 0xe0 #include static FILE *inf; static struct exec bh; static long seg_strings; static struct outhead hh; #define readf(a, b, c) (fread((char *)(a), (b), (int)(c), inf)) int rd_open(f) char *f; { if ((inf = fopen(f, "rb")) == NULL) return 0; return 1; } rd_ohead(h) struct outhead *h; { if (! readf(&bh, sizeof(struct exec), 1)) rd_fatal(); if (N_BADMAG(bh)) rd_fatal(); h->oh_magic = O_CONVERTED; h->oh_stamp = 0; h->oh_nsect = 4; h->oh_nname = 3 + bh.a_syms / sizeof(struct nlist); h->oh_nrelo = (bh.a_trsize + bh.a_drsize) / RELOC_SIZE; h->oh_flags = h->oh_nrelo ? HF_LINK : 0; #if defined(sun) if (bh.a_magic == ZMAGIC) bh.a_text -= sizeof(struct exec); #endif h->oh_nemit = bh.a_text + bh.a_data; #if defined(sun) if (bh.a_magic == ZMAGIC) bh.a_text += sizeof(struct exec); #endif fseek(inf, N_STROFF(bh), 0); h->oh_nchar = getw(inf) + 6 + 6 + 5 - 4; /* ".text", ".data", ".bss", minus the size word */ seg_strings = h->oh_nchar - 17; fseek(inf, N_TXTOFF(bh)+bh.a_text+bh.a_data, 0); hh = *h; #if defined(sun) if (bh.a_magic == ZMAGIC) bh.a_text -= sizeof(struct exec); #endif } /*ARGSUSED1*/ rd_name(names, count) register struct outname *names; unsigned int count; /* ignored; complete namelist is read */ { names->on_valu = 0; names->on_foff = seg_strings + OFF_CHAR(hh); names->on_desc = 0; names->on_type = S_MIN | S_SCT; names++; names->on_valu = 0; names->on_foff = seg_strings + OFF_CHAR(hh) + 6; names->on_desc = 0; names->on_type = (S_MIN+2) | S_SCT; names++; names->on_valu = 0; names->on_foff = seg_strings + OFF_CHAR(hh) + 12; names->on_desc = 0; names->on_type = (S_MIN+3) | S_SCT; names++; count = bh.a_syms / sizeof(struct nlist); while (count > 0) { struct nlist n; if (! readf(&n, sizeof(struct nlist), 1)) rd_fatal(); count--; names->on_desc = n.n_desc; if (n.n_un.n_strx - 4 < 0) names->on_foff = 0; else names->on_foff = OFF_CHAR(hh) - 4 + n.n_un.n_strx; names->on_valu = n.n_value; if (n.n_type & N_STAB) { names->on_type = n.n_type << 8; names++; continue; } switch(n.n_type & ~N_EXT) { case N_ABS: names->on_type = S_ABS; break; case N_TEXT: names->on_type = S_MIN; break; case N_DATA: names->on_type = S_MIN + 2; names->on_valu -= bh.a_text; break; case N_BSS: names->on_type = S_MIN + 3; names->on_valu -= bh.a_text + bh.a_data; break; case N_UNDF: if (! names->on_valu) { names->on_type = S_UND; break; } names->on_type = (S_MIN + 3) | S_COM; break; case N_FN: names->on_type = S_FIL; break; default: rd_fatal(); } if (n.n_type & N_EXT) names->on_type |= S_EXT; names++; } } extern char *strcpy(); rd_string(strings, count) register char *strings; long count; { #if defined(sun) if (bh.a_magic == ZMAGIC) bh.a_text += sizeof(struct exec); #endif fseek(inf, N_STROFF(bh)+4, 0); if (! readf(strings, (int)count-17, 1)) rd_fatal(); strings += count-17; strcpy(strings, ".text"); strings += 6; strcpy(strings, ".data"); strings += 6; strcpy(strings, ".bss"); } rd_close() { fclose(inf); } #endif #endif #if defined(i386) #include #include struct xexec { unsigned short x_magic; #define XMAGIC 01006 unsigned short x_ext; long x_text; long x_data; long x_bss; long x_syms; long x_reloc; long x_entry; char x_cpu; char x_relsym; unsigned short x_renv; }; struct xseg { unsigned short xs_type; unsigned short xs_attr; unsigned short xs_seg; unsigned short xs_sres; long xs_filpos; long xs_psize; long xs_vsize; long xs_rbase; long xs_lres; long xs_lres2; }; static FILE *inf; static struct outname *names; static char *strings; #define readf(a, b, c) (fread((char *)(a), (b), (int)(c), inf)) #define getshort(val, p) (val = (*p++ & 0377), val |= (*p++ & 0377) << 8) #define getlong(val, p) (val = (*p++ & 0377), \ val |= (*p++ & 0377) << 8, \ val |= (*p++ & 0377L) << 16, \ val |= (*p++ & 0377L) << 24) static get_names(h, sz) struct outhead *h; long sz; { register char *xnms = malloc((unsigned) sz); register char *p; register struct outname *onm = (struct outname *) malloc((((unsigned)sz+8)/9)*sizeof(struct outname)); struct xnm { unsigned short s_type, s_seg; long s_value; } xnm; if (xnms == 0 || onm == 0) No_Mem(); if (!readf(xnms, (unsigned) sz, 1)) rd_fatal(); names = onm; strings = p = xnms; while (sz > 0) { getshort(xnm.s_type, xnms); getshort(xnm.s_seg, xnms); getlong(xnm.s_value, xnms); onm->on_desc = 0; if (xnm.s_type & S_STB) { onm->on_type = xnm.s_type; onm->on_desc = xnm.s_seg; } else { switch(xnm.s_type & 0x1f) { case 0x1f: onm->on_type = S_FIL; break; case 0x8: onm->on_type = S_SCT; break; case 0: onm->on_type = S_UND; break; case 1: onm->on_type = S_ABS; break; default: onm->on_type = xnm.s_type & 0x1f; break; } } if (xnm.s_type & 0x20) onm->on_type |= S_EXT; onm->on_valu = xnm.s_value; sz -= 9; if (*xnms == '\0') { onm->on_foff = -1; xnms++; } else { onm->on_foff = p - strings; while (*p++ = *xnms++) sz--; } onm++; } h->oh_nname = onm - names; h->oh_nchar = p - strings; while (--onm >= names) { if (onm->on_foff == -1) onm->on_foff = 0; else onm->on_foff += OFF_CHAR(*h); } names = (struct outname *) realloc((char *) names, h->oh_nname * sizeof(struct outname)); strings = realloc(strings, (unsigned) h->oh_nchar); } int rd_open(f) char *f; { if ((inf = fopen(f, "rb")) == NULL) return 0; return 1; } rd_ohead(h) struct outhead *h; { int sepid; struct xexec xhdr; struct xseg xseg[3]; if (! readf(&xhdr, sizeof(xhdr), 1)) rd_fatal(); if (xhdr.x_magic != XMAGIC) rd_fatal(); h->oh_magic = O_CONVERTED; h->oh_stamp = 0; h->oh_nsect = 4; h->oh_nrelo = 0; h->oh_flags = 0; h->oh_nemit = xhdr.x_text+xhdr.x_data; sepid = (xhdr.x_renv & 02) ? 1 : 0; fseek(inf, 0140L, 0); if (! readf(&xseg[0], sizeof(xseg[0]), sepid + 2)) rd_fatal(); fseek(inf, xseg[sepid+1].xs_filpos, 0); get_names(h, xhdr.x_syms); fclose(inf); } rd_name(nm, count) struct outname *nm; unsigned int count; { memcpy(nm, names, (int) count * sizeof(struct outname)); free((char *) names); } rd_string(nm, count) char *nm; long count; { memcpy(nm, strings, (int) count); free((char *) strings); } rd_close() { } #endif #if defined(solaris) #include #include #include struct nlist { union { char *n_name; long n_strx; } n_un; unsigned char n_type; char n_other; short n_desc; unsigned long n_value; }; static int fildes; static Elf *elf; static Elf32_Ehdr *ehdr; static struct nlist *dbtab; static char *dbstringtab; static Elf32_Sym *tab; static char *stringtab; static struct outhead hh; static struct nlist *maxdn; #define N_STAB 0xe0 int rd_open(f) char *f; { if ((fildes = open(f, 0)) < 0) return 0; elf_version(EV_CURRENT); if ((elf = elf_begin(fildes, ELF_C_READ, (Elf *) 0)) == 0) { close(fildes); return 0; } if ((ehdr = elf32_getehdr(elf)) == NULL) { elf_end(elf); close(fildes); return 0; } return 1; } rd_ohead(h) struct outhead *h; { Elf_Scn *scn = 0; Elf32_Shdr *shdr; Elf_Data *sectnames; Elf_Data *dt; register struct nlist *dn; register Elf32_Sym *n; long text_offset, data_offset, bss_offset, fun_offset; int fixnamoff = 0, newfixnamoff = 0; h->oh_magic = O_CONVERTED; h->oh_stamp = 0; h->oh_nsect = 4; h->oh_nrelo = 0; h->oh_flags = 0; h->oh_nemit = 0; h->oh_nname = 0; scn = elf_getscn(elf, (size_t) ehdr->e_shstrndx); sectnames = elf_getdata(scn, (Elf_Data *) 0); scn = 0; while ((scn = elf_nextscn(elf, scn)) != 0) { shdr = elf32_getshdr(scn); switch(shdr->sh_type) { case SHT_PROGBITS: /* Get stab symbol table. Elf does not know about it, and, unfortunately, no relocation is done on it. */ h->oh_nemit += shdr->sh_size; if (! strcmp(".stab", (char *)(sectnames->d_buf)+shdr->sh_name)) { dt = elf_getdata(scn, (Elf_Data *) 0); if (dt->d_size == 0) { fatal("(part of) symbol table is missing"); } dbtab = (struct nlist *) Malloc(dt->d_size); memcpy((char *) dbtab, (char *) dt->d_buf, dt->d_size); maxdn = (struct nlist *)((char *)dbtab+dt->d_size); break; } break; case SHT_STRTAB: /* Get the stab string table, as well as the usual string table. */ if (! strcmp(".stabstr", (char *)(sectnames->d_buf)+shdr->sh_name)) { dt = elf_getdata(scn, (Elf_Data *) 0); if (dt->d_size == 0) { fatal("(part of) symbol table is missing"); } dbstringtab = dt->d_buf; h->oh_nchar = dt->d_size; break; } if (! strcmp(".strtab", (char *)(sectnames->d_buf)+shdr->sh_name)) { dt = elf_getdata(scn, (Elf_Data *) 0); if (dt->d_size == 0) { fatal("(part of) symbol table is missing"); } stringtab = dt->d_buf; } break; case SHT_SYMTAB: /* Get the symbol table. */ if (! strcmp(".symtab", (char *)(sectnames->d_buf)+shdr->sh_name)) { dt = elf_getdata(scn, (Elf_Data *) 0); if (dt->d_size == 0) { fatal("(part of) symbol table is missing"); } tab = dt->d_buf; } break; } } /* Convert offsets in stab symbol table. */ n = tab; dn = dbtab; while (dn < maxdn) { int i; if (dn->n_un.n_strx) { dn->n_un.n_strx += fixnamoff; } switch(dn->n_type) { case 0: fixnamoff = newfixnamoff; newfixnamoff += dn->n_value; break; case N_SO: h->oh_nname++; i = 0; while (i < 3) { while (stringtab[n->st_name] != 'B') n++; if (! strcmp("Btext.text", &(stringtab[n->st_name]))) { text_offset = n->st_value; i++; } else if (! strcmp("Bdata.data", &(stringtab[n->st_name]))) { data_offset = n->st_value; i++; } else if (! strcmp("Bbss.bss", &(stringtab[n->st_name]))) { bss_offset = n->st_value; i++; } n++; } break; case N_GSYM: h->oh_nname++; /* Fortunately, we don't use this in ACK, so we don't have to handle it here. The problem is that we don't know which segment it comes from. */ break; case N_STSYM: h->oh_nname++; dn->n_value += data_offset; break; case N_LCSYM: h->oh_nname++; dn->n_value += bss_offset; break; case N_FUN: h->oh_nname++; dn->n_value += text_offset; fun_offset = dn->n_value; break; case N_MAIN: dn->n_value += text_offset; break; case N_LBRAC: case N_RBRAC: case N_SLINE: h->oh_nname++; dn->n_value += fun_offset; break; case N_SOL: case N_EINCL: case N_BINCL: case N_PSYM: case N_SSYM: case N_SCOPE: case N_RSYM: case N_LSYM: h->oh_nname++; /* Nothing to be done. */ break; } dn++; } hh = *h; } rd_name(nm, count) struct outname *nm; unsigned int count; { register struct nlist *dn = dbtab; register struct outname *n = nm; while (dn < maxdn) { if (dn->n_type & N_STAB) { n->on_type = dn->n_type << 8; n->on_valu = dn->n_value; n->on_desc = dn->n_desc; if (dn->n_un.n_strx == 0) n->on_foff = 0; else n->on_foff = OFF_CHAR(hh) + dn->n_un.n_strx; n++; } dn++; } free(dbtab); } rd_string(nm, count) char *nm; long count; { memcpy(nm, dbstringtab, count); } rd_close() { elf_end(elf); close(fildes); } #endif