xref: /plan9/sys/src/libmach/obj.c (revision 8a34c8c5ee1cf0942ec4f65ff605869b55820ac1)
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 */
247dd7cddfSDavid du Colombier 	_is5(char*),
25b0dcc5a8SDavid du Colombier 	_is6(char*),
267dd7cddfSDavid du Colombier 	_is7(char*),
27bd389b36SDavid du Colombier 	_is8(char*),
2847ad9175SDavid du Colombier 	_is9(char*),
29bd389b36SDavid du Colombier 	_isk(char*),
307dd7cddfSDavid du Colombier 	_isq(char*),
31bd389b36SDavid du Colombier 	_isv(char*),
32b0dcc5a8SDavid du Colombier 	_isu(char*),
33bd389b36SDavid du Colombier 	_read2(Biobuf*, Prog*),
347dd7cddfSDavid du Colombier 	_read5(Biobuf*, Prog*),
35b0dcc5a8SDavid du Colombier 	_read6(Biobuf*, Prog*),
367dd7cddfSDavid du Colombier 	_read7(Biobuf*, Prog*),
37bd389b36SDavid du Colombier 	_read8(Biobuf*, Prog*),
3847ad9175SDavid du Colombier 	_read9(Biobuf*, Prog*),
39bd389b36SDavid du Colombier 	_readk(Biobuf*, Prog*),
407dd7cddfSDavid du Colombier 	_readq(Biobuf*, Prog*),
41b0dcc5a8SDavid du Colombier 	_readv(Biobuf*, Prog*),
42b0dcc5a8SDavid du Colombier 	_readu(Biobuf*, Prog*);
433e12c5d1SDavid du Colombier 
443e12c5d1SDavid du Colombier typedef struct Obj	Obj;
45219b2ee8SDavid du Colombier typedef struct Symtab	Symtab;
463e12c5d1SDavid du Colombier 
47bd389b36SDavid du Colombier struct	Obj		/* functions to handle each intermediate (.$O) file */
48bd389b36SDavid du Colombier {
49219b2ee8SDavid du Colombier 	char	*name;				/* name of each $O file */
50bd389b36SDavid du Colombier 	int	(*is)(char*);			/* test for each type of $O file */
51bd389b36SDavid du Colombier 	int	(*read)(Biobuf*, Prog*);	/* read for each type of $O file*/
523e12c5d1SDavid du Colombier };
533e12c5d1SDavid du Colombier 
54bd389b36SDavid du Colombier static Obj	obj[] =
55bd389b36SDavid du Colombier {			/* functions to identify and parse each type of obj */
56219b2ee8SDavid du Colombier 	[Obj68020]	"68020 .2",	_is2, _read2,
57b0dcc5a8SDavid du Colombier 	[ObjAmd64]	"amd64 .6",	_is6, _read6,
587dd7cddfSDavid du Colombier 	[ObjArm]	"arm .5",	_is5, _read5,
59*8a34c8c5SDavid du Colombier 	[ObjAlpha]	"alpha .7",	_is7, _read7,
60375daca8SDavid du Colombier 	[Obj386]	"386 .8",	_is8, _read8,
61375daca8SDavid du Colombier 	[ObjSparc]	"sparc .k",	_isk, _readk,
62375daca8SDavid du Colombier 	[ObjPower]	"power .q",	_isq, _readq,
63375daca8SDavid du Colombier 	[ObjMips]	"mips .v",	_isv, _readv,
64b0dcc5a8SDavid du Colombier 	[ObjSparc64]	"sparc64 .u",	_isu, _readu,
6547ad9175SDavid du Colombier 	[ObjPower64]	"power64 .9",	_is9, _read9,
66bd389b36SDavid du Colombier 	[Maxobjtype]	0, 0
673e12c5d1SDavid du Colombier };
683e12c5d1SDavid du Colombier 
69219b2ee8SDavid du Colombier struct	Symtab
70219b2ee8SDavid du Colombier {
71219b2ee8SDavid du Colombier 	struct	Sym 	s;
72219b2ee8SDavid du Colombier 	struct	Symtab	*next;
73219b2ee8SDavid du Colombier };
743e12c5d1SDavid du Colombier 
75219b2ee8SDavid du Colombier static	Symtab *hash[NHASH];
76219b2ee8SDavid du Colombier static	Sym	*names[NNAMES];	/* working set of active names */
77219b2ee8SDavid du Colombier 
78219b2ee8SDavid du Colombier static	int	processprog(Prog*,int);	/* decode each symbol reference */
79219b2ee8SDavid du Colombier static	void	objreset(void);
800c547597SDavid du Colombier static	void	objlookup(int, char *, int, uint);
81219b2ee8SDavid du Colombier static	void 	objupdate(int, int);
823e12c5d1SDavid du Colombier 
83bd389b36SDavid du Colombier int
objtype(Biobuf * bp,char ** name)84219b2ee8SDavid du Colombier objtype(Biobuf *bp, char **name)
85bd389b36SDavid du Colombier {
86bd389b36SDavid du Colombier 	int i;
87bd389b36SDavid du Colombier 	char buf[MAXIS];
88bd389b36SDavid du Colombier 
89bd389b36SDavid du Colombier 	if(Bread(bp, buf, MAXIS) < MAXIS)
90bd389b36SDavid du Colombier 		return -1;
91bd389b36SDavid du Colombier 	Bseek(bp, -MAXIS, 1);
927dd7cddfSDavid du Colombier 	for (i = 0; i < Maxobjtype; i++) {
937dd7cddfSDavid du Colombier 		if (obj[i].is && (*obj[i].is)(buf)) {
94219b2ee8SDavid du Colombier 			if (name)
95219b2ee8SDavid du Colombier 				*name = obj[i].name;
96bd389b36SDavid du Colombier 			return i;
97bd389b36SDavid du Colombier 		}
98219b2ee8SDavid du Colombier 	}
99bd389b36SDavid du Colombier 	return -1;
100bd389b36SDavid du Colombier }
101bd389b36SDavid du Colombier 
102bd389b36SDavid du Colombier int
isar(Biobuf * bp)103bd389b36SDavid du Colombier isar(Biobuf *bp)
104bd389b36SDavid du Colombier {
105bd389b36SDavid du Colombier 	int n;
106bd389b36SDavid du Colombier 	char magbuf[SARMAG];
107bd389b36SDavid du Colombier 
108bd389b36SDavid du Colombier 	n = Bread(bp, magbuf, SARMAG);
109bd389b36SDavid du Colombier 	if(n == SARMAG && strncmp(magbuf, ARMAG, SARMAG) == 0)
110bd389b36SDavid du Colombier 		return 1;
111bd389b36SDavid du Colombier 	return 0;
112bd389b36SDavid du Colombier }
1133e12c5d1SDavid du Colombier 
1143e12c5d1SDavid du Colombier /*
1153e12c5d1SDavid du Colombier  * determine what kind of object file this is and process it.
1163e12c5d1SDavid du Colombier  * return whether or not this was a recognized intermediate file.
1173e12c5d1SDavid du Colombier  */
1183e12c5d1SDavid du Colombier int
readobj(Biobuf * bp,int objtype)119bd389b36SDavid du Colombier readobj(Biobuf *bp, int objtype)
1203e12c5d1SDavid du Colombier {
1213e12c5d1SDavid du Colombier 	Prog p;
1223e12c5d1SDavid du Colombier 
1237dd7cddfSDavid du Colombier 	if (objtype < 0 || objtype >= Maxobjtype || obj[objtype].is == 0)
124bd389b36SDavid du Colombier 		return 1;
125219b2ee8SDavid du Colombier 	objreset();
126bd389b36SDavid du Colombier 	while ((*obj[objtype].read)(bp, &p))
127219b2ee8SDavid du Colombier 		if (!processprog(&p, 1))
128bd389b36SDavid du Colombier 			return 0;
129bd389b36SDavid du Colombier 	return 1;
1303e12c5d1SDavid du Colombier }
131bd389b36SDavid du Colombier 
132bd389b36SDavid du Colombier int
readar(Biobuf * bp,int objtype,vlong end,int doautos)1334de34a7eSDavid du Colombier readar(Biobuf *bp, int objtype, vlong end, int doautos)
134bd389b36SDavid du Colombier {
135bd389b36SDavid du Colombier 	Prog p;
136bd389b36SDavid du Colombier 
1377dd7cddfSDavid du Colombier 	if (objtype < 0 || objtype >= Maxobjtype || obj[objtype].is == 0)
138bd389b36SDavid du Colombier 		return 1;
139219b2ee8SDavid du Colombier 	objreset();
1407dd7cddfSDavid du Colombier 	while ((*obj[objtype].read)(bp, &p) && Boffset(bp) < end)
141219b2ee8SDavid du Colombier 		if (!processprog(&p, doautos))
142bd389b36SDavid du Colombier 			return 0;
143bd389b36SDavid du Colombier 	return 1;
144bd389b36SDavid du Colombier }
145bd389b36SDavid du Colombier 
146bd389b36SDavid du Colombier /*
147bd389b36SDavid du Colombier  *	decode a symbol reference or definition
148bd389b36SDavid du Colombier  */
149bd389b36SDavid du Colombier static	int
processprog(Prog * p,int doautos)150219b2ee8SDavid du Colombier processprog(Prog *p, int doautos)
151bd389b36SDavid du Colombier {
152bd389b36SDavid du Colombier 	if(p->kind == aNone)
153bd389b36SDavid du Colombier 		return 1;
154bd389b36SDavid du Colombier 	if(p->sym < 0 || p->sym >= NNAMES)
155bd389b36SDavid du Colombier 		return 0;
156bd389b36SDavid du Colombier 	switch(p->kind)
157bd389b36SDavid du Colombier 	{
1583e12c5d1SDavid du Colombier 	case aName:
159219b2ee8SDavid du Colombier 		if (!doautos)
160219b2ee8SDavid du Colombier 		if(p->type != 'U' && p->type != 'b')
161219b2ee8SDavid du Colombier 			break;
1620c547597SDavid du Colombier 		objlookup(p->sym, p->id, p->type, p->sig);
1633e12c5d1SDavid du Colombier 		break;
1643e12c5d1SDavid du Colombier 	case aText:
165bd389b36SDavid du Colombier 		objupdate(p->sym, 'T');
1663e12c5d1SDavid du Colombier 		break;
1673e12c5d1SDavid du Colombier 	case aData:
168bd389b36SDavid du Colombier 		objupdate(p->sym, 'D');
169bd389b36SDavid du Colombier 		break;
170bd389b36SDavid du Colombier 	default:
1713e12c5d1SDavid du Colombier 		break;
1723e12c5d1SDavid du Colombier 	}
173bd389b36SDavid du Colombier 	return 1;
1743e12c5d1SDavid du Colombier }
1753e12c5d1SDavid du Colombier 
1763e12c5d1SDavid du Colombier /*
1773e12c5d1SDavid du Colombier  * find the entry for s in the symbol array.
1783e12c5d1SDavid du Colombier  * make a new entry if it is not already there.
1793e12c5d1SDavid du Colombier  */
180bd389b36SDavid du Colombier static void
objlookup(int id,char * name,int type,uint sig)1810c547597SDavid du Colombier objlookup(int id, char *name, int type, uint sig)
1823e12c5d1SDavid du Colombier {
183219b2ee8SDavid du Colombier 	long h;
184219b2ee8SDavid du Colombier 	char *cp;
1853e12c5d1SDavid du Colombier 	Sym *s;
186219b2ee8SDavid du Colombier 	Symtab *sp;
1873e12c5d1SDavid du Colombier 
188219b2ee8SDavid du Colombier 	s = names[id];
189219b2ee8SDavid du Colombier 	if(s && strcmp(s->name, name) == 0) {
1903e12c5d1SDavid du Colombier 		s->type = type;
1910c547597SDavid du Colombier 		s->sig = sig;
192bd389b36SDavid du Colombier 		return;
1933e12c5d1SDavid du Colombier 	}
194219b2ee8SDavid du Colombier 
195219b2ee8SDavid du Colombier 	h = *name;
196219b2ee8SDavid du Colombier 	for(cp = name+1; *cp; h += *cp++)
197219b2ee8SDavid du Colombier 		h *= HASHMUL;
198219b2ee8SDavid du Colombier 	if(h < 0)
199219b2ee8SDavid du Colombier 		h = ~h;
200219b2ee8SDavid du Colombier 	h &= (NHASH-1);
201219b2ee8SDavid du Colombier 	if (type == 'U' || type == 'b' || islocal(type)) {
202219b2ee8SDavid du Colombier 		for(sp = hash[h]; sp; sp = sp->next)
203219b2ee8SDavid du Colombier 			if(strcmp(sp->s.name, name) == 0) {
204219b2ee8SDavid du Colombier 				switch(sp->s.type) {
2053e12c5d1SDavid du Colombier 				case 'T':
2063e12c5d1SDavid du Colombier 				case 'D':
2073e12c5d1SDavid du Colombier 				case 'U':
2083e12c5d1SDavid du Colombier 					if (type == 'U') {
209219b2ee8SDavid du Colombier 						names[id] = &sp->s;
210bd389b36SDavid du Colombier 						return;
2113e12c5d1SDavid du Colombier 					}
2123e12c5d1SDavid du Colombier 					break;
2133e12c5d1SDavid du Colombier 				case 't':
2143e12c5d1SDavid du Colombier 				case 'd':
2153e12c5d1SDavid du Colombier 				case 'b':
2163e12c5d1SDavid du Colombier 					if (type == 'b') {
217219b2ee8SDavid du Colombier 						names[id] = &sp->s;
218bd389b36SDavid du Colombier 						return;
2193e12c5d1SDavid du Colombier 					}
2203e12c5d1SDavid du Colombier 					break;
2213e12c5d1SDavid du Colombier 				case 'a':
2223e12c5d1SDavid du Colombier 				case 'p':
223219b2ee8SDavid du Colombier 					if (islocal(type)) {
224219b2ee8SDavid du Colombier 						names[id] = &sp->s;
225bd389b36SDavid du Colombier 						return;
2263e12c5d1SDavid du Colombier 					}
2273e12c5d1SDavid du Colombier 					break;
2283e12c5d1SDavid du Colombier 				default:
2293e12c5d1SDavid du Colombier 					break;
2303e12c5d1SDavid du Colombier 				}
231bd389b36SDavid du Colombier 			}
232bd389b36SDavid du Colombier 	}
233219b2ee8SDavid du Colombier 	sp = malloc(sizeof(Symtab));
234219b2ee8SDavid du Colombier 	sp->s.name = name;
235219b2ee8SDavid du Colombier 	sp->s.type = type;
2360c547597SDavid du Colombier 	sp->s.sig = sig;
237219b2ee8SDavid du Colombier 	sp->s.value = islocal(type) ? MAXOFF : 0;
238219b2ee8SDavid du Colombier 	names[id] = &sp->s;
239219b2ee8SDavid du Colombier 	sp->next = hash[h];
240219b2ee8SDavid du Colombier 	hash[h] = sp;
241bd389b36SDavid du Colombier 	return;
242bd389b36SDavid du Colombier }
243bd389b36SDavid du Colombier /*
244219b2ee8SDavid du Colombier  *	traverse the symbol lists
245bd389b36SDavid du Colombier  */
246219b2ee8SDavid du Colombier void
objtraverse(void (* fn)(Sym *,void *),void * pointer)247219b2ee8SDavid du Colombier objtraverse(void (*fn)(Sym*, void*), void *pointer)
248bd389b36SDavid du Colombier {
249219b2ee8SDavid du Colombier 	int i;
250219b2ee8SDavid du Colombier 	Symtab *s;
251219b2ee8SDavid du Colombier 
252219b2ee8SDavid du Colombier 	for(i = 0; i < NHASH; i++)
253219b2ee8SDavid du Colombier 		for(s = hash[i]; s; s = s->next)
254219b2ee8SDavid du Colombier 			(*fn)(&s->s, pointer);
2553e12c5d1SDavid du Colombier }
2563e12c5d1SDavid du Colombier 
2573e12c5d1SDavid du Colombier /*
2583e12c5d1SDavid du Colombier  * update the offset information for a 'a' or 'p' symbol in an intermediate file
2593e12c5d1SDavid du Colombier  */
2603e12c5d1SDavid du Colombier void
_offset(int id,vlong off)2614de34a7eSDavid du Colombier _offset(int id, vlong off)
2623e12c5d1SDavid du Colombier {
2633e12c5d1SDavid du Colombier 	Sym *s;
2643e12c5d1SDavid du Colombier 
265219b2ee8SDavid du Colombier 	s = names[id];
266219b2ee8SDavid du Colombier 	if (s && s->name[0] && islocal(s->type) && s->value > off)
2673e12c5d1SDavid du Colombier 		s->value = off;
2683e12c5d1SDavid du Colombier }
2693e12c5d1SDavid du Colombier 
2703e12c5d1SDavid du Colombier /*
2713e12c5d1SDavid du Colombier  * update the type of a global text or data symbol
2723e12c5d1SDavid du Colombier  */
273219b2ee8SDavid du Colombier static void
objupdate(int id,int type)274bd389b36SDavid du Colombier objupdate(int id, int type)
2753e12c5d1SDavid du Colombier {
2763e12c5d1SDavid du Colombier 	Sym *s;
2773e12c5d1SDavid du Colombier 
278219b2ee8SDavid du Colombier 	s = names[id];
279219b2ee8SDavid du Colombier 	if (s && s->name[0])
280219b2ee8SDavid du Colombier 		if (s->type == 'U')
2813e12c5d1SDavid du Colombier 			s->type = type;
282219b2ee8SDavid du Colombier 		else if (s->type == 'b')
2833e12c5d1SDavid du Colombier 			s->type = tolower(type);
2843e12c5d1SDavid du Colombier }
2853e12c5d1SDavid du Colombier 
2863e12c5d1SDavid du Colombier /*
2873e12c5d1SDavid du Colombier  * look for the next file in an archive
2883e12c5d1SDavid du Colombier  */
2893e12c5d1SDavid du Colombier int
nextar(Biobuf * bp,int offset,char * buf)290bd389b36SDavid du Colombier nextar(Biobuf *bp, int offset, char *buf)
2913e12c5d1SDavid du Colombier {
2923e12c5d1SDavid du Colombier 	struct ar_hdr a;
2933e12c5d1SDavid du Colombier 	int i, r;
2943e12c5d1SDavid du Colombier 	long arsize;
295bd389b36SDavid du Colombier 
296bd389b36SDavid du Colombier 	if (offset&01)
297bd389b36SDavid du Colombier 		offset++;
298bd389b36SDavid du Colombier 	Bseek(bp, offset, 0);
299bd389b36SDavid du Colombier 	r = Bread(bp, &a, SAR_HDR);
3003e12c5d1SDavid du Colombier 	if(r != SAR_HDR)
3013e12c5d1SDavid du Colombier 		return 0;
302bd389b36SDavid du Colombier 	if(strncmp(a.fmag, ARFMAG, sizeof(a.fmag)))
303bd389b36SDavid du Colombier 		return -1;
304219b2ee8SDavid du Colombier 	for(i=0; i<sizeof(a.name) && i<SARNAME && a.name[i] != ' '; i++)
305bd389b36SDavid du Colombier 		buf[i] = a.name[i];
306bd389b36SDavid du Colombier 	buf[i] = 0;
3074de34a7eSDavid du Colombier 	arsize = strtol(a.size, 0, 0);
308bd389b36SDavid du Colombier 	if (arsize&1)
309bd389b36SDavid du Colombier 		arsize++;
310bd389b36SDavid du Colombier 	return arsize + SAR_HDR;
3113e12c5d1SDavid du Colombier }
3123e12c5d1SDavid du Colombier 
313bd389b36SDavid du Colombier static void
objreset(void)314bd389b36SDavid du Colombier objreset(void)
315bd389b36SDavid du Colombier {
316219b2ee8SDavid du Colombier 	int i;
317219b2ee8SDavid du Colombier 	Symtab *s, *n;
318219b2ee8SDavid du Colombier 
319219b2ee8SDavid du Colombier 	for(i = 0; i < NHASH; i++) {
320219b2ee8SDavid du Colombier 		for(s = hash[i]; s; s = n) {
321219b2ee8SDavid du Colombier 			n = s->next;
322219b2ee8SDavid du Colombier 			free(s->s.name);
323219b2ee8SDavid du Colombier 			free(s);
3243e12c5d1SDavid du Colombier 		}
325219b2ee8SDavid du Colombier 		hash[i] = 0;
326219b2ee8SDavid du Colombier 	}
327219b2ee8SDavid du Colombier 	memset(names, 0, sizeof names);
3283e12c5d1SDavid du Colombier }
329