xref: /plan9-contrib/sys/src/cmd/9l/obj.c (revision fbadb1c4d4463e58337ffb1ed396c9caee5d1889)
1*fbadb1c4SDavid du Colombier #define	EXTERN
2*fbadb1c4SDavid du Colombier #include	"l.h"
3*fbadb1c4SDavid du Colombier #include	<ar.h>
4*fbadb1c4SDavid du Colombier 
5*fbadb1c4SDavid du Colombier #ifndef	DEFAULT
6*fbadb1c4SDavid du Colombier #define	DEFAULT	'9'
7*fbadb1c4SDavid du Colombier #endif
8*fbadb1c4SDavid du Colombier 
9*fbadb1c4SDavid du Colombier char	*noname		= "<none>";
10*fbadb1c4SDavid du Colombier char	symname[]	= SYMDEF;
11*fbadb1c4SDavid du Colombier char	thechar		= '9';
12*fbadb1c4SDavid du Colombier char	*thestring 	= "power64";
13*fbadb1c4SDavid du Colombier 
14*fbadb1c4SDavid du Colombier char**	libdir;
15*fbadb1c4SDavid du Colombier int	nlibdir	= 0;
16*fbadb1c4SDavid du Colombier static	int	maxlibdir = 0;
17*fbadb1c4SDavid du Colombier 
18*fbadb1c4SDavid du Colombier /*
19*fbadb1c4SDavid du Colombier  * -H[0135] are unmaintained.
20*fbadb1c4SDavid du Colombier  *	-H0 -T0x200000 -R0		was boot	[?]
21*fbadb1c4SDavid du Colombier  *	-H1 -T4128 -R4096		was plan9 32-bit (q.out) format
22*fbadb1c4SDavid du Colombier  *	-H2 -T0x100028 -R0x100000	is plan9 format
23*fbadb1c4SDavid du Colombier  *	-H3 -T0x02010000 -D0x00001000	was raw
24*fbadb1c4SDavid du Colombier  *	-H5 -T0x80010000 -t0x10000	was ELF32, phys = 10000, vaddr = 0x8001... [suspect]
25*fbadb1c4SDavid du Colombier  *	-H9 -T0x100028 -R0x100000	is ELF64
26*fbadb1c4SDavid du Colombier  *	-H10 -T0x100028 -R0x100000	is bizarre IBM/A2 ELF64 variant
27*fbadb1c4SDavid du Colombier  */
28*fbadb1c4SDavid du Colombier 
29*fbadb1c4SDavid du Colombier void
usage(void)30*fbadb1c4SDavid du Colombier usage(void)
31*fbadb1c4SDavid du Colombier {
32*fbadb1c4SDavid du Colombier 	diag("usage: %s [-options] objects", argv0);
33*fbadb1c4SDavid du Colombier 	errorexit();
34*fbadb1c4SDavid du Colombier }
35*fbadb1c4SDavid du Colombier 
36*fbadb1c4SDavid du Colombier static int
isobjfile(char * f)37*fbadb1c4SDavid du Colombier isobjfile(char *f)
38*fbadb1c4SDavid du Colombier {
39*fbadb1c4SDavid du Colombier 	int n, v;
40*fbadb1c4SDavid du Colombier 	Biobuf *b;
41*fbadb1c4SDavid du Colombier 	char buf1[5], buf2[SARMAG];
42*fbadb1c4SDavid du Colombier 
43*fbadb1c4SDavid du Colombier 	b = Bopen(f, OREAD);
44*fbadb1c4SDavid du Colombier 	if(b == nil)
45*fbadb1c4SDavid du Colombier 		return 0;
46*fbadb1c4SDavid du Colombier 	n = Bread(b, buf1, 5);
47*fbadb1c4SDavid du Colombier 	if(n == 5 && (buf1[2] == 1 && buf1[3] == '<' || buf1[3] == 1 && buf1[4] == '<'))
48*fbadb1c4SDavid du Colombier 		v = 1;	/* good enough for our purposes */
49*fbadb1c4SDavid du Colombier 	else{
50*fbadb1c4SDavid du Colombier 		Bseek(b, 0, 0);
51*fbadb1c4SDavid du Colombier 		n = Bread(b, buf2, SARMAG);
52*fbadb1c4SDavid du Colombier 		v = n == SARMAG && strncmp(buf2, ARMAG, SARMAG) == 0;
53*fbadb1c4SDavid du Colombier 	}
54*fbadb1c4SDavid du Colombier 	Bterm(b);
55*fbadb1c4SDavid du Colombier 	return v;
56*fbadb1c4SDavid du Colombier }
57*fbadb1c4SDavid du Colombier 
58*fbadb1c4SDavid du Colombier void
main(int argc,char * argv[])59*fbadb1c4SDavid du Colombier main(int argc, char *argv[])
60*fbadb1c4SDavid du Colombier {
61*fbadb1c4SDavid du Colombier 	int c;
62*fbadb1c4SDavid du Colombier 	char *a;
63*fbadb1c4SDavid du Colombier 	char name[LIBNAMELEN];
64*fbadb1c4SDavid du Colombier 
65*fbadb1c4SDavid du Colombier 	Binit(&bso, 1, OWRITE);
66*fbadb1c4SDavid du Colombier 	cout = -1;
67*fbadb1c4SDavid du Colombier 	listinit();
68*fbadb1c4SDavid du Colombier 	outfile = 0;
69*fbadb1c4SDavid du Colombier 	nerrors = 0;
70*fbadb1c4SDavid du Colombier 	curtext = P;
71*fbadb1c4SDavid du Colombier 	HEADTYPE = -1;
72*fbadb1c4SDavid du Colombier 	INITTEXT = -1;
73*fbadb1c4SDavid du Colombier 	INITTEXTP = -1;
74*fbadb1c4SDavid du Colombier 	INITDAT = -1;
75*fbadb1c4SDavid du Colombier 	INITRND = -1;
76*fbadb1c4SDavid du Colombier 	INITENTRY = 0;
77*fbadb1c4SDavid du Colombier 
78*fbadb1c4SDavid du Colombier 	ARGBEGIN {
79*fbadb1c4SDavid du Colombier 	default:
80*fbadb1c4SDavid du Colombier 		c = ARGC();
81*fbadb1c4SDavid du Colombier 		if(c >= 0 && c < sizeof(debug))
82*fbadb1c4SDavid du Colombier 			debug[c]++;
83*fbadb1c4SDavid du Colombier 		break;
84*fbadb1c4SDavid du Colombier 	case 'o':
85*fbadb1c4SDavid du Colombier 		outfile = ARGF();
86*fbadb1c4SDavid du Colombier 		break;
87*fbadb1c4SDavid du Colombier 	case 'E':
88*fbadb1c4SDavid du Colombier 		a = ARGF();
89*fbadb1c4SDavid du Colombier 		if(a)
90*fbadb1c4SDavid du Colombier 			INITENTRY = a;
91*fbadb1c4SDavid du Colombier 		break;
92*fbadb1c4SDavid du Colombier 	case 'L':
93*fbadb1c4SDavid du Colombier 		addlibpath(EARGF(usage()));
94*fbadb1c4SDavid du Colombier 		break;
95*fbadb1c4SDavid du Colombier 	case 'T':
96*fbadb1c4SDavid du Colombier 		a = ARGF();
97*fbadb1c4SDavid du Colombier 		if(a)
98*fbadb1c4SDavid du Colombier 			INITTEXT = atolwhex(a);
99*fbadb1c4SDavid du Colombier 		break;
100*fbadb1c4SDavid du Colombier 	case 'P':
101*fbadb1c4SDavid du Colombier 		a = ARGF();
102*fbadb1c4SDavid du Colombier 		if(a)
103*fbadb1c4SDavid du Colombier 			INITTEXTP = atolwhex(a);
104*fbadb1c4SDavid du Colombier 		break;
105*fbadb1c4SDavid du Colombier 	case 'D':
106*fbadb1c4SDavid du Colombier 		a = ARGF();
107*fbadb1c4SDavid du Colombier 		if(a)
108*fbadb1c4SDavid du Colombier 			INITDAT = atolwhex(a);
109*fbadb1c4SDavid du Colombier 		break;
110*fbadb1c4SDavid du Colombier 	case 'R':
111*fbadb1c4SDavid du Colombier 		a = ARGF();
112*fbadb1c4SDavid du Colombier 		if(a)
113*fbadb1c4SDavid du Colombier 			INITRND = atolwhex(a);
114*fbadb1c4SDavid du Colombier 		break;
115*fbadb1c4SDavid du Colombier 	case 'H':
116*fbadb1c4SDavid du Colombier 		a = ARGF();
117*fbadb1c4SDavid du Colombier 		if(a)
118*fbadb1c4SDavid du Colombier 			HEADTYPE = atolwhex(a);
119*fbadb1c4SDavid du Colombier 		break;
120*fbadb1c4SDavid du Colombier 	case 'x':	/* produce export table */
121*fbadb1c4SDavid du Colombier 		doexp = 1;
122*fbadb1c4SDavid du Colombier 		if(argv[1] != nil && argv[1][0] != '-' && !isobjfile(argv[1]))
123*fbadb1c4SDavid du Colombier 			readundefs(ARGF(), SEXPORT);
124*fbadb1c4SDavid du Colombier 		break;
125*fbadb1c4SDavid du Colombier 	case 'u':	/* produce dynamically loadable module */
126*fbadb1c4SDavid du Colombier 		dlm = 1;
127*fbadb1c4SDavid du Colombier 		if(argv[1] != nil && argv[1][0] != '-' && !isobjfile(argv[1]))
128*fbadb1c4SDavid du Colombier 			readundefs(ARGF(), SIMPORT);
129*fbadb1c4SDavid du Colombier 		break;
130*fbadb1c4SDavid du Colombier 	} ARGEND
131*fbadb1c4SDavid du Colombier 	USED(argc);
132*fbadb1c4SDavid du Colombier 	if(*argv == 0)
133*fbadb1c4SDavid du Colombier 		usage();
134*fbadb1c4SDavid du Colombier 	if(!debug['9'] && !debug['U'] && !debug['B'])
135*fbadb1c4SDavid du Colombier 		debug[DEFAULT] = 1;
136*fbadb1c4SDavid du Colombier 	a = getenv("ccroot");
137*fbadb1c4SDavid du Colombier 	if(a != nil && *a != '\0') {
138*fbadb1c4SDavid du Colombier 		if(!fileexists(a)) {
139*fbadb1c4SDavid du Colombier 			diag("nonexistent $ccroot: %s", a);
140*fbadb1c4SDavid du Colombier 			errorexit();
141*fbadb1c4SDavid du Colombier 		}
142*fbadb1c4SDavid du Colombier 	}else
143*fbadb1c4SDavid du Colombier 		a = "";
144*fbadb1c4SDavid du Colombier 	snprint(name, sizeof(name), "%s/%s/lib", a, thestring);
145*fbadb1c4SDavid du Colombier 	addlibpath(name);
146*fbadb1c4SDavid du Colombier 	r0iszero = debug['0'] == 0;
147*fbadb1c4SDavid du Colombier 	if(HEADTYPE == -1) {
148*fbadb1c4SDavid du Colombier 		if(debug['U'])
149*fbadb1c4SDavid du Colombier 			HEADTYPE = 0;
150*fbadb1c4SDavid du Colombier 		if(debug['B'])
151*fbadb1c4SDavid du Colombier 			HEADTYPE = 1;
152*fbadb1c4SDavid du Colombier 		if(debug['9'])
153*fbadb1c4SDavid du Colombier 			HEADTYPE = 2;
154*fbadb1c4SDavid du Colombier 	}
155*fbadb1c4SDavid du Colombier 	switch(HEADTYPE) {
156*fbadb1c4SDavid du Colombier 	default:
157*fbadb1c4SDavid du Colombier 		diag("unknown -H option");
158*fbadb1c4SDavid du Colombier 		errorexit();
159*fbadb1c4SDavid du Colombier 
160*fbadb1c4SDavid du Colombier 	case 0:	/* boot */
161*fbadb1c4SDavid du Colombier 		HEADR = 32L;
162*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
163*fbadb1c4SDavid du Colombier 			INITTEXT = 0x200000L;
164*fbadb1c4SDavid du Colombier 		if(INITDAT == -1)
165*fbadb1c4SDavid du Colombier 			INITDAT = 0;
166*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
167*fbadb1c4SDavid du Colombier 			INITRND = 4096L;
168*fbadb1c4SDavid du Colombier 		break;
169*fbadb1c4SDavid du Colombier 	case 1:	/* plan 9 q.out */
170*fbadb1c4SDavid du Colombier 		HEADR = 32L;
171*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
172*fbadb1c4SDavid du Colombier 			INITTEXT = 4128;
173*fbadb1c4SDavid du Colombier 		if(INITDAT == -1)
174*fbadb1c4SDavid du Colombier 			INITDAT = 0;
175*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
176*fbadb1c4SDavid du Colombier 			INITRND = 4096;
177*fbadb1c4SDavid du Colombier 		break;
178*fbadb1c4SDavid du Colombier 	case 2:	/* plan 9 */
179*fbadb1c4SDavid du Colombier 		HEADR = 32L+8L;
180*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
181*fbadb1c4SDavid du Colombier 			INITTEXT = 0x100000+HEADR;
182*fbadb1c4SDavid du Colombier 		if(INITDAT == -1)
183*fbadb1c4SDavid du Colombier 			INITDAT = 0;
184*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
185*fbadb1c4SDavid du Colombier 			INITRND = 0x100000;
186*fbadb1c4SDavid du Colombier 		break;
187*fbadb1c4SDavid du Colombier 	case 3:	/* raw */
188*fbadb1c4SDavid du Colombier 		HEADR = 0;
189*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
190*fbadb1c4SDavid du Colombier 			INITTEXT = 4128;
191*fbadb1c4SDavid du Colombier 		if(INITDAT == -1) {
192*fbadb1c4SDavid du Colombier 			INITDAT = 0;
193*fbadb1c4SDavid du Colombier 			INITRND = 4;
194*fbadb1c4SDavid du Colombier 		}
195*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
196*fbadb1c4SDavid du Colombier 			INITRND = 0;
197*fbadb1c4SDavid du Colombier 		break;
198*fbadb1c4SDavid du Colombier 	case 5:	/* elf executable */
199*fbadb1c4SDavid du Colombier 		HEADR = rnd(Ehdr32sz+3*Phdr32sz, 16);
200*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
201*fbadb1c4SDavid du Colombier 			INITTEXT = 0x00400000L+HEADR;
202*fbadb1c4SDavid du Colombier 		if(INITDAT == -1)
203*fbadb1c4SDavid du Colombier 			INITDAT = 0x10000000;
204*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
205*fbadb1c4SDavid du Colombier 			INITRND = 0;
206*fbadb1c4SDavid du Colombier 		break;
207*fbadb1c4SDavid du Colombier 	case 9:	/* ELF64 executable */
208*fbadb1c4SDavid du Colombier 	case 10:
209*fbadb1c4SDavid du Colombier 		HEADR = rnd(Ehdr64sz+3*Phdr64sz, 16);
210*fbadb1c4SDavid du Colombier 		if(INITTEXT == -1)
211*fbadb1c4SDavid du Colombier 			INITTEXT = 0x100000+HEADR;
212*fbadb1c4SDavid du Colombier 		if(INITDAT == -1)
213*fbadb1c4SDavid du Colombier 			INITDAT = 0;
214*fbadb1c4SDavid du Colombier 		if(INITRND == -1)
215*fbadb1c4SDavid du Colombier 			INITRND = 0x100000;
216*fbadb1c4SDavid du Colombier 		break;
217*fbadb1c4SDavid du Colombier 	}
218*fbadb1c4SDavid du Colombier 	if (INITTEXTP == -1)
219*fbadb1c4SDavid du Colombier 		INITTEXTP = INITTEXT;
220*fbadb1c4SDavid du Colombier 	if(INITDAT != 0 && INITRND != 0)
221*fbadb1c4SDavid du Colombier 		print("warning: -D0x%llux is ignored because of -R0x%lux\n",
222*fbadb1c4SDavid du Colombier 			INITDAT, INITRND);
223*fbadb1c4SDavid du Colombier 	if(debug['v'])
224*fbadb1c4SDavid du Colombier 		Bprint(&bso, "HEADER = -H0x%x -T0x%llux -D0x%llux -R0x%lux\n",
225*fbadb1c4SDavid du Colombier 			HEADTYPE, INITTEXT, INITDAT, INITRND);
226*fbadb1c4SDavid du Colombier 	Bflush(&bso);
227*fbadb1c4SDavid du Colombier 	zprg.as = AGOK;
228*fbadb1c4SDavid du Colombier 	zprg.reg = NREG;
229*fbadb1c4SDavid du Colombier 	zprg.from.name = D_NONE;
230*fbadb1c4SDavid du Colombier 	zprg.from.type = D_NONE;
231*fbadb1c4SDavid du Colombier 	zprg.from.reg = NREG;
232*fbadb1c4SDavid du Colombier 	zprg.from3 = zprg.from;
233*fbadb1c4SDavid du Colombier 	zprg.to = zprg.from;
234*fbadb1c4SDavid du Colombier 	buildop();
235*fbadb1c4SDavid du Colombier 	histgen = 0;
236*fbadb1c4SDavid du Colombier 	textp = P;
237*fbadb1c4SDavid du Colombier 	datap = P;
238*fbadb1c4SDavid du Colombier 	pc = 0;
239*fbadb1c4SDavid du Colombier 	dtype = 4;
240*fbadb1c4SDavid du Colombier 	if(outfile == 0)
241*fbadb1c4SDavid du Colombier 		outfile = "9.out";
242*fbadb1c4SDavid du Colombier 	cout = create(outfile, 1, 0775);
243*fbadb1c4SDavid du Colombier 	if(cout < 0) {
244*fbadb1c4SDavid du Colombier 		diag("cannot create %s: %r", outfile);
245*fbadb1c4SDavid du Colombier 		errorexit();
246*fbadb1c4SDavid du Colombier 	}
247*fbadb1c4SDavid du Colombier 	nuxiinit();
248*fbadb1c4SDavid du Colombier 	version = 0;
249*fbadb1c4SDavid du Colombier 	cbp = buf.cbuf;
250*fbadb1c4SDavid du Colombier 	cbc = sizeof(buf.cbuf);
251*fbadb1c4SDavid du Colombier 	firstp = prg();
252*fbadb1c4SDavid du Colombier 	lastp = firstp;
253*fbadb1c4SDavid du Colombier 
254*fbadb1c4SDavid du Colombier 	if(INITENTRY == 0) {
255*fbadb1c4SDavid du Colombier 		INITENTRY = "_main";
256*fbadb1c4SDavid du Colombier 		if(debug['p'])
257*fbadb1c4SDavid du Colombier 			INITENTRY = "_mainp";
258*fbadb1c4SDavid du Colombier 		if(!debug['l'])
259*fbadb1c4SDavid du Colombier 			lookup(INITENTRY, 0)->type = SXREF;
260*fbadb1c4SDavid du Colombier 	} else {
261*fbadb1c4SDavid du Colombier 		/*
262*fbadb1c4SDavid du Colombier 		 * It's possibly a number handed in by -En (this is
263*fbadb1c4SDavid du Colombier 		 * the test made by entryvalue() when creating the appropriate
264*fbadb1c4SDavid du Colombier 		 * header). Calling lookup() on that will create the symbol 'n'
265*fbadb1c4SDavid du Colombier 		 * printing an undef() diagnostic later.
266*fbadb1c4SDavid du Colombier 		 */
267*fbadb1c4SDavid du Colombier 		a = INITENTRY;
268*fbadb1c4SDavid du Colombier 		if(!(*a >= '0' && *a <= '9'))
269*fbadb1c4SDavid du Colombier 			lookup(INITENTRY, 0)->type = SXREF;
270*fbadb1c4SDavid du Colombier 	}
271*fbadb1c4SDavid du Colombier 
272*fbadb1c4SDavid du Colombier 	while(*argv)
273*fbadb1c4SDavid du Colombier 		objfile(*argv++);
274*fbadb1c4SDavid du Colombier 	if(!debug['l'])
275*fbadb1c4SDavid du Colombier 		loadlib();
276*fbadb1c4SDavid du Colombier 	firstp = firstp->link;
277*fbadb1c4SDavid du Colombier 	if(firstp == P)
278*fbadb1c4SDavid du Colombier 		goto out;
279*fbadb1c4SDavid du Colombier 	if(doexp || dlm){
280*fbadb1c4SDavid du Colombier 		EXPTAB = "_exporttab";
281*fbadb1c4SDavid du Colombier 		zerosig(EXPTAB);
282*fbadb1c4SDavid du Colombier 		zerosig("etext");
283*fbadb1c4SDavid du Colombier 		zerosig("edata");
284*fbadb1c4SDavid du Colombier 		zerosig("end");
285*fbadb1c4SDavid du Colombier 		if(dlm){
286*fbadb1c4SDavid du Colombier 			import();
287*fbadb1c4SDavid du Colombier 			HEADTYPE = 2;
288*fbadb1c4SDavid du Colombier 			INITTEXT = 0;
289*fbadb1c4SDavid du Colombier 			INITDAT = 0;
290*fbadb1c4SDavid du Colombier 			INITRND = 8;
291*fbadb1c4SDavid du Colombier 			INITENTRY = EXPTAB;
292*fbadb1c4SDavid du Colombier 		}
293*fbadb1c4SDavid du Colombier 		export();
294*fbadb1c4SDavid du Colombier 	}
295*fbadb1c4SDavid du Colombier 	patch();
296*fbadb1c4SDavid du Colombier 	if(debug['p'])
297*fbadb1c4SDavid du Colombier 		if(debug['1'])
298*fbadb1c4SDavid du Colombier 			doprof1();
299*fbadb1c4SDavid du Colombier 		else
300*fbadb1c4SDavid du Colombier 			doprof2();
301*fbadb1c4SDavid du Colombier 	dodata();
302*fbadb1c4SDavid du Colombier 	follow();
303*fbadb1c4SDavid du Colombier 	if(firstp == P)
304*fbadb1c4SDavid du Colombier 		goto out;
305*fbadb1c4SDavid du Colombier 	noops();
306*fbadb1c4SDavid du Colombier 	span();
307*fbadb1c4SDavid du Colombier 	asmb();
308*fbadb1c4SDavid du Colombier 	undef();
309*fbadb1c4SDavid du Colombier 
310*fbadb1c4SDavid du Colombier out:
311*fbadb1c4SDavid du Colombier 	if(debug['v']) {
312*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%5.2f cpu time\n", cputime());
313*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%ld memory used\n", tothunk);
314*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%d sizeof adr\n", sizeof(Adr));
315*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%d sizeof prog\n", sizeof(Prog));
316*fbadb1c4SDavid du Colombier 	}
317*fbadb1c4SDavid du Colombier 	errorexit();
318*fbadb1c4SDavid du Colombier }
319*fbadb1c4SDavid du Colombier 
320*fbadb1c4SDavid du Colombier void
addlibpath(char * arg)321*fbadb1c4SDavid du Colombier addlibpath(char *arg)
322*fbadb1c4SDavid du Colombier {
323*fbadb1c4SDavid du Colombier 	char **p;
324*fbadb1c4SDavid du Colombier 
325*fbadb1c4SDavid du Colombier 	if(nlibdir >= maxlibdir) {
326*fbadb1c4SDavid du Colombier 		if(maxlibdir == 0)
327*fbadb1c4SDavid du Colombier 			maxlibdir = 8;
328*fbadb1c4SDavid du Colombier 		else
329*fbadb1c4SDavid du Colombier 			maxlibdir *= 2;
330*fbadb1c4SDavid du Colombier 		p = malloc(maxlibdir*sizeof(*p));
331*fbadb1c4SDavid du Colombier 		if(p == nil) {
332*fbadb1c4SDavid du Colombier 			diag("out of memory");
333*fbadb1c4SDavid du Colombier 			errorexit();
334*fbadb1c4SDavid du Colombier 		}
335*fbadb1c4SDavid du Colombier 		memmove(p, libdir, nlibdir*sizeof(*p));
336*fbadb1c4SDavid du Colombier 		free(libdir);
337*fbadb1c4SDavid du Colombier 		libdir = p;
338*fbadb1c4SDavid du Colombier 	}
339*fbadb1c4SDavid du Colombier 	libdir[nlibdir++] = strdup(arg);
340*fbadb1c4SDavid du Colombier }
341*fbadb1c4SDavid du Colombier 
342*fbadb1c4SDavid du Colombier char*
findlib(char * file)343*fbadb1c4SDavid du Colombier findlib(char *file)
344*fbadb1c4SDavid du Colombier {
345*fbadb1c4SDavid du Colombier 	int i;
346*fbadb1c4SDavid du Colombier 	char name[LIBNAMELEN];
347*fbadb1c4SDavid du Colombier 
348*fbadb1c4SDavid du Colombier 	for(i = 0; i < nlibdir; i++) {
349*fbadb1c4SDavid du Colombier 		snprint(name, sizeof(name), "%s/%s", libdir[i], file);
350*fbadb1c4SDavid du Colombier 		if(fileexists(name))
351*fbadb1c4SDavid du Colombier 			return libdir[i];
352*fbadb1c4SDavid du Colombier 	}
353*fbadb1c4SDavid du Colombier 	return nil;
354*fbadb1c4SDavid du Colombier }
355*fbadb1c4SDavid du Colombier 
356*fbadb1c4SDavid du Colombier void
loadlib(void)357*fbadb1c4SDavid du Colombier loadlib(void)
358*fbadb1c4SDavid du Colombier {
359*fbadb1c4SDavid du Colombier 	int i;
360*fbadb1c4SDavid du Colombier 	long h;
361*fbadb1c4SDavid du Colombier 	Sym *s;
362*fbadb1c4SDavid du Colombier 
363*fbadb1c4SDavid du Colombier loop:
364*fbadb1c4SDavid du Colombier 	xrefresolv = 0;
365*fbadb1c4SDavid du Colombier 	for(i=0; i<libraryp; i++) {
366*fbadb1c4SDavid du Colombier 		if(debug['v'])
367*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%5.2f autolib: %s (from %s)\n", cputime(), library[i], libraryobj[i]);
368*fbadb1c4SDavid du Colombier 		objfile(library[i]);
369*fbadb1c4SDavid du Colombier 	}
370*fbadb1c4SDavid du Colombier 	if(xrefresolv)
371*fbadb1c4SDavid du Colombier 	for(h=0; h<nelem(hash); h++)
372*fbadb1c4SDavid du Colombier 	for(s = hash[h]; s != S; s = s->link)
373*fbadb1c4SDavid du Colombier 		if(s->type == SXREF)
374*fbadb1c4SDavid du Colombier 			goto loop;
375*fbadb1c4SDavid du Colombier }
376*fbadb1c4SDavid du Colombier 
377*fbadb1c4SDavid du Colombier void
errorexit(void)378*fbadb1c4SDavid du Colombier errorexit(void)
379*fbadb1c4SDavid du Colombier {
380*fbadb1c4SDavid du Colombier 
381*fbadb1c4SDavid du Colombier 	Bflush(&bso);
382*fbadb1c4SDavid du Colombier 	if(nerrors) {
383*fbadb1c4SDavid du Colombier 		if(cout >= 0)
384*fbadb1c4SDavid du Colombier 			remove(outfile);
385*fbadb1c4SDavid du Colombier 		exits("error");
386*fbadb1c4SDavid du Colombier 	}
387*fbadb1c4SDavid du Colombier 	exits(0);
388*fbadb1c4SDavid du Colombier }
389*fbadb1c4SDavid du Colombier 
390*fbadb1c4SDavid du Colombier void
objfile(char * file)391*fbadb1c4SDavid du Colombier objfile(char *file)
392*fbadb1c4SDavid du Colombier {
393*fbadb1c4SDavid du Colombier 	long off, esym, cnt, l;
394*fbadb1c4SDavid du Colombier 	int f, work;
395*fbadb1c4SDavid du Colombier 	Sym *s;
396*fbadb1c4SDavid du Colombier 	char magbuf[SARMAG];
397*fbadb1c4SDavid du Colombier 	char name[LIBNAMELEN], pname[LIBNAMELEN];
398*fbadb1c4SDavid du Colombier 	struct ar_hdr arhdr;
399*fbadb1c4SDavid du Colombier 	char *e, *start, *stop;
400*fbadb1c4SDavid du Colombier 
401*fbadb1c4SDavid du Colombier 	if(debug['v'])
402*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%5.2f ldobj: %s\n", cputime(), file);
403*fbadb1c4SDavid du Colombier 	Bflush(&bso);
404*fbadb1c4SDavid du Colombier 	if(file[0] == '-' && file[1] == 'l') {
405*fbadb1c4SDavid du Colombier 		snprint(pname, sizeof(pname), "lib%s.a", file+2);
406*fbadb1c4SDavid du Colombier 		e = findlib(pname);
407*fbadb1c4SDavid du Colombier 		if(e == nil) {
408*fbadb1c4SDavid du Colombier 			diag("cannot find library: %s", file);
409*fbadb1c4SDavid du Colombier 			errorexit();
410*fbadb1c4SDavid du Colombier 		}
411*fbadb1c4SDavid du Colombier 		snprint(name, sizeof(name), "%s/%s", e, pname);
412*fbadb1c4SDavid du Colombier 		file = name;
413*fbadb1c4SDavid du Colombier 	}
414*fbadb1c4SDavid du Colombier 	f = open(file, 0);
415*fbadb1c4SDavid du Colombier 	if(f < 0) {
416*fbadb1c4SDavid du Colombier 		diag("cannot open %s: %r", file);
417*fbadb1c4SDavid du Colombier 		errorexit();
418*fbadb1c4SDavid du Colombier 	}
419*fbadb1c4SDavid du Colombier 	l = read(f, magbuf, SARMAG);
420*fbadb1c4SDavid du Colombier 	if(l != SARMAG || strncmp(magbuf, ARMAG, SARMAG)){
421*fbadb1c4SDavid du Colombier 		/* load it as a regular file */
422*fbadb1c4SDavid du Colombier 		l = seek(f, 0L, 2);
423*fbadb1c4SDavid du Colombier 		seek(f, 0L, 0);
424*fbadb1c4SDavid du Colombier 		ldobj(f, l, file);
425*fbadb1c4SDavid du Colombier 		close(f);
426*fbadb1c4SDavid du Colombier 		return;
427*fbadb1c4SDavid du Colombier 	}
428*fbadb1c4SDavid du Colombier 
429*fbadb1c4SDavid du Colombier 	l = read(f, &arhdr, SAR_HDR);
430*fbadb1c4SDavid du Colombier 	if(l != SAR_HDR) {
431*fbadb1c4SDavid du Colombier 		diag("%s: short read on archive file symbol header", file);
432*fbadb1c4SDavid du Colombier 		goto out;
433*fbadb1c4SDavid du Colombier 	}
434*fbadb1c4SDavid du Colombier 	if(strncmp(arhdr.name, symname, strlen(symname))) {
435*fbadb1c4SDavid du Colombier 		diag("%s: first entry not symbol header", file);
436*fbadb1c4SDavid du Colombier 		goto out;
437*fbadb1c4SDavid du Colombier 	}
438*fbadb1c4SDavid du Colombier 
439*fbadb1c4SDavid du Colombier 	esym = SARMAG + SAR_HDR + atolwhex(arhdr.size);
440*fbadb1c4SDavid du Colombier 	off = SARMAG + SAR_HDR;
441*fbadb1c4SDavid du Colombier 
442*fbadb1c4SDavid du Colombier 	/*
443*fbadb1c4SDavid du Colombier 	 * just bang the whole symbol file into memory
444*fbadb1c4SDavid du Colombier 	 */
445*fbadb1c4SDavid du Colombier 	seek(f, off, 0);
446*fbadb1c4SDavid du Colombier 	cnt = esym - off;
447*fbadb1c4SDavid du Colombier 	start = malloc(cnt + 10);
448*fbadb1c4SDavid du Colombier 	cnt = read(f, start, cnt);
449*fbadb1c4SDavid du Colombier 	if(cnt <= 0){
450*fbadb1c4SDavid du Colombier 		close(f);
451*fbadb1c4SDavid du Colombier 		return;
452*fbadb1c4SDavid du Colombier 	}
453*fbadb1c4SDavid du Colombier 	stop = &start[cnt];
454*fbadb1c4SDavid du Colombier 	memset(stop, 0, 10);
455*fbadb1c4SDavid du Colombier 
456*fbadb1c4SDavid du Colombier 	work = 1;
457*fbadb1c4SDavid du Colombier 	while(work){
458*fbadb1c4SDavid du Colombier 		if(debug['v'])
459*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%5.2f library pass: %s\n", cputime(), file);
460*fbadb1c4SDavid du Colombier 		Bflush(&bso);
461*fbadb1c4SDavid du Colombier 		work = 0;
462*fbadb1c4SDavid du Colombier 		for(e = start; e < stop; e = strchr(e+5, 0) + 1) {
463*fbadb1c4SDavid du Colombier 			s = lookup(e+5, 0);
464*fbadb1c4SDavid du Colombier 			if(s->type != SXREF)
465*fbadb1c4SDavid du Colombier 				continue;
466*fbadb1c4SDavid du Colombier 			sprint(pname, "%s(%s)", file, s->name);
467*fbadb1c4SDavid du Colombier 			if(debug['v'])
468*fbadb1c4SDavid du Colombier 				Bprint(&bso, "%5.2f library: %s\n", cputime(), pname);
469*fbadb1c4SDavid du Colombier 			Bflush(&bso);
470*fbadb1c4SDavid du Colombier 			l = e[1] & 0xff;
471*fbadb1c4SDavid du Colombier 			l |= (e[2] & 0xff) << 8;
472*fbadb1c4SDavid du Colombier 			l |= (e[3] & 0xff) << 16;
473*fbadb1c4SDavid du Colombier 			l |= (e[4] & 0xff) << 24;
474*fbadb1c4SDavid du Colombier 			seek(f, l, 0);
475*fbadb1c4SDavid du Colombier 			/* need readn to read the dumps (at least) */
476*fbadb1c4SDavid du Colombier 			l = readn(f, &arhdr, SAR_HDR);
477*fbadb1c4SDavid du Colombier 			if(l != SAR_HDR)
478*fbadb1c4SDavid du Colombier 				goto bad;
479*fbadb1c4SDavid du Colombier 			if(strncmp(arhdr.fmag, ARFMAG, sizeof(arhdr.fmag)))
480*fbadb1c4SDavid du Colombier 				goto bad;
481*fbadb1c4SDavid du Colombier 			l = atolwhex(arhdr.size);
482*fbadb1c4SDavid du Colombier 			ldobj(f, l, pname);
483*fbadb1c4SDavid du Colombier 			if(s->type == SXREF) {
484*fbadb1c4SDavid du Colombier 				diag("%s: failed to load: %s", file, s->name);
485*fbadb1c4SDavid du Colombier 				errorexit();
486*fbadb1c4SDavid du Colombier 			}
487*fbadb1c4SDavid du Colombier 			work = 1;
488*fbadb1c4SDavid du Colombier 			xrefresolv = 1;
489*fbadb1c4SDavid du Colombier 		}
490*fbadb1c4SDavid du Colombier 	}
491*fbadb1c4SDavid du Colombier 	return;
492*fbadb1c4SDavid du Colombier 
493*fbadb1c4SDavid du Colombier bad:
494*fbadb1c4SDavid du Colombier 	diag("%s: bad or out of date archive", file);
495*fbadb1c4SDavid du Colombier out:
496*fbadb1c4SDavid du Colombier 	close(f);
497*fbadb1c4SDavid du Colombier }
498*fbadb1c4SDavid du Colombier 
499*fbadb1c4SDavid du Colombier int
zaddr(uchar * p,Adr * a,Sym * h[])500*fbadb1c4SDavid du Colombier zaddr(uchar *p, Adr *a, Sym *h[])
501*fbadb1c4SDavid du Colombier {
502*fbadb1c4SDavid du Colombier 	int i, c;
503*fbadb1c4SDavid du Colombier 	int l;
504*fbadb1c4SDavid du Colombier 	Sym *s;
505*fbadb1c4SDavid du Colombier 	Auto *u;
506*fbadb1c4SDavid du Colombier 
507*fbadb1c4SDavid du Colombier 	c = p[2];
508*fbadb1c4SDavid du Colombier 	if(c < 0 || c > NSYM){
509*fbadb1c4SDavid du Colombier 		print("sym out of range: %d\n", c);
510*fbadb1c4SDavid du Colombier 		p[0] = AEND+1;
511*fbadb1c4SDavid du Colombier 		return 0;
512*fbadb1c4SDavid du Colombier 	}
513*fbadb1c4SDavid du Colombier 	a->type = p[0];
514*fbadb1c4SDavid du Colombier 	a->reg = p[1];
515*fbadb1c4SDavid du Colombier 	a->sym = h[c];
516*fbadb1c4SDavid du Colombier 	a->name = p[3];
517*fbadb1c4SDavid du Colombier 	c = 4;
518*fbadb1c4SDavid du Colombier 
519*fbadb1c4SDavid du Colombier 	if(a->reg > NREG) {
520*fbadb1c4SDavid du Colombier 		print("register out of range %d\n", a->reg);
521*fbadb1c4SDavid du Colombier 		p[0] = AEND+1;
522*fbadb1c4SDavid du Colombier 		return 0;	/*  force real diagnostic */
523*fbadb1c4SDavid du Colombier 	}
524*fbadb1c4SDavid du Colombier 
525*fbadb1c4SDavid du Colombier 	switch(a->type) {
526*fbadb1c4SDavid du Colombier 	default:
527*fbadb1c4SDavid du Colombier 		print("unknown type %d\n", a->type);
528*fbadb1c4SDavid du Colombier 		p[0] = AEND+1;
529*fbadb1c4SDavid du Colombier 		return 0;	/* force real diagnostic */
530*fbadb1c4SDavid du Colombier 
531*fbadb1c4SDavid du Colombier 	case D_NONE:
532*fbadb1c4SDavid du Colombier 	case D_REG:
533*fbadb1c4SDavid du Colombier 	case D_FREG:
534*fbadb1c4SDavid du Colombier 	case D_CREG:
535*fbadb1c4SDavid du Colombier 	case D_FPSCR:
536*fbadb1c4SDavid du Colombier 	case D_MSR:
537*fbadb1c4SDavid du Colombier 	case D_OPT:
538*fbadb1c4SDavid du Colombier 		break;
539*fbadb1c4SDavid du Colombier 
540*fbadb1c4SDavid du Colombier 	case D_SPR:
541*fbadb1c4SDavid du Colombier 	case D_DCR:
542*fbadb1c4SDavid du Colombier 	case D_BRANCH:
543*fbadb1c4SDavid du Colombier 	case D_OREG:
544*fbadb1c4SDavid du Colombier 	case D_CONST:
545*fbadb1c4SDavid du Colombier 		l = p[4] | (p[5]<<8) |
546*fbadb1c4SDavid du Colombier 			(p[6]<<16) | (p[7]<<24);
547*fbadb1c4SDavid du Colombier 		a->offset = l;
548*fbadb1c4SDavid du Colombier 		c += 4;
549*fbadb1c4SDavid du Colombier 		break;
550*fbadb1c4SDavid du Colombier 
551*fbadb1c4SDavid du Colombier 	case D_DCONST:
552*fbadb1c4SDavid du Colombier 		l = p[4] | (p[5]<<8) | (p[6]<<16) | (p[7]<<24);
553*fbadb1c4SDavid du Colombier 		a->offset = (uvlong)l & 0xFFFFFFFFUL;
554*fbadb1c4SDavid du Colombier 		l = p[8] | (p[9]<<8) | (p[10]<<16) | (p[11]<<24);
555*fbadb1c4SDavid du Colombier 		a->offset |= (vlong)l << 32;
556*fbadb1c4SDavid du Colombier 		c += 8;
557*fbadb1c4SDavid du Colombier 		a->type = D_CONST;
558*fbadb1c4SDavid du Colombier 		break;
559*fbadb1c4SDavid du Colombier 
560*fbadb1c4SDavid du Colombier 	case D_SCONST:
561*fbadb1c4SDavid du Colombier 		memmove(a->sval, p+4, NSNAME);
562*fbadb1c4SDavid du Colombier 		c += NSNAME;
563*fbadb1c4SDavid du Colombier 		break;
564*fbadb1c4SDavid du Colombier 
565*fbadb1c4SDavid du Colombier 	case D_FCONST:
566*fbadb1c4SDavid du Colombier 		a->ieee.l = p[4] | (p[5]<<8) |
567*fbadb1c4SDavid du Colombier 			(p[6]<<16) | (p[7]<<24);
568*fbadb1c4SDavid du Colombier 		a->ieee.h = p[8] | (p[9]<<8) |
569*fbadb1c4SDavid du Colombier 			(p[10]<<16) | (p[11]<<24);
570*fbadb1c4SDavid du Colombier 		c += 8;
571*fbadb1c4SDavid du Colombier 		break;
572*fbadb1c4SDavid du Colombier 	}
573*fbadb1c4SDavid du Colombier 	s = a->sym;
574*fbadb1c4SDavid du Colombier 	if(s == S)
575*fbadb1c4SDavid du Colombier 		goto out;
576*fbadb1c4SDavid du Colombier 	i = a->name;
577*fbadb1c4SDavid du Colombier 	if(i != D_AUTO && i != D_PARAM)
578*fbadb1c4SDavid du Colombier 		goto out;
579*fbadb1c4SDavid du Colombier 
580*fbadb1c4SDavid du Colombier 	l = a->offset;
581*fbadb1c4SDavid du Colombier 	for(u=curauto; u; u=u->link)
582*fbadb1c4SDavid du Colombier 		if(u->sym == s)
583*fbadb1c4SDavid du Colombier 		if(u->type == i) {
584*fbadb1c4SDavid du Colombier 			if(u->aoffset > l)
585*fbadb1c4SDavid du Colombier 				u->aoffset = l;
586*fbadb1c4SDavid du Colombier 			goto out;
587*fbadb1c4SDavid du Colombier 		}
588*fbadb1c4SDavid du Colombier 
589*fbadb1c4SDavid du Colombier 	u = malloc(sizeof(Auto));
590*fbadb1c4SDavid du Colombier 
591*fbadb1c4SDavid du Colombier 	u->link = curauto;
592*fbadb1c4SDavid du Colombier 	curauto = u;
593*fbadb1c4SDavid du Colombier 	u->sym = s;
594*fbadb1c4SDavid du Colombier 	u->aoffset = l;
595*fbadb1c4SDavid du Colombier 	u->type = i;
596*fbadb1c4SDavid du Colombier out:
597*fbadb1c4SDavid du Colombier 	return c;
598*fbadb1c4SDavid du Colombier }
599*fbadb1c4SDavid du Colombier 
600*fbadb1c4SDavid du Colombier void
addlib(char * obj)601*fbadb1c4SDavid du Colombier addlib(char *obj)
602*fbadb1c4SDavid du Colombier {
603*fbadb1c4SDavid du Colombier 	char fn1[LIBNAMELEN], fn2[LIBNAMELEN], comp[LIBNAMELEN], *p, *name;
604*fbadb1c4SDavid du Colombier 	int i, search;
605*fbadb1c4SDavid du Colombier 
606*fbadb1c4SDavid du Colombier 	if(histfrogp <= 0)
607*fbadb1c4SDavid du Colombier 		return;
608*fbadb1c4SDavid du Colombier 
609*fbadb1c4SDavid du Colombier 	name = fn1;
610*fbadb1c4SDavid du Colombier 	search = 0;
611*fbadb1c4SDavid du Colombier 	if(histfrog[0]->name[1] == '/') {
612*fbadb1c4SDavid du Colombier 		sprint(name, "");
613*fbadb1c4SDavid du Colombier 		i = 1;
614*fbadb1c4SDavid du Colombier 	} else if(histfrog[0]->name[1] == '.') {
615*fbadb1c4SDavid du Colombier 		sprint(name, ".");
616*fbadb1c4SDavid du Colombier 		i = 0;
617*fbadb1c4SDavid du Colombier 	} else {
618*fbadb1c4SDavid du Colombier 		sprint(name, "");
619*fbadb1c4SDavid du Colombier 		i = 0;
620*fbadb1c4SDavid du Colombier 		search = 1;
621*fbadb1c4SDavid du Colombier 	}
622*fbadb1c4SDavid du Colombier 
623*fbadb1c4SDavid du Colombier 	for(; i<histfrogp; i++) {
624*fbadb1c4SDavid du Colombier 		snprint(comp, sizeof comp, histfrog[i]->name+1);
625*fbadb1c4SDavid du Colombier 		for(;;) {
626*fbadb1c4SDavid du Colombier 			p = strstr(comp, "$O");
627*fbadb1c4SDavid du Colombier 			if(p == 0)
628*fbadb1c4SDavid du Colombier 				break;
629*fbadb1c4SDavid du Colombier 			memmove(p+1, p+2, strlen(p+2)+1);
630*fbadb1c4SDavid du Colombier 			p[0] = thechar;
631*fbadb1c4SDavid du Colombier 		}
632*fbadb1c4SDavid du Colombier 		for(;;) {
633*fbadb1c4SDavid du Colombier 			p = strstr(comp, "$M");
634*fbadb1c4SDavid du Colombier 			if(p == 0)
635*fbadb1c4SDavid du Colombier 				break;
636*fbadb1c4SDavid du Colombier 			if(strlen(comp)+strlen(thestring)-2+1 >= sizeof comp) {
637*fbadb1c4SDavid du Colombier 				diag("library component too long");
638*fbadb1c4SDavid du Colombier 				return;
639*fbadb1c4SDavid du Colombier 			}
640*fbadb1c4SDavid du Colombier 			memmove(p+strlen(thestring), p+2, strlen(p+2)+1);
641*fbadb1c4SDavid du Colombier 			memmove(p, thestring, strlen(thestring));
642*fbadb1c4SDavid du Colombier 		}
643*fbadb1c4SDavid du Colombier 		if(strlen(fn1) + strlen(comp) + 3 >= sizeof(fn1)) {
644*fbadb1c4SDavid du Colombier 			diag("library component too long");
645*fbadb1c4SDavid du Colombier 			return;
646*fbadb1c4SDavid du Colombier 		}
647*fbadb1c4SDavid du Colombier 		if(i > 0 || !search)
648*fbadb1c4SDavid du Colombier 			strcat(fn1, "/");
649*fbadb1c4SDavid du Colombier 		strcat(fn1, comp);
650*fbadb1c4SDavid du Colombier 	}
651*fbadb1c4SDavid du Colombier 
652*fbadb1c4SDavid du Colombier 	cleanname(name);
653*fbadb1c4SDavid du Colombier 
654*fbadb1c4SDavid du Colombier 	if(search){
655*fbadb1c4SDavid du Colombier 		p = findlib(name);
656*fbadb1c4SDavid du Colombier 		if(p != nil){
657*fbadb1c4SDavid du Colombier 			snprint(fn2, sizeof(fn2), "%s/%s", p, name);
658*fbadb1c4SDavid du Colombier 			name = fn2;
659*fbadb1c4SDavid du Colombier 		}
660*fbadb1c4SDavid du Colombier 	}
661*fbadb1c4SDavid du Colombier 
662*fbadb1c4SDavid du Colombier 	for(i=0; i<libraryp; i++)
663*fbadb1c4SDavid du Colombier 		if(strcmp(name, library[i]) == 0)
664*fbadb1c4SDavid du Colombier 			return;
665*fbadb1c4SDavid du Colombier 	if(libraryp == nelem(library)){
666*fbadb1c4SDavid du Colombier 		diag("too many autolibs; skipping %s", name);
667*fbadb1c4SDavid du Colombier 		return;
668*fbadb1c4SDavid du Colombier 	}
669*fbadb1c4SDavid du Colombier 
670*fbadb1c4SDavid du Colombier 	p = malloc(strlen(name) + 1);
671*fbadb1c4SDavid du Colombier 	strcpy(p, name);
672*fbadb1c4SDavid du Colombier 	library[libraryp] = p;
673*fbadb1c4SDavid du Colombier 	p = malloc(strlen(obj) + 1);
674*fbadb1c4SDavid du Colombier 	strcpy(p, obj);
675*fbadb1c4SDavid du Colombier 	libraryobj[libraryp] = p;
676*fbadb1c4SDavid du Colombier 	libraryp++;
677*fbadb1c4SDavid du Colombier }
678*fbadb1c4SDavid du Colombier 
679*fbadb1c4SDavid du Colombier void
addhist(long line,int type)680*fbadb1c4SDavid du Colombier addhist(long line, int type)
681*fbadb1c4SDavid du Colombier {
682*fbadb1c4SDavid du Colombier 	Auto *u;
683*fbadb1c4SDavid du Colombier 	Sym *s;
684*fbadb1c4SDavid du Colombier 	int i, j, k;
685*fbadb1c4SDavid du Colombier 
686*fbadb1c4SDavid du Colombier 	u = malloc(sizeof(Auto));
687*fbadb1c4SDavid du Colombier 	s = malloc(sizeof(Sym));
688*fbadb1c4SDavid du Colombier 	s->name = malloc(2*(histfrogp+1) + 1);
689*fbadb1c4SDavid du Colombier 
690*fbadb1c4SDavid du Colombier 	u->sym = s;
691*fbadb1c4SDavid du Colombier 	u->type = type;
692*fbadb1c4SDavid du Colombier 	u->aoffset = line;
693*fbadb1c4SDavid du Colombier 	u->link = curhist;
694*fbadb1c4SDavid du Colombier 	curhist = u;
695*fbadb1c4SDavid du Colombier 
696*fbadb1c4SDavid du Colombier 	j = 1;
697*fbadb1c4SDavid du Colombier 	for(i=0; i<histfrogp; i++) {
698*fbadb1c4SDavid du Colombier 		k = histfrog[i]->value;
699*fbadb1c4SDavid du Colombier 		s->name[j+0] = k>>8;
700*fbadb1c4SDavid du Colombier 		s->name[j+1] = k;
701*fbadb1c4SDavid du Colombier 		j += 2;
702*fbadb1c4SDavid du Colombier 	}
703*fbadb1c4SDavid du Colombier }
704*fbadb1c4SDavid du Colombier 
705*fbadb1c4SDavid du Colombier void
histtoauto(void)706*fbadb1c4SDavid du Colombier histtoauto(void)
707*fbadb1c4SDavid du Colombier {
708*fbadb1c4SDavid du Colombier 	Auto *l;
709*fbadb1c4SDavid du Colombier 
710*fbadb1c4SDavid du Colombier 	while(l = curhist) {
711*fbadb1c4SDavid du Colombier 		curhist = l->link;
712*fbadb1c4SDavid du Colombier 		l->link = curauto;
713*fbadb1c4SDavid du Colombier 		curauto = l;
714*fbadb1c4SDavid du Colombier 	}
715*fbadb1c4SDavid du Colombier }
716*fbadb1c4SDavid du Colombier 
717*fbadb1c4SDavid du Colombier void
collapsefrog(Sym * s)718*fbadb1c4SDavid du Colombier collapsefrog(Sym *s)
719*fbadb1c4SDavid du Colombier {
720*fbadb1c4SDavid du Colombier 	int i;
721*fbadb1c4SDavid du Colombier 
722*fbadb1c4SDavid du Colombier 	/*
723*fbadb1c4SDavid du Colombier 	 * bad encoding of path components only allows
724*fbadb1c4SDavid du Colombier 	 * MAXHIST components. if there is an overflow,
725*fbadb1c4SDavid du Colombier 	 * first try to collapse xxx/..
726*fbadb1c4SDavid du Colombier 	 */
727*fbadb1c4SDavid du Colombier 	for(i=1; i<histfrogp; i++)
728*fbadb1c4SDavid du Colombier 		if(strcmp(histfrog[i]->name+1, "..") == 0) {
729*fbadb1c4SDavid du Colombier 			memmove(histfrog+i-1, histfrog+i+1,
730*fbadb1c4SDavid du Colombier 				(histfrogp-i-1)*sizeof(histfrog[0]));
731*fbadb1c4SDavid du Colombier 			histfrogp--;
732*fbadb1c4SDavid du Colombier 			goto out;
733*fbadb1c4SDavid du Colombier 		}
734*fbadb1c4SDavid du Colombier 
735*fbadb1c4SDavid du Colombier 	/*
736*fbadb1c4SDavid du Colombier 	 * next try to collapse .
737*fbadb1c4SDavid du Colombier 	 */
738*fbadb1c4SDavid du Colombier 	for(i=0; i<histfrogp; i++)
739*fbadb1c4SDavid du Colombier 		if(strcmp(histfrog[i]->name+1, ".") == 0) {
740*fbadb1c4SDavid du Colombier 			memmove(histfrog+i, histfrog+i+1,
741*fbadb1c4SDavid du Colombier 				(histfrogp-i-1)*sizeof(histfrog[0]));
742*fbadb1c4SDavid du Colombier 			goto out;
743*fbadb1c4SDavid du Colombier 		}
744*fbadb1c4SDavid du Colombier 
745*fbadb1c4SDavid du Colombier 	/*
746*fbadb1c4SDavid du Colombier 	 * last chance, just truncate from front
747*fbadb1c4SDavid du Colombier 	 */
748*fbadb1c4SDavid du Colombier 	memmove(histfrog+0, histfrog+1,
749*fbadb1c4SDavid du Colombier 		(histfrogp-1)*sizeof(histfrog[0]));
750*fbadb1c4SDavid du Colombier 
751*fbadb1c4SDavid du Colombier out:
752*fbadb1c4SDavid du Colombier 	histfrog[histfrogp-1] = s;
753*fbadb1c4SDavid du Colombier }
754*fbadb1c4SDavid du Colombier 
755*fbadb1c4SDavid du Colombier void
nopout(Prog * p)756*fbadb1c4SDavid du Colombier nopout(Prog *p)
757*fbadb1c4SDavid du Colombier {
758*fbadb1c4SDavid du Colombier 	p->as = ANOP;
759*fbadb1c4SDavid du Colombier 	p->from.type = D_NONE;
760*fbadb1c4SDavid du Colombier 	p->to.type = D_NONE;
761*fbadb1c4SDavid du Colombier }
762*fbadb1c4SDavid du Colombier 
763*fbadb1c4SDavid du Colombier uchar*
readsome(int f,uchar * buf,uchar * good,uchar * stop,int max)764*fbadb1c4SDavid du Colombier readsome(int f, uchar *buf, uchar *good, uchar *stop, int max)
765*fbadb1c4SDavid du Colombier {
766*fbadb1c4SDavid du Colombier 	int n;
767*fbadb1c4SDavid du Colombier 
768*fbadb1c4SDavid du Colombier 	n = stop - good;
769*fbadb1c4SDavid du Colombier 	memmove(buf, good, stop - good);
770*fbadb1c4SDavid du Colombier 	stop = buf + n;
771*fbadb1c4SDavid du Colombier 	n = MAXIO - n;
772*fbadb1c4SDavid du Colombier 	if(n > max)
773*fbadb1c4SDavid du Colombier 		n = max;
774*fbadb1c4SDavid du Colombier 	n = read(f, stop, n);
775*fbadb1c4SDavid du Colombier 	if(n <= 0)
776*fbadb1c4SDavid du Colombier 		return 0;
777*fbadb1c4SDavid du Colombier 	return stop + n;
778*fbadb1c4SDavid du Colombier }
779*fbadb1c4SDavid du Colombier 
780*fbadb1c4SDavid du Colombier void
ldobj(int f,long c,char * pn)781*fbadb1c4SDavid du Colombier ldobj(int f, long c, char *pn)
782*fbadb1c4SDavid du Colombier {
783*fbadb1c4SDavid du Colombier 	Prog *p, *t;
784*fbadb1c4SDavid du Colombier 	Sym *h[NSYM], *s, *di;
785*fbadb1c4SDavid du Colombier 	int v, o, r, skip;
786*fbadb1c4SDavid du Colombier 	vlong ipc;
787*fbadb1c4SDavid du Colombier 	uchar *bloc, *bsize, *stop;
788*fbadb1c4SDavid du Colombier 	ulong sig;
789*fbadb1c4SDavid du Colombier 	static int files;
790*fbadb1c4SDavid du Colombier 	static char **filen;
791*fbadb1c4SDavid du Colombier 	char **nfilen;
792*fbadb1c4SDavid du Colombier 
793*fbadb1c4SDavid du Colombier 	if((files&15) == 0){
794*fbadb1c4SDavid du Colombier 		nfilen = malloc((files+16)*sizeof(char*));
795*fbadb1c4SDavid du Colombier 		memmove(nfilen, filen, files*sizeof(char*));
796*fbadb1c4SDavid du Colombier 		free(filen);
797*fbadb1c4SDavid du Colombier 		filen = nfilen;
798*fbadb1c4SDavid du Colombier 	}
799*fbadb1c4SDavid du Colombier 	filen[files++] = strdup(pn);
800*fbadb1c4SDavid du Colombier 
801*fbadb1c4SDavid du Colombier 	bsize = buf.xbuf;
802*fbadb1c4SDavid du Colombier 	bloc = buf.xbuf;
803*fbadb1c4SDavid du Colombier 	di = S;
804*fbadb1c4SDavid du Colombier 
805*fbadb1c4SDavid du Colombier newloop:
806*fbadb1c4SDavid du Colombier 	memset(h, 0, sizeof(h));
807*fbadb1c4SDavid du Colombier 	histfrogp = 0;
808*fbadb1c4SDavid du Colombier 	version++;
809*fbadb1c4SDavid du Colombier 	ipc = pc;
810*fbadb1c4SDavid du Colombier 	skip = 0;
811*fbadb1c4SDavid du Colombier 
812*fbadb1c4SDavid du Colombier loop:
813*fbadb1c4SDavid du Colombier 	if(c <= 0)
814*fbadb1c4SDavid du Colombier 		goto eof;
815*fbadb1c4SDavid du Colombier 	r = bsize - bloc;
816*fbadb1c4SDavid du Colombier 	if(r < 100 && r < c) {		/* enough for largest prog */
817*fbadb1c4SDavid du Colombier 		bsize = readsome(f, buf.xbuf, bloc, bsize, c);
818*fbadb1c4SDavid du Colombier 		if(bsize == 0)
819*fbadb1c4SDavid du Colombier 			goto eof;
820*fbadb1c4SDavid du Colombier 		bloc = buf.xbuf;
821*fbadb1c4SDavid du Colombier 		goto loop;
822*fbadb1c4SDavid du Colombier 	}
823*fbadb1c4SDavid du Colombier 	o = bloc[0] | (bloc[1] << 8);		/* as */
824*fbadb1c4SDavid du Colombier 	if(o <= 0 || o >= ALAST) {
825*fbadb1c4SDavid du Colombier 		diag("%s: opcode out of range %d", pn, o);
826*fbadb1c4SDavid du Colombier 		print("	probably not a .%c file\n", thechar);
827*fbadb1c4SDavid du Colombier 		errorexit();
828*fbadb1c4SDavid du Colombier 	}
829*fbadb1c4SDavid du Colombier 	if(o == ANAME || o == ASIGNAME) {
830*fbadb1c4SDavid du Colombier 		sig = 0;
831*fbadb1c4SDavid du Colombier 		if(o == ASIGNAME) {
832*fbadb1c4SDavid du Colombier 			sig = bloc[2] | (bloc[3]<<8) | (bloc[4]<<16) | (bloc[5]<<24);
833*fbadb1c4SDavid du Colombier 			bloc += 4;
834*fbadb1c4SDavid du Colombier 			c -= 4;
835*fbadb1c4SDavid du Colombier 		}
836*fbadb1c4SDavid du Colombier 		stop = memchr(&bloc[4], 0, bsize-&bloc[4]);
837*fbadb1c4SDavid du Colombier 		if(stop == 0){
838*fbadb1c4SDavid du Colombier 			bsize = readsome(f, buf.xbuf, bloc, bsize, c);
839*fbadb1c4SDavid du Colombier 			if(bsize == 0)
840*fbadb1c4SDavid du Colombier 				goto eof;
841*fbadb1c4SDavid du Colombier 			bloc = buf.xbuf;
842*fbadb1c4SDavid du Colombier 			stop = memchr(&bloc[4], 0, bsize-&bloc[4]);
843*fbadb1c4SDavid du Colombier 			if(stop == 0){
844*fbadb1c4SDavid du Colombier 				fprint(2, "%s: name too long\n", pn);
845*fbadb1c4SDavid du Colombier 				errorexit();
846*fbadb1c4SDavid du Colombier 			}
847*fbadb1c4SDavid du Colombier 		}
848*fbadb1c4SDavid du Colombier 		v = bloc[2];	/* type */
849*fbadb1c4SDavid du Colombier 		o = bloc[3];	/* sym */
850*fbadb1c4SDavid du Colombier 		bloc += 4;
851*fbadb1c4SDavid du Colombier 		c -= 4;
852*fbadb1c4SDavid du Colombier 
853*fbadb1c4SDavid du Colombier 		r = 0;
854*fbadb1c4SDavid du Colombier 		if(v == D_STATIC)
855*fbadb1c4SDavid du Colombier 			r = version;
856*fbadb1c4SDavid du Colombier 		s = lookup((char*)bloc, r);
857*fbadb1c4SDavid du Colombier 		c -= &stop[1] - bloc;
858*fbadb1c4SDavid du Colombier 		bloc = stop + 1;
859*fbadb1c4SDavid du Colombier 		if(sig != 0){
860*fbadb1c4SDavid du Colombier 			if(s->sig != 0 && s->sig != sig)
861*fbadb1c4SDavid du Colombier 				diag("incompatible type signatures %lux(%s) and %lux(%s) for %s", s->sig, filen[s->file], sig, pn, s->name);
862*fbadb1c4SDavid du Colombier 			s->sig = sig;
863*fbadb1c4SDavid du Colombier 			s->file = files-1;
864*fbadb1c4SDavid du Colombier 		}
865*fbadb1c4SDavid du Colombier 
866*fbadb1c4SDavid du Colombier 
867*fbadb1c4SDavid du Colombier 		if(debug['W'])
868*fbadb1c4SDavid du Colombier 			print("	ANAME	%s\n", s->name);
869*fbadb1c4SDavid du Colombier 		h[o] = s;
870*fbadb1c4SDavid du Colombier 		if((v == D_EXTERN || v == D_STATIC) && s->type == 0)
871*fbadb1c4SDavid du Colombier 			s->type = SXREF;
872*fbadb1c4SDavid du Colombier 		if(v == D_FILE) {
873*fbadb1c4SDavid du Colombier 			if(s->type != SFILE) {
874*fbadb1c4SDavid du Colombier 				histgen++;
875*fbadb1c4SDavid du Colombier 				s->type = SFILE;
876*fbadb1c4SDavid du Colombier 				s->value = histgen;
877*fbadb1c4SDavid du Colombier 			}
878*fbadb1c4SDavid du Colombier 			if(histfrogp < MAXHIST) {
879*fbadb1c4SDavid du Colombier 				histfrog[histfrogp] = s;
880*fbadb1c4SDavid du Colombier 				histfrogp++;
881*fbadb1c4SDavid du Colombier 			} else
882*fbadb1c4SDavid du Colombier 				collapsefrog(s);
883*fbadb1c4SDavid du Colombier 		}
884*fbadb1c4SDavid du Colombier 		goto loop;
885*fbadb1c4SDavid du Colombier 	}
886*fbadb1c4SDavid du Colombier 
887*fbadb1c4SDavid du Colombier 	if(nhunk < sizeof(Prog))
888*fbadb1c4SDavid du Colombier 		gethunk();
889*fbadb1c4SDavid du Colombier 	p = (Prog*)hunk;
890*fbadb1c4SDavid du Colombier 	nhunk -= sizeof(Prog);
891*fbadb1c4SDavid du Colombier 	hunk += sizeof(Prog);
892*fbadb1c4SDavid du Colombier 
893*fbadb1c4SDavid du Colombier 	p->as = o;
894*fbadb1c4SDavid du Colombier 	p->reg = bloc[2] & 0x3f;
895*fbadb1c4SDavid du Colombier 	if(bloc[2] & 0x80)
896*fbadb1c4SDavid du Colombier 		p->mark = NOSCHED;
897*fbadb1c4SDavid du Colombier 	p->line = bloc[3] | (bloc[4]<<8) | (bloc[5]<<16) | (bloc[6]<<24);
898*fbadb1c4SDavid du Colombier 	r = zaddr(bloc+7, &p->from, h) + 7;
899*fbadb1c4SDavid du Colombier 	if(bloc[2] & 0x40)
900*fbadb1c4SDavid du Colombier 		r += zaddr(bloc+r, &p->from3, h);
901*fbadb1c4SDavid du Colombier 	else
902*fbadb1c4SDavid du Colombier 		p->from3 = zprg.from3;
903*fbadb1c4SDavid du Colombier 	r += zaddr(bloc+r, &p->to, h);
904*fbadb1c4SDavid du Colombier 	bloc += r;
905*fbadb1c4SDavid du Colombier 	c -= r;
906*fbadb1c4SDavid du Colombier 
907*fbadb1c4SDavid du Colombier 	if(p->reg < 0 || p->reg > NREG)
908*fbadb1c4SDavid du Colombier 		diag("register out of range %d", p->reg);
909*fbadb1c4SDavid du Colombier 
910*fbadb1c4SDavid du Colombier 	p->link = P;
911*fbadb1c4SDavid du Colombier 	p->cond = P;
912*fbadb1c4SDavid du Colombier 
913*fbadb1c4SDavid du Colombier 	if(debug['W'])
914*fbadb1c4SDavid du Colombier 		print("%P\n", p);
915*fbadb1c4SDavid du Colombier 
916*fbadb1c4SDavid du Colombier 	switch(o) {
917*fbadb1c4SDavid du Colombier 	case AHISTORY:
918*fbadb1c4SDavid du Colombier 		if(p->to.offset == -1) {
919*fbadb1c4SDavid du Colombier 			addlib(pn);
920*fbadb1c4SDavid du Colombier 			histfrogp = 0;
921*fbadb1c4SDavid du Colombier 			goto loop;
922*fbadb1c4SDavid du Colombier 		}
923*fbadb1c4SDavid du Colombier 		addhist(p->line, D_FILE);		/* 'z' */
924*fbadb1c4SDavid du Colombier 		if(p->to.offset)
925*fbadb1c4SDavid du Colombier 			addhist(p->to.offset, D_FILE1);	/* 'Z' */
926*fbadb1c4SDavid du Colombier 		histfrogp = 0;
927*fbadb1c4SDavid du Colombier 		goto loop;
928*fbadb1c4SDavid du Colombier 
929*fbadb1c4SDavid du Colombier 	case AEND:
930*fbadb1c4SDavid du Colombier 		histtoauto();
931*fbadb1c4SDavid du Colombier 		if(curtext != P)
932*fbadb1c4SDavid du Colombier 			curtext->to.autom = curauto;
933*fbadb1c4SDavid du Colombier 		curauto = 0;
934*fbadb1c4SDavid du Colombier 		curtext = P;
935*fbadb1c4SDavid du Colombier 		if(c)
936*fbadb1c4SDavid du Colombier 			goto newloop;
937*fbadb1c4SDavid du Colombier 		return;
938*fbadb1c4SDavid du Colombier 
939*fbadb1c4SDavid du Colombier 	case AGLOBL:
940*fbadb1c4SDavid du Colombier 		s = p->from.sym;
941*fbadb1c4SDavid du Colombier 		if(s == S) {
942*fbadb1c4SDavid du Colombier 			diag("GLOBL must have a name\n%P", p);
943*fbadb1c4SDavid du Colombier 			errorexit();
944*fbadb1c4SDavid du Colombier 		}
945*fbadb1c4SDavid du Colombier 		if(s->type == 0 || s->type == SXREF) {
946*fbadb1c4SDavid du Colombier 			s->type = SBSS;
947*fbadb1c4SDavid du Colombier 			s->value = 0;
948*fbadb1c4SDavid du Colombier 		}
949*fbadb1c4SDavid du Colombier 		if(s->type != SBSS) {
950*fbadb1c4SDavid du Colombier 			diag("redefinition: %s\n%P", s->name, p);
951*fbadb1c4SDavid du Colombier 			s->type = SBSS;
952*fbadb1c4SDavid du Colombier 			s->value = 0;
953*fbadb1c4SDavid du Colombier 		}
954*fbadb1c4SDavid du Colombier 		if(p->to.offset > s->value)
955*fbadb1c4SDavid du Colombier 			s->value = p->to.offset;
956*fbadb1c4SDavid du Colombier 		break;
957*fbadb1c4SDavid du Colombier 
958*fbadb1c4SDavid du Colombier 	case ADYNT:
959*fbadb1c4SDavid du Colombier 		if(p->to.sym == S) {
960*fbadb1c4SDavid du Colombier 			diag("DYNT without a sym\n%P", p);
961*fbadb1c4SDavid du Colombier 			break;
962*fbadb1c4SDavid du Colombier 		}
963*fbadb1c4SDavid du Colombier 		di = p->to.sym;
964*fbadb1c4SDavid du Colombier 		p->reg = 4;
965*fbadb1c4SDavid du Colombier 		if(di->type == SXREF) {
966*fbadb1c4SDavid du Colombier 			if(debug['z'])
967*fbadb1c4SDavid du Colombier 				Bprint(&bso, "%P set to %d\n", p, dtype);
968*fbadb1c4SDavid du Colombier 			di->type = SCONST;
969*fbadb1c4SDavid du Colombier 			di->value = dtype;
970*fbadb1c4SDavid du Colombier 			dtype += 4;
971*fbadb1c4SDavid du Colombier 		}
972*fbadb1c4SDavid du Colombier 		if(p->from.sym == S)
973*fbadb1c4SDavid du Colombier 			break;
974*fbadb1c4SDavid du Colombier 
975*fbadb1c4SDavid du Colombier 		p->from.offset = di->value;
976*fbadb1c4SDavid du Colombier 		p->from.sym->type = SDATA;
977*fbadb1c4SDavid du Colombier 		if(curtext == P) {
978*fbadb1c4SDavid du Colombier 			diag("DYNT not in text: %P", p);
979*fbadb1c4SDavid du Colombier 			break;
980*fbadb1c4SDavid du Colombier 		}
981*fbadb1c4SDavid du Colombier 		p->to.sym = curtext->from.sym;
982*fbadb1c4SDavid du Colombier 		p->to.type = D_CONST;
983*fbadb1c4SDavid du Colombier 		p->link = datap;
984*fbadb1c4SDavid du Colombier 		datap = p;
985*fbadb1c4SDavid du Colombier 		break;
986*fbadb1c4SDavid du Colombier 
987*fbadb1c4SDavid du Colombier 	case AINIT:
988*fbadb1c4SDavid du Colombier 		if(p->from.sym == S) {
989*fbadb1c4SDavid du Colombier 			diag("INIT without a sym\n%P", p);
990*fbadb1c4SDavid du Colombier 			break;
991*fbadb1c4SDavid du Colombier 		}
992*fbadb1c4SDavid du Colombier 		if(di == S) {
993*fbadb1c4SDavid du Colombier 			diag("INIT without previous DYNT\n%P", p);
994*fbadb1c4SDavid du Colombier 			break;
995*fbadb1c4SDavid du Colombier 		}
996*fbadb1c4SDavid du Colombier 		p->from.offset = di->value;
997*fbadb1c4SDavid du Colombier 		p->from.sym->type = SDATA;
998*fbadb1c4SDavid du Colombier 		p->link = datap;
999*fbadb1c4SDavid du Colombier 		datap = p;
1000*fbadb1c4SDavid du Colombier 		break;
1001*fbadb1c4SDavid du Colombier 
1002*fbadb1c4SDavid du Colombier 	case ADATA:
1003*fbadb1c4SDavid du Colombier 		p->link = datap;
1004*fbadb1c4SDavid du Colombier 		datap = p;
1005*fbadb1c4SDavid du Colombier 		break;
1006*fbadb1c4SDavid du Colombier 
1007*fbadb1c4SDavid du Colombier 	case AGOK:
1008*fbadb1c4SDavid du Colombier 		diag("unknown opcode\n%P", p);
1009*fbadb1c4SDavid du Colombier 		p->pc = pc;
1010*fbadb1c4SDavid du Colombier 		pc++;
1011*fbadb1c4SDavid du Colombier 		break;
1012*fbadb1c4SDavid du Colombier 
1013*fbadb1c4SDavid du Colombier 	case ATEXT:
1014*fbadb1c4SDavid du Colombier 		if(curtext != P) {
1015*fbadb1c4SDavid du Colombier 			histtoauto();
1016*fbadb1c4SDavid du Colombier 			curtext->to.autom = curauto;
1017*fbadb1c4SDavid du Colombier 			curauto = 0;
1018*fbadb1c4SDavid du Colombier 		}
1019*fbadb1c4SDavid du Colombier 		curtext = p;
1020*fbadb1c4SDavid du Colombier 		autosize = (p->to.offset+7L) & ~7L;
1021*fbadb1c4SDavid du Colombier 		p->to.offset = autosize;
1022*fbadb1c4SDavid du Colombier 		autosize += 8;
1023*fbadb1c4SDavid du Colombier 		s = p->from.sym;
1024*fbadb1c4SDavid du Colombier 		if(s == S) {
1025*fbadb1c4SDavid du Colombier 			diag("TEXT must have a name\n%P", p);
1026*fbadb1c4SDavid du Colombier 			errorexit();
1027*fbadb1c4SDavid du Colombier 		}
1028*fbadb1c4SDavid du Colombier 		if(s->type != 0 && s->type != SXREF) {
1029*fbadb1c4SDavid du Colombier 			if(p->reg & DUPOK) {
1030*fbadb1c4SDavid du Colombier 				skip = 1;
1031*fbadb1c4SDavid du Colombier 				goto casedef;
1032*fbadb1c4SDavid du Colombier 			}
1033*fbadb1c4SDavid du Colombier 			diag("redefinition: %s\n%P", s->name, p);
1034*fbadb1c4SDavid du Colombier 		}
1035*fbadb1c4SDavid du Colombier 		s->type = STEXT;
1036*fbadb1c4SDavid du Colombier 		s->value = pc;
1037*fbadb1c4SDavid du Colombier 		if(textp != P) {
1038*fbadb1c4SDavid du Colombier 			for(t = textp; t->cond != P; t = t->cond)
1039*fbadb1c4SDavid du Colombier 				;
1040*fbadb1c4SDavid du Colombier 			t->cond = p;
1041*fbadb1c4SDavid du Colombier 		} else
1042*fbadb1c4SDavid du Colombier 			textp = p;
1043*fbadb1c4SDavid du Colombier 		lastp->link = p;
1044*fbadb1c4SDavid du Colombier 		lastp = p;
1045*fbadb1c4SDavid du Colombier 		p->pc = pc;
1046*fbadb1c4SDavid du Colombier 		pc++;
1047*fbadb1c4SDavid du Colombier 		break;
1048*fbadb1c4SDavid du Colombier 
1049*fbadb1c4SDavid du Colombier 	case AFMOVS:
1050*fbadb1c4SDavid du Colombier 		if(skip)
1051*fbadb1c4SDavid du Colombier 			goto casedef;
1052*fbadb1c4SDavid du Colombier 
1053*fbadb1c4SDavid du Colombier 		if(p->from.type == D_FCONST) {
1054*fbadb1c4SDavid du Colombier 			/* size sb 9 max */
1055*fbadb1c4SDavid du Colombier 			sprint(literal, "$%lux", ieeedtof(&p->from.ieee));
1056*fbadb1c4SDavid du Colombier 			s = lookup(literal, 0);
1057*fbadb1c4SDavid du Colombier 			if(s->type == 0) {
1058*fbadb1c4SDavid du Colombier 				s->type = SBSS;
1059*fbadb1c4SDavid du Colombier 				s->value = 4;
1060*fbadb1c4SDavid du Colombier 				t = prg();
1061*fbadb1c4SDavid du Colombier 				t->as = ADATA;
1062*fbadb1c4SDavid du Colombier 				t->line = p->line;
1063*fbadb1c4SDavid du Colombier 				t->from.type = D_OREG;
1064*fbadb1c4SDavid du Colombier 				t->from.sym = s;
1065*fbadb1c4SDavid du Colombier 				t->from.name = D_EXTERN;
1066*fbadb1c4SDavid du Colombier 				t->reg = 4;
1067*fbadb1c4SDavid du Colombier 				t->to = p->from;
1068*fbadb1c4SDavid du Colombier 				t->link = datap;
1069*fbadb1c4SDavid du Colombier 				datap = t;
1070*fbadb1c4SDavid du Colombier 			}
1071*fbadb1c4SDavid du Colombier 			p->from.type = D_OREG;
1072*fbadb1c4SDavid du Colombier 			p->from.sym = s;
1073*fbadb1c4SDavid du Colombier 			p->from.name = D_EXTERN;
1074*fbadb1c4SDavid du Colombier 			p->from.offset = 0;
1075*fbadb1c4SDavid du Colombier 		}
1076*fbadb1c4SDavid du Colombier 		goto casedef;
1077*fbadb1c4SDavid du Colombier 
1078*fbadb1c4SDavid du Colombier 	case AFMOVD:
1079*fbadb1c4SDavid du Colombier 		if(skip)
1080*fbadb1c4SDavid du Colombier 			goto casedef;
1081*fbadb1c4SDavid du Colombier 		if(p->from.type == D_FCONST) {
1082*fbadb1c4SDavid du Colombier 			/* size sb 18 max */
1083*fbadb1c4SDavid du Colombier 			sprint(literal, "$%lux.%lux",
1084*fbadb1c4SDavid du Colombier 				p->from.ieee.l, p->from.ieee.h);
1085*fbadb1c4SDavid du Colombier 			s = lookup(literal, 0);
1086*fbadb1c4SDavid du Colombier 			if(s->type == 0) {
1087*fbadb1c4SDavid du Colombier 				s->type = SBSS;
1088*fbadb1c4SDavid du Colombier 				s->value = 8;
1089*fbadb1c4SDavid du Colombier 				t = prg();
1090*fbadb1c4SDavid du Colombier 				t->as = ADATA;
1091*fbadb1c4SDavid du Colombier 				t->line = p->line;
1092*fbadb1c4SDavid du Colombier 				t->from.type = D_OREG;
1093*fbadb1c4SDavid du Colombier 				t->from.sym = s;
1094*fbadb1c4SDavid du Colombier 				t->from.name = D_EXTERN;
1095*fbadb1c4SDavid du Colombier 				t->reg = 8;
1096*fbadb1c4SDavid du Colombier 				t->to = p->from;
1097*fbadb1c4SDavid du Colombier 				t->link = datap;
1098*fbadb1c4SDavid du Colombier 				datap = t;
1099*fbadb1c4SDavid du Colombier 			}
1100*fbadb1c4SDavid du Colombier 			p->from.type = D_OREG;
1101*fbadb1c4SDavid du Colombier 			p->from.sym = s;
1102*fbadb1c4SDavid du Colombier 			p->from.name = D_EXTERN;
1103*fbadb1c4SDavid du Colombier 			p->from.offset = 0;
1104*fbadb1c4SDavid du Colombier 		}
1105*fbadb1c4SDavid du Colombier 		goto casedef;
1106*fbadb1c4SDavid du Colombier 
1107*fbadb1c4SDavid du Colombier 	case ASUBC:
1108*fbadb1c4SDavid du Colombier 		if(p->from.type == D_CONST) {
1109*fbadb1c4SDavid du Colombier 			p->from.offset = -p->from.offset;
1110*fbadb1c4SDavid du Colombier 			p->as = AADDC;
1111*fbadb1c4SDavid du Colombier 		}
1112*fbadb1c4SDavid du Colombier 		goto casedef;
1113*fbadb1c4SDavid du Colombier 
1114*fbadb1c4SDavid du Colombier 	case ASUBCCC:
1115*fbadb1c4SDavid du Colombier 		if(p->from.type == D_CONST) {
1116*fbadb1c4SDavid du Colombier 			p->from.offset = -p->from.offset;
1117*fbadb1c4SDavid du Colombier 			p->as = AADDCCC;
1118*fbadb1c4SDavid du Colombier 		}
1119*fbadb1c4SDavid du Colombier 		goto casedef;
1120*fbadb1c4SDavid du Colombier 
1121*fbadb1c4SDavid du Colombier 	case ASUB:
1122*fbadb1c4SDavid du Colombier 		if(p->from.type == D_CONST) {
1123*fbadb1c4SDavid du Colombier 			p->from.offset = -p->from.offset;
1124*fbadb1c4SDavid du Colombier 			p->as = AADD;
1125*fbadb1c4SDavid du Colombier 		}
1126*fbadb1c4SDavid du Colombier 		goto casedef;
1127*fbadb1c4SDavid du Colombier 
1128*fbadb1c4SDavid du Colombier 	case AMOVD:
1129*fbadb1c4SDavid du Colombier 		if(skip)
1130*fbadb1c4SDavid du Colombier 			goto casedef;
1131*fbadb1c4SDavid du Colombier 		if(p->from.type == D_CONST && !isuint32(p->from.offset) && !isint32(p->from.offset)) {
1132*fbadb1c4SDavid du Colombier 			/* make a literal */
1133*fbadb1c4SDavid du Colombier 			sprint(literal, "$%llux", p->from.offset);
1134*fbadb1c4SDavid du Colombier 			s = lookup(literal, 0);
1135*fbadb1c4SDavid du Colombier 			if(s->type == 0) {
1136*fbadb1c4SDavid du Colombier 				s->type = SBSS;
1137*fbadb1c4SDavid du Colombier 				s->value = 8;
1138*fbadb1c4SDavid du Colombier 				t = prg();
1139*fbadb1c4SDavid du Colombier 				t->as = ADATA;
1140*fbadb1c4SDavid du Colombier 				t->line = p->line;
1141*fbadb1c4SDavid du Colombier 				t->from.type = D_OREG;
1142*fbadb1c4SDavid du Colombier 				t->from.sym = s;
1143*fbadb1c4SDavid du Colombier 				t->from.name = D_EXTERN;
1144*fbadb1c4SDavid du Colombier 				t->reg = 8;
1145*fbadb1c4SDavid du Colombier 				t->to = p->from;
1146*fbadb1c4SDavid du Colombier 				t->link = datap;
1147*fbadb1c4SDavid du Colombier 				datap = t;
1148*fbadb1c4SDavid du Colombier 			}
1149*fbadb1c4SDavid du Colombier 			p->from.type = D_OREG;
1150*fbadb1c4SDavid du Colombier 			p->from.sym = s;
1151*fbadb1c4SDavid du Colombier 			p->from.name = D_EXTERN;
1152*fbadb1c4SDavid du Colombier 			p->from.offset = 0;
1153*fbadb1c4SDavid du Colombier 		}
1154*fbadb1c4SDavid du Colombier 		goto casedef;
1155*fbadb1c4SDavid du Colombier 
1156*fbadb1c4SDavid du Colombier 	default:
1157*fbadb1c4SDavid du Colombier 	casedef:
1158*fbadb1c4SDavid du Colombier 		if(skip)
1159*fbadb1c4SDavid du Colombier 			nopout(p);
1160*fbadb1c4SDavid du Colombier 
1161*fbadb1c4SDavid du Colombier 		if(p->to.type == D_BRANCH)
1162*fbadb1c4SDavid du Colombier 			p->to.offset += ipc;
1163*fbadb1c4SDavid du Colombier 		lastp->link = p;
1164*fbadb1c4SDavid du Colombier 		lastp = p;
1165*fbadb1c4SDavid du Colombier 		p->pc = pc;
1166*fbadb1c4SDavid du Colombier 		pc++;
1167*fbadb1c4SDavid du Colombier 		break;
1168*fbadb1c4SDavid du Colombier 	}
1169*fbadb1c4SDavid du Colombier 	goto loop;
1170*fbadb1c4SDavid du Colombier 
1171*fbadb1c4SDavid du Colombier eof:
1172*fbadb1c4SDavid du Colombier 	diag("truncated object file: %s", pn);
1173*fbadb1c4SDavid du Colombier }
1174*fbadb1c4SDavid du Colombier 
1175*fbadb1c4SDavid du Colombier Sym*
lookup(char * symb,int v)1176*fbadb1c4SDavid du Colombier lookup(char *symb, int v)
1177*fbadb1c4SDavid du Colombier {
1178*fbadb1c4SDavid du Colombier 	Sym *s;
1179*fbadb1c4SDavid du Colombier 	char *p;
1180*fbadb1c4SDavid du Colombier 	long h;
1181*fbadb1c4SDavid du Colombier 	int c, l;
1182*fbadb1c4SDavid du Colombier 
1183*fbadb1c4SDavid du Colombier 	h = v;
1184*fbadb1c4SDavid du Colombier 	for(p=symb; c = *p; p++)
1185*fbadb1c4SDavid du Colombier 		h = h+h+h + c;
1186*fbadb1c4SDavid du Colombier 	l = (p - symb) + 1;
1187*fbadb1c4SDavid du Colombier 	h &= 0xffffff;
1188*fbadb1c4SDavid du Colombier 	h %= NHASH;
1189*fbadb1c4SDavid du Colombier 	for(s = hash[h]; s != S; s = s->link)
1190*fbadb1c4SDavid du Colombier 		if(s->version == v)
1191*fbadb1c4SDavid du Colombier 		if(memcmp(s->name, symb, l) == 0)
1192*fbadb1c4SDavid du Colombier 			return s;
1193*fbadb1c4SDavid du Colombier 
1194*fbadb1c4SDavid du Colombier 	while(nhunk < sizeof(Sym))
1195*fbadb1c4SDavid du Colombier 		gethunk();
1196*fbadb1c4SDavid du Colombier 	s = (Sym*)hunk;
1197*fbadb1c4SDavid du Colombier 	nhunk -= sizeof(Sym);
1198*fbadb1c4SDavid du Colombier 	hunk += sizeof(Sym);
1199*fbadb1c4SDavid du Colombier 
1200*fbadb1c4SDavid du Colombier 	s->name = malloc(l + 1);
1201*fbadb1c4SDavid du Colombier 	memmove(s->name, symb, l);
1202*fbadb1c4SDavid du Colombier 
1203*fbadb1c4SDavid du Colombier 	s->link = hash[h];
1204*fbadb1c4SDavid du Colombier 	s->type = 0;
1205*fbadb1c4SDavid du Colombier 	s->version = v;
1206*fbadb1c4SDavid du Colombier 	s->value = 0;
1207*fbadb1c4SDavid du Colombier 	s->sig = 0;
1208*fbadb1c4SDavid du Colombier 	hash[h] = s;
1209*fbadb1c4SDavid du Colombier 	return s;
1210*fbadb1c4SDavid du Colombier }
1211*fbadb1c4SDavid du Colombier 
1212*fbadb1c4SDavid du Colombier Prog*
prg(void)1213*fbadb1c4SDavid du Colombier prg(void)
1214*fbadb1c4SDavid du Colombier {
1215*fbadb1c4SDavid du Colombier 	Prog *p;
1216*fbadb1c4SDavid du Colombier 	int n;
1217*fbadb1c4SDavid du Colombier 
1218*fbadb1c4SDavid du Colombier 	n = (sizeof(Prog) + 3) & ~3;
1219*fbadb1c4SDavid du Colombier 	while(nhunk < n)
1220*fbadb1c4SDavid du Colombier 		gethunk();
1221*fbadb1c4SDavid du Colombier 
1222*fbadb1c4SDavid du Colombier 	p = (Prog*)hunk;
1223*fbadb1c4SDavid du Colombier 	nhunk -= n;
1224*fbadb1c4SDavid du Colombier 	hunk += n;
1225*fbadb1c4SDavid du Colombier 
1226*fbadb1c4SDavid du Colombier 	*p = zprg;
1227*fbadb1c4SDavid du Colombier 	return p;
1228*fbadb1c4SDavid du Colombier }
1229*fbadb1c4SDavid du Colombier 
1230*fbadb1c4SDavid du Colombier void
gethunk(void)1231*fbadb1c4SDavid du Colombier gethunk(void)
1232*fbadb1c4SDavid du Colombier {
1233*fbadb1c4SDavid du Colombier 	char *h;
1234*fbadb1c4SDavid du Colombier 	long nh;
1235*fbadb1c4SDavid du Colombier 
1236*fbadb1c4SDavid du Colombier 	nh = NHUNK;
1237*fbadb1c4SDavid du Colombier 	if(tothunk >= 5L*NHUNK) {
1238*fbadb1c4SDavid du Colombier 		nh = 5L*NHUNK;
1239*fbadb1c4SDavid du Colombier 		if(tothunk >= 25L*NHUNK)
1240*fbadb1c4SDavid du Colombier 			nh = 25L*NHUNK;
1241*fbadb1c4SDavid du Colombier 	}
1242*fbadb1c4SDavid du Colombier 	h = mysbrk(nh);
1243*fbadb1c4SDavid du Colombier 	if(h == (char *)-1) {
1244*fbadb1c4SDavid du Colombier 		diag("out of memory");
1245*fbadb1c4SDavid du Colombier 		errorexit();
1246*fbadb1c4SDavid du Colombier 	}
1247*fbadb1c4SDavid du Colombier 
1248*fbadb1c4SDavid du Colombier 	hunk = h;
1249*fbadb1c4SDavid du Colombier 	nhunk = nh;
1250*fbadb1c4SDavid du Colombier 	tothunk += nh;
1251*fbadb1c4SDavid du Colombier }
1252*fbadb1c4SDavid du Colombier 
1253*fbadb1c4SDavid du Colombier void
doprof1(void)1254*fbadb1c4SDavid du Colombier doprof1(void)
1255*fbadb1c4SDavid du Colombier {
1256*fbadb1c4SDavid du Colombier 	Sym *s;
1257*fbadb1c4SDavid du Colombier 	long n;
1258*fbadb1c4SDavid du Colombier 	Prog *p, *q;
1259*fbadb1c4SDavid du Colombier 
1260*fbadb1c4SDavid du Colombier 	if(debug['v'])
1261*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%5.2f profile 1\n", cputime());
1262*fbadb1c4SDavid du Colombier 	Bflush(&bso);
1263*fbadb1c4SDavid du Colombier 	s = lookup("__mcount", 0);
1264*fbadb1c4SDavid du Colombier 	n = 1;
1265*fbadb1c4SDavid du Colombier 	for(p = firstp->link; p != P; p = p->link) {
1266*fbadb1c4SDavid du Colombier 		if(p->as == ATEXT) {
1267*fbadb1c4SDavid du Colombier 			q = prg();
1268*fbadb1c4SDavid du Colombier 			q->line = p->line;
1269*fbadb1c4SDavid du Colombier 			q->link = datap;
1270*fbadb1c4SDavid du Colombier 			datap = q;
1271*fbadb1c4SDavid du Colombier 			q->as = ADATA;
1272*fbadb1c4SDavid du Colombier 			q->from.type = D_OREG;
1273*fbadb1c4SDavid du Colombier 			q->from.name = D_EXTERN;
1274*fbadb1c4SDavid du Colombier 			q->from.offset = n*4;		/* TO DO: check */
1275*fbadb1c4SDavid du Colombier 			q->from.sym = s;
1276*fbadb1c4SDavid du Colombier 			q->reg = 4;
1277*fbadb1c4SDavid du Colombier 			q->to = p->from;
1278*fbadb1c4SDavid du Colombier 			q->to.type = D_CONST;
1279*fbadb1c4SDavid du Colombier 
1280*fbadb1c4SDavid du Colombier 			q = prg();
1281*fbadb1c4SDavid du Colombier 			q->line = p->line;
1282*fbadb1c4SDavid du Colombier 			q->pc = p->pc;
1283*fbadb1c4SDavid du Colombier 			q->link = p->link;
1284*fbadb1c4SDavid du Colombier 			p->link = q;
1285*fbadb1c4SDavid du Colombier 			p = q;
1286*fbadb1c4SDavid du Colombier 			p->as = AMOVW;
1287*fbadb1c4SDavid du Colombier 			p->from.type = D_OREG;
1288*fbadb1c4SDavid du Colombier 			p->from.name = D_EXTERN;
1289*fbadb1c4SDavid du Colombier 			p->from.sym = s;
1290*fbadb1c4SDavid du Colombier 			p->from.offset = n*4 + 4;
1291*fbadb1c4SDavid du Colombier 			p->to.type = D_REG;
1292*fbadb1c4SDavid du Colombier 			p->to.reg = REGTMP;
1293*fbadb1c4SDavid du Colombier 
1294*fbadb1c4SDavid du Colombier 			q = prg();
1295*fbadb1c4SDavid du Colombier 			q->line = p->line;
1296*fbadb1c4SDavid du Colombier 			q->pc = p->pc;
1297*fbadb1c4SDavid du Colombier 			q->link = p->link;
1298*fbadb1c4SDavid du Colombier 			p->link = q;
1299*fbadb1c4SDavid du Colombier 			p = q;
1300*fbadb1c4SDavid du Colombier 			p->as = AADD;
1301*fbadb1c4SDavid du Colombier 			p->from.type = D_CONST;
1302*fbadb1c4SDavid du Colombier 			p->from.offset = 1;
1303*fbadb1c4SDavid du Colombier 			p->to.type = D_REG;
1304*fbadb1c4SDavid du Colombier 			p->to.reg = REGTMP;
1305*fbadb1c4SDavid du Colombier 
1306*fbadb1c4SDavid du Colombier 			q = prg();
1307*fbadb1c4SDavid du Colombier 			q->line = p->line;
1308*fbadb1c4SDavid du Colombier 			q->pc = p->pc;
1309*fbadb1c4SDavid du Colombier 			q->link = p->link;
1310*fbadb1c4SDavid du Colombier 			p->link = q;
1311*fbadb1c4SDavid du Colombier 			p = q;
1312*fbadb1c4SDavid du Colombier 			p->as = AMOVW;
1313*fbadb1c4SDavid du Colombier 			p->from.type = D_REG;
1314*fbadb1c4SDavid du Colombier 			p->from.reg = REGTMP;
1315*fbadb1c4SDavid du Colombier 			p->to.type = D_OREG;
1316*fbadb1c4SDavid du Colombier 			p->to.name = D_EXTERN;
1317*fbadb1c4SDavid du Colombier 			p->to.sym = s;
1318*fbadb1c4SDavid du Colombier 			p->to.offset = n*4 + 4;
1319*fbadb1c4SDavid du Colombier 
1320*fbadb1c4SDavid du Colombier 			n += 2;
1321*fbadb1c4SDavid du Colombier 			continue;
1322*fbadb1c4SDavid du Colombier 		}
1323*fbadb1c4SDavid du Colombier 	}
1324*fbadb1c4SDavid du Colombier 	q = prg();
1325*fbadb1c4SDavid du Colombier 	q->line = 0;
1326*fbadb1c4SDavid du Colombier 	q->link = datap;
1327*fbadb1c4SDavid du Colombier 	datap = q;
1328*fbadb1c4SDavid du Colombier 
1329*fbadb1c4SDavid du Colombier 	q->as = ADATA;
1330*fbadb1c4SDavid du Colombier 	q->from.type = D_OREG;
1331*fbadb1c4SDavid du Colombier 	q->from.name = D_EXTERN;
1332*fbadb1c4SDavid du Colombier 	q->from.sym = s;
1333*fbadb1c4SDavid du Colombier 	q->reg = 4;
1334*fbadb1c4SDavid du Colombier 	q->to.type = D_CONST;
1335*fbadb1c4SDavid du Colombier 	q->to.offset = n;
1336*fbadb1c4SDavid du Colombier 
1337*fbadb1c4SDavid du Colombier 	s->type = SBSS;
1338*fbadb1c4SDavid du Colombier 	s->value = n*4;
1339*fbadb1c4SDavid du Colombier }
1340*fbadb1c4SDavid du Colombier 
1341*fbadb1c4SDavid du Colombier void
doprof2(void)1342*fbadb1c4SDavid du Colombier doprof2(void)
1343*fbadb1c4SDavid du Colombier {
1344*fbadb1c4SDavid du Colombier 	Sym *s2, *s4;
1345*fbadb1c4SDavid du Colombier 	Prog *p, *q, *ps2, *ps4;
1346*fbadb1c4SDavid du Colombier 
1347*fbadb1c4SDavid du Colombier 	if(debug['v'])
1348*fbadb1c4SDavid du Colombier 		Bprint(&bso, "%5.2f profile 2\n", cputime());
1349*fbadb1c4SDavid du Colombier 	Bflush(&bso);
1350*fbadb1c4SDavid du Colombier 
1351*fbadb1c4SDavid du Colombier 	s2 = lookup("_profin", 0);
1352*fbadb1c4SDavid du Colombier 	s4 = lookup("_profout", 0);
1353*fbadb1c4SDavid du Colombier 	if(s2->type != STEXT || s4->type != STEXT) {
1354*fbadb1c4SDavid du Colombier 		diag("_profin/_profout not defined");
1355*fbadb1c4SDavid du Colombier 		return;
1356*fbadb1c4SDavid du Colombier 	}
1357*fbadb1c4SDavid du Colombier 
1358*fbadb1c4SDavid du Colombier 	ps2 = P;
1359*fbadb1c4SDavid du Colombier 	ps4 = P;
1360*fbadb1c4SDavid du Colombier 	for(p = firstp; p != P; p = p->link) {
1361*fbadb1c4SDavid du Colombier 		if(p->as == ATEXT) {
1362*fbadb1c4SDavid du Colombier 			if(p->from.sym == s2) {
1363*fbadb1c4SDavid du Colombier 				p->reg = 1;
1364*fbadb1c4SDavid du Colombier 				ps2 = p;
1365*fbadb1c4SDavid du Colombier 			}
1366*fbadb1c4SDavid du Colombier 			if(p->from.sym == s4) {
1367*fbadb1c4SDavid du Colombier 				p->reg = 1;
1368*fbadb1c4SDavid du Colombier 				ps4 = p;
1369*fbadb1c4SDavid du Colombier 			}
1370*fbadb1c4SDavid du Colombier 		}
1371*fbadb1c4SDavid du Colombier 	}
1372*fbadb1c4SDavid du Colombier 	for(p = firstp; p != P; p = p->link) {
1373*fbadb1c4SDavid du Colombier 		if(p->as == ATEXT) {
1374*fbadb1c4SDavid du Colombier 			curtext = p;
1375*fbadb1c4SDavid du Colombier 
1376*fbadb1c4SDavid du Colombier 			if(p->reg & NOPROF) {	/* dont profile */
1377*fbadb1c4SDavid du Colombier 				for(;;) {
1378*fbadb1c4SDavid du Colombier 					q = p->link;
1379*fbadb1c4SDavid du Colombier 					if(q == P)
1380*fbadb1c4SDavid du Colombier 						break;
1381*fbadb1c4SDavid du Colombier 					if(q->as == ATEXT)
1382*fbadb1c4SDavid du Colombier 						break;
1383*fbadb1c4SDavid du Colombier 					p = q;
1384*fbadb1c4SDavid du Colombier 				}
1385*fbadb1c4SDavid du Colombier 				continue;
1386*fbadb1c4SDavid du Colombier 			}
1387*fbadb1c4SDavid du Colombier 
1388*fbadb1c4SDavid du Colombier 			/*
1389*fbadb1c4SDavid du Colombier 			 * BL	profin
1390*fbadb1c4SDavid du Colombier 			 */
1391*fbadb1c4SDavid du Colombier 			q = prg();
1392*fbadb1c4SDavid du Colombier 			q->line = p->line;
1393*fbadb1c4SDavid du Colombier 			q->pc = p->pc;
1394*fbadb1c4SDavid du Colombier 			q->link = p->link;
1395*fbadb1c4SDavid du Colombier 			p->link = q;
1396*fbadb1c4SDavid du Colombier 			p = q;
1397*fbadb1c4SDavid du Colombier 			p->as = ABL;
1398*fbadb1c4SDavid du Colombier 			p->to.type = D_BRANCH;
1399*fbadb1c4SDavid du Colombier 			p->cond = ps2;
1400*fbadb1c4SDavid du Colombier 			p->to.sym = s2;
1401*fbadb1c4SDavid du Colombier 
1402*fbadb1c4SDavid du Colombier 			continue;
1403*fbadb1c4SDavid du Colombier 		}
1404*fbadb1c4SDavid du Colombier 		if(p->as == ARETURN) {
1405*fbadb1c4SDavid du Colombier 
1406*fbadb1c4SDavid du Colombier 			/*
1407*fbadb1c4SDavid du Colombier 			 * RETURN
1408*fbadb1c4SDavid du Colombier 			 */
1409*fbadb1c4SDavid du Colombier 			q = prg();
1410*fbadb1c4SDavid du Colombier 			q->as = ARETURN;
1411*fbadb1c4SDavid du Colombier 			q->from = p->from;
1412*fbadb1c4SDavid du Colombier 			q->to = p->to;
1413*fbadb1c4SDavid du Colombier 			q->link = p->link;
1414*fbadb1c4SDavid du Colombier 			p->link = q;
1415*fbadb1c4SDavid du Colombier 
1416*fbadb1c4SDavid du Colombier 			/*
1417*fbadb1c4SDavid du Colombier 			 * BL profout
1418*fbadb1c4SDavid du Colombier 			 */
1419*fbadb1c4SDavid du Colombier 			p->as = ABL;
1420*fbadb1c4SDavid du Colombier 			p->from = zprg.from;
1421*fbadb1c4SDavid du Colombier 			p->to = zprg.to;
1422*fbadb1c4SDavid du Colombier 			p->to.type = D_BRANCH;
1423*fbadb1c4SDavid du Colombier 			p->cond = ps4;
1424*fbadb1c4SDavid du Colombier 			p->to.sym = s4;
1425*fbadb1c4SDavid du Colombier 
1426*fbadb1c4SDavid du Colombier 			p = q;
1427*fbadb1c4SDavid du Colombier 
1428*fbadb1c4SDavid du Colombier 			continue;
1429*fbadb1c4SDavid du Colombier 		}
1430*fbadb1c4SDavid du Colombier 	}
1431*fbadb1c4SDavid du Colombier }
1432*fbadb1c4SDavid du Colombier 
1433*fbadb1c4SDavid du Colombier void
nuxiinit(void)1434*fbadb1c4SDavid du Colombier nuxiinit(void)
1435*fbadb1c4SDavid du Colombier {
1436*fbadb1c4SDavid du Colombier 	int i, c;
1437*fbadb1c4SDavid du Colombier 
1438*fbadb1c4SDavid du Colombier 	for(i=0; i<4; i++) {
1439*fbadb1c4SDavid du Colombier 		c = find1(0x01020304L, i+1);
1440*fbadb1c4SDavid du Colombier 		if(i >= 2)
1441*fbadb1c4SDavid du Colombier 			inuxi2[i-2] = c;
1442*fbadb1c4SDavid du Colombier 		if(i >= 3)
1443*fbadb1c4SDavid du Colombier 			inuxi1[i-3] = c;
1444*fbadb1c4SDavid du Colombier 		inuxi4[i] = c;
1445*fbadb1c4SDavid du Colombier 		inuxi8[i] = c+4;
1446*fbadb1c4SDavid du Colombier 		inuxi8[i+4] = c;
1447*fbadb1c4SDavid du Colombier 		fnuxi4[i] = c;
1448*fbadb1c4SDavid du Colombier 		fnuxi8[i] = c+4;
1449*fbadb1c4SDavid du Colombier 		fnuxi8[i+4] = c;
1450*fbadb1c4SDavid du Colombier 	}
1451*fbadb1c4SDavid du Colombier 	if(debug['v']) {
1452*fbadb1c4SDavid du Colombier 		Bprint(&bso, "inuxi = ");
1453*fbadb1c4SDavid du Colombier 		for(i=0; i<1; i++)
1454*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", inuxi1[i]);
1455*fbadb1c4SDavid du Colombier 		Bprint(&bso, " ");
1456*fbadb1c4SDavid du Colombier 		for(i=0; i<2; i++)
1457*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", inuxi2[i]);
1458*fbadb1c4SDavid du Colombier 		Bprint(&bso, " ");
1459*fbadb1c4SDavid du Colombier 		for(i=0; i<4; i++)
1460*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", inuxi4[i]);
1461*fbadb1c4SDavid du Colombier 		Bprint(&bso, " ");
1462*fbadb1c4SDavid du Colombier 		for(i=0; i<8; i++)
1463*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", inuxi8[i]);
1464*fbadb1c4SDavid du Colombier 		Bprint(&bso, "\nfnuxi = ");
1465*fbadb1c4SDavid du Colombier 		for(i=0; i<4; i++)
1466*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", fnuxi4[i]);
1467*fbadb1c4SDavid du Colombier 		Bprint(&bso, "\nfnuxi = ");
1468*fbadb1c4SDavid du Colombier 		for(i=0; i<8; i++)
1469*fbadb1c4SDavid du Colombier 			Bprint(&bso, "%d", fnuxi8[i]);
1470*fbadb1c4SDavid du Colombier 		Bprint(&bso, "\n");
1471*fbadb1c4SDavid du Colombier 	}
1472*fbadb1c4SDavid du Colombier 	Bflush(&bso);
1473*fbadb1c4SDavid du Colombier }
1474*fbadb1c4SDavid du Colombier 
1475*fbadb1c4SDavid du Colombier int
find1(long l,int c)1476*fbadb1c4SDavid du Colombier find1(long l, int c)
1477*fbadb1c4SDavid du Colombier {
1478*fbadb1c4SDavid du Colombier 	char *p;
1479*fbadb1c4SDavid du Colombier 	int i;
1480*fbadb1c4SDavid du Colombier 
1481*fbadb1c4SDavid du Colombier 	p = (char*)&l;
1482*fbadb1c4SDavid du Colombier 	for(i=0; i<4; i++)
1483*fbadb1c4SDavid du Colombier 		if(*p++ == c)
1484*fbadb1c4SDavid du Colombier 			return i;
1485*fbadb1c4SDavid du Colombier 	return 0;
1486*fbadb1c4SDavid du Colombier }
1487*fbadb1c4SDavid du Colombier 
1488*fbadb1c4SDavid du Colombier long
ieeedtof(Ieee * ieeep)1489*fbadb1c4SDavid du Colombier ieeedtof(Ieee *ieeep)
1490*fbadb1c4SDavid du Colombier {
1491*fbadb1c4SDavid du Colombier 	int exp;
1492*fbadb1c4SDavid du Colombier 	long v;
1493*fbadb1c4SDavid du Colombier 
1494*fbadb1c4SDavid du Colombier 	if(ieeep->h == 0)
1495*fbadb1c4SDavid du Colombier 		return 0;
1496*fbadb1c4SDavid du Colombier 	exp = (ieeep->h>>20) & ((1L<<11)-1L);
1497*fbadb1c4SDavid du Colombier 	exp -= (1L<<10) - 2L;
1498*fbadb1c4SDavid du Colombier 	v = (ieeep->h & 0xfffffL) << 3;
1499*fbadb1c4SDavid du Colombier 	v |= (ieeep->l >> 29) & 0x7L;
1500*fbadb1c4SDavid du Colombier 	if((ieeep->l >> 28) & 1) {
1501*fbadb1c4SDavid du Colombier 		v++;
1502*fbadb1c4SDavid du Colombier 		if(v & 0x800000L) {
1503*fbadb1c4SDavid du Colombier 			v = (v & 0x7fffffL) >> 1;
1504*fbadb1c4SDavid du Colombier 			exp++;
1505*fbadb1c4SDavid du Colombier 		}
1506*fbadb1c4SDavid du Colombier 	}
1507*fbadb1c4SDavid du Colombier 	if(exp <= -126 || exp >= 130)
1508*fbadb1c4SDavid du Colombier 		diag("double fp to single fp overflow");
1509*fbadb1c4SDavid du Colombier 	v |= ((exp + 126) & 0xffL) << 23;
1510*fbadb1c4SDavid du Colombier 	v |= ieeep->h & 0x80000000L;
1511*fbadb1c4SDavid du Colombier 	return v;
1512*fbadb1c4SDavid du Colombier }
1513*fbadb1c4SDavid du Colombier 
1514*fbadb1c4SDavid du Colombier double
ieeedtod(Ieee * ieeep)1515*fbadb1c4SDavid du Colombier ieeedtod(Ieee *ieeep)
1516*fbadb1c4SDavid du Colombier {
1517*fbadb1c4SDavid du Colombier 	Ieee e;
1518*fbadb1c4SDavid du Colombier 	double fr;
1519*fbadb1c4SDavid du Colombier 	int exp;
1520*fbadb1c4SDavid du Colombier 
1521*fbadb1c4SDavid du Colombier 	if(ieeep->h & (1L<<31)) {
1522*fbadb1c4SDavid du Colombier 		e.h = ieeep->h & ~(1L<<31);
1523*fbadb1c4SDavid du Colombier 		e.l = ieeep->l;
1524*fbadb1c4SDavid du Colombier 		return -ieeedtod(&e);
1525*fbadb1c4SDavid du Colombier 	}
1526*fbadb1c4SDavid du Colombier 	if(ieeep->l == 0 && ieeep->h == 0)
1527*fbadb1c4SDavid du Colombier 		return 0;
1528*fbadb1c4SDavid du Colombier 	fr = ieeep->l & ((1L<<16)-1L);
1529*fbadb1c4SDavid du Colombier 	fr /= 1L<<16;
1530*fbadb1c4SDavid du Colombier 	fr += (ieeep->l>>16) & ((1L<<16)-1L);
1531*fbadb1c4SDavid du Colombier 	fr /= 1L<<16;
1532*fbadb1c4SDavid du Colombier 	fr += (ieeep->h & (1L<<20)-1L) | (1L<<20);
1533*fbadb1c4SDavid du Colombier 	fr /= 1L<<21;
1534*fbadb1c4SDavid du Colombier 	exp = (ieeep->h>>20) & ((1L<<11)-1L);
1535*fbadb1c4SDavid du Colombier 	exp -= (1L<<10) - 2L;
1536*fbadb1c4SDavid du Colombier 	return ldexp(fr, exp);
1537*fbadb1c4SDavid du Colombier }
1538*fbadb1c4SDavid du Colombier 
1539*fbadb1c4SDavid du Colombier void
undefsym(Sym * s)1540*fbadb1c4SDavid du Colombier undefsym(Sym *s)
1541*fbadb1c4SDavid du Colombier {
1542*fbadb1c4SDavid du Colombier 	int n;
1543*fbadb1c4SDavid du Colombier 
1544*fbadb1c4SDavid du Colombier 	n = imports;
1545*fbadb1c4SDavid du Colombier 	if(s->value != 0)
1546*fbadb1c4SDavid du Colombier 		diag("value != 0 on SXREF");
1547*fbadb1c4SDavid du Colombier 	if(n >= 1<<Rindex)
1548*fbadb1c4SDavid du Colombier 		diag("import index %d out of range", n);
1549*fbadb1c4SDavid du Colombier 	s->value = n<<Roffset;
1550*fbadb1c4SDavid du Colombier 	s->type = SUNDEF;
1551*fbadb1c4SDavid du Colombier 	imports++;
1552*fbadb1c4SDavid du Colombier }
1553*fbadb1c4SDavid du Colombier 
1554*fbadb1c4SDavid du Colombier void
zerosig(char * sp)1555*fbadb1c4SDavid du Colombier zerosig(char *sp)
1556*fbadb1c4SDavid du Colombier {
1557*fbadb1c4SDavid du Colombier 	Sym *s;
1558*fbadb1c4SDavid du Colombier 
1559*fbadb1c4SDavid du Colombier 	s = lookup(sp, 0);
1560*fbadb1c4SDavid du Colombier 	s->sig = 0;
1561*fbadb1c4SDavid du Colombier }
1562*fbadb1c4SDavid du Colombier 
1563*fbadb1c4SDavid du Colombier void
readundefs(char * f,int t)1564*fbadb1c4SDavid du Colombier readundefs(char *f, int t)
1565*fbadb1c4SDavid du Colombier {
1566*fbadb1c4SDavid du Colombier 	int i, n;
1567*fbadb1c4SDavid du Colombier 	Sym *s;
1568*fbadb1c4SDavid du Colombier 	Biobuf *b;
1569*fbadb1c4SDavid du Colombier 	char *l, buf[256], *fields[64];
1570*fbadb1c4SDavid du Colombier 
1571*fbadb1c4SDavid du Colombier 	if(f == nil)
1572*fbadb1c4SDavid du Colombier 		return;
1573*fbadb1c4SDavid du Colombier 	b = Bopen(f, OREAD);
1574*fbadb1c4SDavid du Colombier 	if(b == nil){
1575*fbadb1c4SDavid du Colombier 		diag("could not open %s: %r", f);
1576*fbadb1c4SDavid du Colombier 		errorexit();
1577*fbadb1c4SDavid du Colombier 	}
1578*fbadb1c4SDavid du Colombier 	while((l = Brdline(b, '\n')) != nil){
1579*fbadb1c4SDavid du Colombier 		n = Blinelen(b);
1580*fbadb1c4SDavid du Colombier 		if(n >= sizeof(buf)){
1581*fbadb1c4SDavid du Colombier 			diag("%s: line too long", f);
1582*fbadb1c4SDavid du Colombier 			errorexit();
1583*fbadb1c4SDavid du Colombier 		}
1584*fbadb1c4SDavid du Colombier 		memmove(buf, l, n);
1585*fbadb1c4SDavid du Colombier 		buf[n-1] = '\0';
1586*fbadb1c4SDavid du Colombier 		n = getfields(buf, fields, nelem(fields), 1, " \t\r\n");
1587*fbadb1c4SDavid du Colombier 		if(n == nelem(fields)){
1588*fbadb1c4SDavid du Colombier 			diag("%s: bad format", f);
1589*fbadb1c4SDavid du Colombier 			errorexit();
1590*fbadb1c4SDavid du Colombier 		}
1591*fbadb1c4SDavid du Colombier 		for(i = 0; i < n; i++){
1592*fbadb1c4SDavid du Colombier 			s = lookup(fields[i], 0);
1593*fbadb1c4SDavid du Colombier 			s->type = SXREF;
1594*fbadb1c4SDavid du Colombier 			s->subtype = t;
1595*fbadb1c4SDavid du Colombier 			if(t == SIMPORT)
1596*fbadb1c4SDavid du Colombier 				nimports++;
1597*fbadb1c4SDavid du Colombier 			else
1598*fbadb1c4SDavid du Colombier 				nexports++;
1599*fbadb1c4SDavid du Colombier 		}
1600*fbadb1c4SDavid du Colombier 	}
1601*fbadb1c4SDavid du Colombier 	Bterm(b);
1602*fbadb1c4SDavid du Colombier }
1603