13e12c5d1SDavid du Colombier /*
23e12c5d1SDavid du Colombier * obj.c
33e12c5d1SDavid du Colombier * routines universal to all object files
43e12c5d1SDavid du Colombier */
53e12c5d1SDavid du Colombier #include <u.h>
63e12c5d1SDavid du Colombier #include <libc.h>
73e12c5d1SDavid du Colombier #include <bio.h>
8bd389b36SDavid du Colombier #include <ar.h>
9bd389b36SDavid du Colombier #include <mach.h>
103e12c5d1SDavid du Colombier #include "obj.h"
11bd389b36SDavid du Colombier
12bd389b36SDavid du Colombier #define islocal(t) ((t)=='a' || (t)=='p')
13bd389b36SDavid du Colombier
14bd389b36SDavid du Colombier enum
15bd389b36SDavid du Colombier {
16bd389b36SDavid du Colombier NNAMES = 50,
17bd389b36SDavid du Colombier MAXIS = 8, /* max length to determine if a file is a .? file */
18bd389b36SDavid du Colombier MAXOFF = 0x7fffffff, /* larger than any possible local offset */
19219b2ee8SDavid du Colombier NHASH = 1024, /* must be power of two */
20219b2ee8SDavid du Colombier HASHMUL = 79L,
21bd389b36SDavid du Colombier };
22bd389b36SDavid du Colombier
23bd389b36SDavid du Colombier int _is2(char*), /* in [$OS].c */
24f8bc6aafSDavid du Colombier _is4(char*),
257dd7cddfSDavid du Colombier _is5(char*),
26b0dcc5a8SDavid du Colombier _is6(char*),
277dd7cddfSDavid du Colombier _is7(char*),
28bd389b36SDavid du Colombier _is8(char*),
2947ad9175SDavid du Colombier _is9(char*),
30bd389b36SDavid du Colombier _isk(char*),
317dd7cddfSDavid du Colombier _isq(char*),
32bd389b36SDavid du Colombier _isv(char*),
33b0dcc5a8SDavid du Colombier _isu(char*),
34*ce95e1b3SDavid du Colombier _isi(char*),
35bd389b36SDavid du Colombier _read2(Biobuf*, Prog*),
36f8bc6aafSDavid du Colombier _read4(Biobuf*, Prog*),
377dd7cddfSDavid du Colombier _read5(Biobuf*, Prog*),
38b0dcc5a8SDavid du Colombier _read6(Biobuf*, Prog*),
397dd7cddfSDavid du Colombier _read7(Biobuf*, Prog*),
40bd389b36SDavid du Colombier _read8(Biobuf*, Prog*),
4147ad9175SDavid du Colombier _read9(Biobuf*, Prog*),
42bd389b36SDavid du Colombier _readk(Biobuf*, Prog*),
437dd7cddfSDavid du Colombier _readq(Biobuf*, Prog*),
44b0dcc5a8SDavid du Colombier _readv(Biobuf*, Prog*),
45*ce95e1b3SDavid du Colombier _readu(Biobuf*, Prog*),
46*ce95e1b3SDavid du Colombier _readi(Biobuf*, Prog*);
473e12c5d1SDavid du Colombier
483e12c5d1SDavid du Colombier typedef struct Obj Obj;
49219b2ee8SDavid du Colombier typedef struct Symtab Symtab;
503e12c5d1SDavid du Colombier
51bd389b36SDavid du Colombier struct Obj /* functions to handle each intermediate (.$O) file */
52bd389b36SDavid du Colombier {
53219b2ee8SDavid du Colombier char *name; /* name of each $O file */
54bd389b36SDavid du Colombier int (*is)(char*); /* test for each type of $O file */
55bd389b36SDavid du Colombier int (*read)(Biobuf*, Prog*); /* read for each type of $O file*/
563e12c5d1SDavid du Colombier };
573e12c5d1SDavid du Colombier
58bd389b36SDavid du Colombier static Obj obj[] =
59bd389b36SDavid du Colombier { /* functions to identify and parse each type of obj */
60219b2ee8SDavid du Colombier [Obj68020] "68020 .2", _is2, _read2,
61b0dcc5a8SDavid du Colombier [ObjAmd64] "amd64 .6", _is6, _read6,
627dd7cddfSDavid du Colombier [ObjArm] "arm .5", _is5, _read5,
638a34c8c5SDavid du Colombier [ObjAlpha] "alpha .7", _is7, _read7,
64375daca8SDavid du Colombier [Obj386] "386 .8", _is8, _read8,
65375daca8SDavid du Colombier [ObjSparc] "sparc .k", _isk, _readk,
66375daca8SDavid du Colombier [ObjPower] "power .q", _isq, _readq,
67375daca8SDavid du Colombier [ObjMips] "mips .v", _isv, _readv,
68b0dcc5a8SDavid du Colombier [ObjSparc64] "sparc64 .u", _isu, _readu,
6947ad9175SDavid du Colombier [ObjPower64] "power64 .9", _is9, _read9,
70*ce95e1b3SDavid du Colombier [ObjRiscv] "riscv .i", _isi, _readi,
71*ce95e1b3SDavid du Colombier [ObjRiscv64] "riscv64 .j", _isi, _readi,
72f8bc6aafSDavid du Colombier [ObjMips2] "mips64 .4", _is4, _read4,
73bd389b36SDavid du Colombier [Maxobjtype] 0, 0
743e12c5d1SDavid du Colombier };
753e12c5d1SDavid du Colombier
76219b2ee8SDavid du Colombier struct Symtab
77219b2ee8SDavid du Colombier {
78219b2ee8SDavid du Colombier struct Sym s;
79219b2ee8SDavid du Colombier struct Symtab *next;
80219b2ee8SDavid du Colombier };
813e12c5d1SDavid du Colombier
82219b2ee8SDavid du Colombier static Symtab *hash[NHASH];
83219b2ee8SDavid du Colombier static Sym *names[NNAMES]; /* working set of active names */
84219b2ee8SDavid du Colombier
85219b2ee8SDavid du Colombier static int processprog(Prog*,int); /* decode each symbol reference */
86219b2ee8SDavid du Colombier static void objreset(void);
870c547597SDavid du Colombier static void objlookup(int, char *, int, uint);
88219b2ee8SDavid du Colombier static void objupdate(int, int);
893e12c5d1SDavid du Colombier
90bd389b36SDavid du Colombier int
objtype(Biobuf * bp,char ** name)91219b2ee8SDavid du Colombier objtype(Biobuf *bp, char **name)
92bd389b36SDavid du Colombier {
93bd389b36SDavid du Colombier int i;
94bd389b36SDavid du Colombier char buf[MAXIS];
95bd389b36SDavid du Colombier
96bd389b36SDavid du Colombier if(Bread(bp, buf, MAXIS) < MAXIS)
97bd389b36SDavid du Colombier return -1;
98bd389b36SDavid du Colombier Bseek(bp, -MAXIS, 1);
997dd7cddfSDavid du Colombier for (i = 0; i < Maxobjtype; i++) {
1007dd7cddfSDavid du Colombier if (obj[i].is && (*obj[i].is)(buf)) {
101219b2ee8SDavid du Colombier if (name)
102219b2ee8SDavid du Colombier *name = obj[i].name;
103bd389b36SDavid du Colombier return i;
104bd389b36SDavid du Colombier }
105219b2ee8SDavid du Colombier }
106bd389b36SDavid du Colombier return -1;
107bd389b36SDavid du Colombier }
108bd389b36SDavid du Colombier
109bd389b36SDavid du Colombier int
isar(Biobuf * bp)110bd389b36SDavid du Colombier isar(Biobuf *bp)
111bd389b36SDavid du Colombier {
112bd389b36SDavid du Colombier int n;
113bd389b36SDavid du Colombier char magbuf[SARMAG];
114bd389b36SDavid du Colombier
115bd389b36SDavid du Colombier n = Bread(bp, magbuf, SARMAG);
116bd389b36SDavid du Colombier if(n == SARMAG && strncmp(magbuf, ARMAG, SARMAG) == 0)
117bd389b36SDavid du Colombier return 1;
118bd389b36SDavid du Colombier return 0;
119bd389b36SDavid du Colombier }
1203e12c5d1SDavid du Colombier
1213e12c5d1SDavid du Colombier /*
1223e12c5d1SDavid du Colombier * determine what kind of object file this is and process it.
1233e12c5d1SDavid du Colombier * return whether or not this was a recognized intermediate file.
1243e12c5d1SDavid du Colombier */
1253e12c5d1SDavid du Colombier int
readobj(Biobuf * bp,int objtype)126bd389b36SDavid du Colombier readobj(Biobuf *bp, int objtype)
1273e12c5d1SDavid du Colombier {
1283e12c5d1SDavid du Colombier Prog p;
1293e12c5d1SDavid du Colombier
1307dd7cddfSDavid du Colombier if (objtype < 0 || objtype >= Maxobjtype || obj[objtype].is == 0)
131bd389b36SDavid du Colombier return 1;
132219b2ee8SDavid du Colombier objreset();
133bd389b36SDavid du Colombier while ((*obj[objtype].read)(bp, &p))
134219b2ee8SDavid du Colombier if (!processprog(&p, 1))
135bd389b36SDavid du Colombier return 0;
136bd389b36SDavid du Colombier return 1;
1373e12c5d1SDavid du Colombier }
138bd389b36SDavid du Colombier
139bd389b36SDavid du Colombier int
readar(Biobuf * bp,int objtype,vlong end,int doautos)1404de34a7eSDavid du Colombier readar(Biobuf *bp, int objtype, vlong end, int doautos)
141bd389b36SDavid du Colombier {
142bd389b36SDavid du Colombier Prog p;
143bd389b36SDavid du Colombier
1447dd7cddfSDavid du Colombier if (objtype < 0 || objtype >= Maxobjtype || obj[objtype].is == 0)
145bd389b36SDavid du Colombier return 1;
146219b2ee8SDavid du Colombier objreset();
1477dd7cddfSDavid du Colombier while ((*obj[objtype].read)(bp, &p) && Boffset(bp) < end)
148219b2ee8SDavid du Colombier if (!processprog(&p, doautos))
149bd389b36SDavid du Colombier return 0;
150bd389b36SDavid du Colombier return 1;
151bd389b36SDavid du Colombier }
152bd389b36SDavid du Colombier
153bd389b36SDavid du Colombier /*
154bd389b36SDavid du Colombier * decode a symbol reference or definition
155bd389b36SDavid du Colombier */
156bd389b36SDavid du Colombier static int
processprog(Prog * p,int doautos)157219b2ee8SDavid du Colombier processprog(Prog *p, int doautos)
158bd389b36SDavid du Colombier {
159bd389b36SDavid du Colombier if(p->kind == aNone)
160bd389b36SDavid du Colombier return 1;
161bd389b36SDavid du Colombier if(p->sym < 0 || p->sym >= NNAMES)
162bd389b36SDavid du Colombier return 0;
163bd389b36SDavid du Colombier switch(p->kind)
164bd389b36SDavid du Colombier {
1653e12c5d1SDavid du Colombier case aName:
166219b2ee8SDavid du Colombier if (!doautos)
167219b2ee8SDavid du Colombier if(p->type != 'U' && p->type != 'b')
168219b2ee8SDavid du Colombier break;
1690c547597SDavid du Colombier objlookup(p->sym, p->id, p->type, p->sig);
1703e12c5d1SDavid du Colombier break;
1713e12c5d1SDavid du Colombier case aText:
172bd389b36SDavid du Colombier objupdate(p->sym, 'T');
1733e12c5d1SDavid du Colombier break;
1743e12c5d1SDavid du Colombier case aData:
175bd389b36SDavid du Colombier objupdate(p->sym, 'D');
176bd389b36SDavid du Colombier break;
177bd389b36SDavid du Colombier default:
1783e12c5d1SDavid du Colombier break;
1793e12c5d1SDavid du Colombier }
180bd389b36SDavid du Colombier return 1;
1813e12c5d1SDavid du Colombier }
1823e12c5d1SDavid du Colombier
1833e12c5d1SDavid du Colombier /*
1843e12c5d1SDavid du Colombier * find the entry for s in the symbol array.
1853e12c5d1SDavid du Colombier * make a new entry if it is not already there.
1863e12c5d1SDavid du Colombier */
187bd389b36SDavid du Colombier static void
objlookup(int id,char * name,int type,uint sig)1880c547597SDavid du Colombier objlookup(int id, char *name, int type, uint sig)
1893e12c5d1SDavid du Colombier {
190219b2ee8SDavid du Colombier long h;
191219b2ee8SDavid du Colombier char *cp;
1923e12c5d1SDavid du Colombier Sym *s;
193219b2ee8SDavid du Colombier Symtab *sp;
1943e12c5d1SDavid du Colombier
195219b2ee8SDavid du Colombier s = names[id];
196219b2ee8SDavid du Colombier if(s && strcmp(s->name, name) == 0) {
1973e12c5d1SDavid du Colombier s->type = type;
1980c547597SDavid du Colombier s->sig = sig;
199bd389b36SDavid du Colombier return;
2003e12c5d1SDavid du Colombier }
201219b2ee8SDavid du Colombier
202219b2ee8SDavid du Colombier h = *name;
203219b2ee8SDavid du Colombier for(cp = name+1; *cp; h += *cp++)
204219b2ee8SDavid du Colombier h *= HASHMUL;
205219b2ee8SDavid du Colombier if(h < 0)
206219b2ee8SDavid du Colombier h = ~h;
207219b2ee8SDavid du Colombier h &= (NHASH-1);
208219b2ee8SDavid du Colombier if (type == 'U' || type == 'b' || islocal(type)) {
209219b2ee8SDavid du Colombier for(sp = hash[h]; sp; sp = sp->next)
210219b2ee8SDavid du Colombier if(strcmp(sp->s.name, name) == 0) {
211219b2ee8SDavid du Colombier switch(sp->s.type) {
2123e12c5d1SDavid du Colombier case 'T':
2133e12c5d1SDavid du Colombier case 'D':
2143e12c5d1SDavid du Colombier case 'U':
2153e12c5d1SDavid du Colombier if (type == 'U') {
216219b2ee8SDavid du Colombier names[id] = &sp->s;
217bd389b36SDavid du Colombier return;
2183e12c5d1SDavid du Colombier }
2193e12c5d1SDavid du Colombier break;
2203e12c5d1SDavid du Colombier case 't':
2213e12c5d1SDavid du Colombier case 'd':
2223e12c5d1SDavid du Colombier case 'b':
2233e12c5d1SDavid du Colombier if (type == 'b') {
224219b2ee8SDavid du Colombier names[id] = &sp->s;
225bd389b36SDavid du Colombier return;
2263e12c5d1SDavid du Colombier }
2273e12c5d1SDavid du Colombier break;
2283e12c5d1SDavid du Colombier case 'a':
2293e12c5d1SDavid du Colombier case 'p':
230219b2ee8SDavid du Colombier if (islocal(type)) {
231219b2ee8SDavid du Colombier names[id] = &sp->s;
232bd389b36SDavid du Colombier return;
2333e12c5d1SDavid du Colombier }
2343e12c5d1SDavid du Colombier break;
2353e12c5d1SDavid du Colombier default:
2363e12c5d1SDavid du Colombier break;
2373e12c5d1SDavid du Colombier }
238bd389b36SDavid du Colombier }
239bd389b36SDavid du Colombier }
240219b2ee8SDavid du Colombier sp = malloc(sizeof(Symtab));
241219b2ee8SDavid du Colombier sp->s.name = name;
242219b2ee8SDavid du Colombier sp->s.type = type;
2430c547597SDavid du Colombier sp->s.sig = sig;
244219b2ee8SDavid du Colombier sp->s.value = islocal(type) ? MAXOFF : 0;
245219b2ee8SDavid du Colombier names[id] = &sp->s;
246219b2ee8SDavid du Colombier sp->next = hash[h];
247219b2ee8SDavid du Colombier hash[h] = sp;
248bd389b36SDavid du Colombier return;
249bd389b36SDavid du Colombier }
250bd389b36SDavid du Colombier /*
251219b2ee8SDavid du Colombier * traverse the symbol lists
252bd389b36SDavid du Colombier */
253219b2ee8SDavid du Colombier void
objtraverse(void (* fn)(Sym *,void *),void * pointer)254219b2ee8SDavid du Colombier objtraverse(void (*fn)(Sym*, void*), void *pointer)
255bd389b36SDavid du Colombier {
256219b2ee8SDavid du Colombier int i;
257219b2ee8SDavid du Colombier Symtab *s;
258219b2ee8SDavid du Colombier
259219b2ee8SDavid du Colombier for(i = 0; i < NHASH; i++)
260219b2ee8SDavid du Colombier for(s = hash[i]; s; s = s->next)
261219b2ee8SDavid du Colombier (*fn)(&s->s, pointer);
2623e12c5d1SDavid du Colombier }
2633e12c5d1SDavid du Colombier
2643e12c5d1SDavid du Colombier /*
2653e12c5d1SDavid du Colombier * update the offset information for a 'a' or 'p' symbol in an intermediate file
2663e12c5d1SDavid du Colombier */
2673e12c5d1SDavid du Colombier void
_offset(int id,vlong off)2684de34a7eSDavid du Colombier _offset(int id, vlong off)
2693e12c5d1SDavid du Colombier {
2703e12c5d1SDavid du Colombier Sym *s;
2713e12c5d1SDavid du Colombier
272219b2ee8SDavid du Colombier s = names[id];
273219b2ee8SDavid du Colombier if (s && s->name[0] && islocal(s->type) && s->value > off)
2743e12c5d1SDavid du Colombier s->value = off;
2753e12c5d1SDavid du Colombier }
2763e12c5d1SDavid du Colombier
2773e12c5d1SDavid du Colombier /*
2783e12c5d1SDavid du Colombier * update the type of a global text or data symbol
2793e12c5d1SDavid du Colombier */
280219b2ee8SDavid du Colombier static void
objupdate(int id,int type)281bd389b36SDavid du Colombier objupdate(int id, int type)
2823e12c5d1SDavid du Colombier {
2833e12c5d1SDavid du Colombier Sym *s;
2843e12c5d1SDavid du Colombier
285219b2ee8SDavid du Colombier s = names[id];
286219b2ee8SDavid du Colombier if (s && s->name[0])
287219b2ee8SDavid du Colombier if (s->type == 'U')
2883e12c5d1SDavid du Colombier s->type = type;
289219b2ee8SDavid du Colombier else if (s->type == 'b')
2903e12c5d1SDavid du Colombier s->type = tolower(type);
2913e12c5d1SDavid du Colombier }
2923e12c5d1SDavid du Colombier
2933e12c5d1SDavid du Colombier /*
2943e12c5d1SDavid du Colombier * look for the next file in an archive
2953e12c5d1SDavid du Colombier */
2963e12c5d1SDavid du Colombier int
nextar(Biobuf * bp,int offset,char * buf)297bd389b36SDavid du Colombier nextar(Biobuf *bp, int offset, char *buf)
2983e12c5d1SDavid du Colombier {
2993e12c5d1SDavid du Colombier struct ar_hdr a;
3003e12c5d1SDavid du Colombier int i, r;
3013e12c5d1SDavid du Colombier long arsize;
302bd389b36SDavid du Colombier
303bd389b36SDavid du Colombier if (offset&01)
304bd389b36SDavid du Colombier offset++;
305bd389b36SDavid du Colombier Bseek(bp, offset, 0);
306bd389b36SDavid du Colombier r = Bread(bp, &a, SAR_HDR);
3073e12c5d1SDavid du Colombier if(r != SAR_HDR)
3083e12c5d1SDavid du Colombier return 0;
309bd389b36SDavid du Colombier if(strncmp(a.fmag, ARFMAG, sizeof(a.fmag)))
310bd389b36SDavid du Colombier return -1;
311219b2ee8SDavid du Colombier for(i=0; i<sizeof(a.name) && i<SARNAME && a.name[i] != ' '; i++)
312bd389b36SDavid du Colombier buf[i] = a.name[i];
313bd389b36SDavid du Colombier buf[i] = 0;
3144de34a7eSDavid du Colombier arsize = strtol(a.size, 0, 0);
315bd389b36SDavid du Colombier if (arsize&1)
316bd389b36SDavid du Colombier arsize++;
317bd389b36SDavid du Colombier return arsize + SAR_HDR;
3183e12c5d1SDavid du Colombier }
3193e12c5d1SDavid du Colombier
320bd389b36SDavid du Colombier static void
objreset(void)321bd389b36SDavid du Colombier objreset(void)
322bd389b36SDavid du Colombier {
323219b2ee8SDavid du Colombier int i;
324219b2ee8SDavid du Colombier Symtab *s, *n;
325219b2ee8SDavid du Colombier
326219b2ee8SDavid du Colombier for(i = 0; i < NHASH; i++) {
327219b2ee8SDavid du Colombier for(s = hash[i]; s; s = n) {
328219b2ee8SDavid du Colombier n = s->next;
329219b2ee8SDavid du Colombier free(s->s.name);
330219b2ee8SDavid du Colombier free(s);
3313e12c5d1SDavid du Colombier }
332219b2ee8SDavid du Colombier hash[i] = 0;
333219b2ee8SDavid du Colombier }
334219b2ee8SDavid du Colombier memset(names, 0, sizeof names);
3353e12c5d1SDavid du Colombier }
336