xref: /plan9-contrib/sys/src/cmd/ld/pobj.c (revision 40d015479ed36701ae6dcfd8814f849fc6285e8d)
1*40d01547SDavid du Colombier #include	"l.h"
2*40d01547SDavid du Colombier #include	<ar.h>
3*40d01547SDavid du Colombier 
4*40d01547SDavid du Colombier char	*noname		= "<none>";
5*40d01547SDavid du Colombier char	symname[]	= SYMDEF;
6*40d01547SDavid du Colombier 
7*40d01547SDavid du Colombier char**	libdir;
8*40d01547SDavid du Colombier int	nlibdir	= 0;
9*40d01547SDavid du Colombier static	int	maxlibdir = 0;
10*40d01547SDavid du Colombier 
11*40d01547SDavid du Colombier int
isobjfile(char * f)12*40d01547SDavid du Colombier isobjfile(char *f)
13*40d01547SDavid du Colombier {
14*40d01547SDavid du Colombier 	int n, v;
15*40d01547SDavid du Colombier 	Biobuf *b;
16*40d01547SDavid du Colombier 	char buf1[5], buf2[SARMAG];
17*40d01547SDavid du Colombier 
18*40d01547SDavid du Colombier 	b = Bopen(f, OREAD);
19*40d01547SDavid du Colombier 	if(b == nil)
20*40d01547SDavid du Colombier 		return 0;
21*40d01547SDavid du Colombier 	n = Bread(b, buf1, 5);
22*40d01547SDavid du Colombier 	if(n == 5 && (buf1[2] == 1 && buf1[3] == '<' || buf1[3] == 1 && buf1[4] == '<'))
23*40d01547SDavid du Colombier 		v = 1;	/* good enough for our purposes */
24*40d01547SDavid du Colombier 	else{
25*40d01547SDavid du Colombier 		Bseek(b, 0, 0);
26*40d01547SDavid du Colombier 		n = Bread(b, buf2, SARMAG);
27*40d01547SDavid du Colombier 		v = n == SARMAG && strncmp(buf2, ARMAG, SARMAG) == 0;
28*40d01547SDavid du Colombier 	}
29*40d01547SDavid du Colombier 	Bterm(b);
30*40d01547SDavid du Colombier 	return v;
31*40d01547SDavid du Colombier }
32*40d01547SDavid du Colombier 
33*40d01547SDavid du Colombier void
loadlib(void)34*40d01547SDavid du Colombier loadlib(void)
35*40d01547SDavid du Colombier {
36*40d01547SDavid du Colombier 	int i;
37*40d01547SDavid du Colombier 	long h;
38*40d01547SDavid du Colombier 	Sym *s;
39*40d01547SDavid du Colombier 
40*40d01547SDavid du Colombier loop:
41*40d01547SDavid du Colombier 	xrefresolv = 0;
42*40d01547SDavid du Colombier 	for(i=0; i<libraryp; i++) {
43*40d01547SDavid du Colombier 		if(debug['v'])
44*40d01547SDavid du Colombier 			Bprint(&bso, "%5.2f autolib: %s (from %s)\n", cputime(), library[i], libraryobj[i]);
45*40d01547SDavid du Colombier 		objfile(library[i]);
46*40d01547SDavid du Colombier 	}
47*40d01547SDavid du Colombier 	if(xrefresolv)
48*40d01547SDavid du Colombier 	for(h=0; h<nelem(hash); h++)
49*40d01547SDavid du Colombier 	for(s = hash[h]; s != S; s = s->link)
50*40d01547SDavid du Colombier 		if(s->type == SXREF)
51*40d01547SDavid du Colombier 			goto loop;
52*40d01547SDavid du Colombier }
53*40d01547SDavid du Colombier 
54*40d01547SDavid du Colombier void
objfile(char * file)55*40d01547SDavid du Colombier objfile(char *file)
56*40d01547SDavid du Colombier {
57*40d01547SDavid du Colombier 	long off, esym, cnt, l;
58*40d01547SDavid du Colombier 	int f, work;
59*40d01547SDavid du Colombier 	Sym *s;
60*40d01547SDavid du Colombier 	char magbuf[SARMAG];
61*40d01547SDavid du Colombier 	char name[LIBNAMELEN], pname[LIBNAMELEN];
62*40d01547SDavid du Colombier 	struct ar_hdr arhdr;
63*40d01547SDavid du Colombier 	char *e, *start, *stop;
64*40d01547SDavid du Colombier 
65*40d01547SDavid du Colombier 	if(debug['v'])
66*40d01547SDavid du Colombier 		Bprint(&bso, "%5.2f ldobj: %s\n", cputime(), file);
67*40d01547SDavid du Colombier 	Bflush(&bso);
68*40d01547SDavid du Colombier 	if(file[0] == '-' && file[1] == 'l') {
69*40d01547SDavid du Colombier 		snprint(pname, sizeof(pname), "lib%s.a", file+2);
70*40d01547SDavid du Colombier 		e = findlib(pname);
71*40d01547SDavid du Colombier 		if(e == nil) {
72*40d01547SDavid du Colombier 			diag("cannot find library: %s", file);
73*40d01547SDavid du Colombier 			errorexit();
74*40d01547SDavid du Colombier 		}
75*40d01547SDavid du Colombier 		snprint(name, sizeof(name), "%s/%s", e, pname);
76*40d01547SDavid du Colombier 		file = name;
77*40d01547SDavid du Colombier 	}
78*40d01547SDavid du Colombier 	f = open(file, 0);
79*40d01547SDavid du Colombier 	if(f < 0) {
80*40d01547SDavid du Colombier 		diag("cannot open %s: %r", file);
81*40d01547SDavid du Colombier 		errorexit();
82*40d01547SDavid du Colombier 	}
83*40d01547SDavid du Colombier 	l = read(f, magbuf, SARMAG);
84*40d01547SDavid du Colombier 	if(l != SARMAG || strncmp(magbuf, ARMAG, SARMAG)){
85*40d01547SDavid du Colombier 		/* load it as a regular file */
86*40d01547SDavid du Colombier 		l = seek(f, 0L, 2);
87*40d01547SDavid du Colombier 		seek(f, 0L, 0);
88*40d01547SDavid du Colombier 		ldobj(f, l, file);
89*40d01547SDavid du Colombier 		close(f);
90*40d01547SDavid du Colombier 		return;
91*40d01547SDavid du Colombier 	}
92*40d01547SDavid du Colombier 
93*40d01547SDavid du Colombier 	if(debug['v'])
94*40d01547SDavid du Colombier 		Bprint(&bso, "%5.2f ldlib: %s\n", cputime(), file);
95*40d01547SDavid du Colombier 	l = read(f, &arhdr, SAR_HDR);
96*40d01547SDavid du Colombier 	if(l != SAR_HDR) {
97*40d01547SDavid du Colombier 		diag("%s: short read on archive file symbol header", file);
98*40d01547SDavid du Colombier 		goto out;
99*40d01547SDavid du Colombier 	}
100*40d01547SDavid du Colombier 	if(strncmp(arhdr.name, symname, strlen(symname))) {
101*40d01547SDavid du Colombier 		diag("%s: first entry not symbol header", file);
102*40d01547SDavid du Colombier 		goto out;
103*40d01547SDavid du Colombier 	}
104*40d01547SDavid du Colombier 
105*40d01547SDavid du Colombier 	esym = SARMAG + SAR_HDR + atolwhex(arhdr.size);
106*40d01547SDavid du Colombier 	off = SARMAG + SAR_HDR;
107*40d01547SDavid du Colombier 
108*40d01547SDavid du Colombier 	/*
109*40d01547SDavid du Colombier 	 * just bang the whole symbol file into memory
110*40d01547SDavid du Colombier 	 */
111*40d01547SDavid du Colombier 	seek(f, off, 0);
112*40d01547SDavid du Colombier 	cnt = esym - off;
113*40d01547SDavid du Colombier 	start = malloc(cnt + 10);
114*40d01547SDavid du Colombier 	cnt = read(f, start, cnt);
115*40d01547SDavid du Colombier 	if(cnt <= 0){
116*40d01547SDavid du Colombier 		close(f);
117*40d01547SDavid du Colombier 		return;
118*40d01547SDavid du Colombier 	}
119*40d01547SDavid du Colombier 	stop = &start[cnt];
120*40d01547SDavid du Colombier 	memset(stop, 0, 10);
121*40d01547SDavid du Colombier 
122*40d01547SDavid du Colombier 	work = 1;
123*40d01547SDavid du Colombier 	while(work){
124*40d01547SDavid du Colombier 		if(debug['v'])
125*40d01547SDavid du Colombier 			Bprint(&bso, "%5.2f library pass: %s\n", cputime(), file);
126*40d01547SDavid du Colombier 		Bflush(&bso);
127*40d01547SDavid du Colombier 		work = 0;
128*40d01547SDavid du Colombier 		for(e = start; e < stop; e = strchr(e+5, 0) + 1) {
129*40d01547SDavid du Colombier 			s = lookup(e+5, 0);
130*40d01547SDavid du Colombier 			if(s->type != SXREF)
131*40d01547SDavid du Colombier 				continue;
132*40d01547SDavid du Colombier 			sprint(pname, "%s(%s)", file, s->name);
133*40d01547SDavid du Colombier 			if(debug['v'])
134*40d01547SDavid du Colombier 				Bprint(&bso, "%5.2f library: %s\n", cputime(), pname);
135*40d01547SDavid du Colombier 			Bflush(&bso);
136*40d01547SDavid du Colombier 			l = e[1] & 0xff;
137*40d01547SDavid du Colombier 			l |= (e[2] & 0xff) << 8;
138*40d01547SDavid du Colombier 			l |= (e[3] & 0xff) << 16;
139*40d01547SDavid du Colombier 			l |= (e[4] & 0xff) << 24;
140*40d01547SDavid du Colombier 			seek(f, l, 0);
141*40d01547SDavid du Colombier 			l = read(f, &arhdr, SAR_HDR);
142*40d01547SDavid du Colombier 			if(l != SAR_HDR)
143*40d01547SDavid du Colombier 				goto bad;
144*40d01547SDavid du Colombier 			if(strncmp(arhdr.fmag, ARFMAG, sizeof(arhdr.fmag)))
145*40d01547SDavid du Colombier 				goto bad;
146*40d01547SDavid du Colombier 			l = atolwhex(arhdr.size);
147*40d01547SDavid du Colombier 			ldobj(f, l, pname);
148*40d01547SDavid du Colombier 			if(s->type == SXREF) {
149*40d01547SDavid du Colombier 				diag("%s: failed to load: %s", file, s->name);
150*40d01547SDavid du Colombier 				errorexit();
151*40d01547SDavid du Colombier 			}
152*40d01547SDavid du Colombier 			work = 1;
153*40d01547SDavid du Colombier 			xrefresolv = 1;
154*40d01547SDavid du Colombier 		}
155*40d01547SDavid du Colombier 	}
156*40d01547SDavid du Colombier 	return;
157*40d01547SDavid du Colombier 
158*40d01547SDavid du Colombier bad:
159*40d01547SDavid du Colombier 	diag("%s: bad or out of date archive", file);
160*40d01547SDavid du Colombier out:
161*40d01547SDavid du Colombier 	close(f);
162*40d01547SDavid du Colombier }
163*40d01547SDavid du Colombier 
164*40d01547SDavid du Colombier void
addlibpath(char * arg)165*40d01547SDavid du Colombier addlibpath(char *arg)
166*40d01547SDavid du Colombier {
167*40d01547SDavid du Colombier 	char **p;
168*40d01547SDavid du Colombier 
169*40d01547SDavid du Colombier 	if(nlibdir >= maxlibdir) {
170*40d01547SDavid du Colombier 		if(maxlibdir == 0)
171*40d01547SDavid du Colombier 			maxlibdir = 8;
172*40d01547SDavid du Colombier 		else
173*40d01547SDavid du Colombier 			maxlibdir *= 2;
174*40d01547SDavid du Colombier 		p = malloc(maxlibdir*sizeof(*p));
175*40d01547SDavid du Colombier 		if(p == nil) {
176*40d01547SDavid du Colombier 			diag("out of memory");
177*40d01547SDavid du Colombier 			errorexit();
178*40d01547SDavid du Colombier 		}
179*40d01547SDavid du Colombier 		memmove(p, libdir, nlibdir*sizeof(*p));
180*40d01547SDavid du Colombier 		free(libdir);
181*40d01547SDavid du Colombier 		libdir = p;
182*40d01547SDavid du Colombier 	}
183*40d01547SDavid du Colombier 	libdir[nlibdir++] = strdup(arg);
184*40d01547SDavid du Colombier }
185*40d01547SDavid du Colombier 
186*40d01547SDavid du Colombier char*
findlib(char * file)187*40d01547SDavid du Colombier findlib(char *file)
188*40d01547SDavid du Colombier {
189*40d01547SDavid du Colombier 	int i;
190*40d01547SDavid du Colombier 	char name[LIBNAMELEN];
191*40d01547SDavid du Colombier 
192*40d01547SDavid du Colombier 	for(i = 0; i < nlibdir; i++) {
193*40d01547SDavid du Colombier 		snprint(name, sizeof(name), "%s/%s", libdir[i], file);
194*40d01547SDavid du Colombier 		if(fileexists(name))
195*40d01547SDavid du Colombier 			return libdir[i];
196*40d01547SDavid du Colombier 	}
197*40d01547SDavid du Colombier 	return nil;
198*40d01547SDavid du Colombier }
199*40d01547SDavid du Colombier 
200*40d01547SDavid du Colombier void
addlibroot(void)201*40d01547SDavid du Colombier addlibroot(void)
202*40d01547SDavid du Colombier {
203*40d01547SDavid du Colombier 	char *a;
204*40d01547SDavid du Colombier 	char name[LIBNAMELEN];
205*40d01547SDavid du Colombier 
206*40d01547SDavid du Colombier 	a = getenv("ccroot");
207*40d01547SDavid du Colombier 	if(a != nil && *a != '\0') {
208*40d01547SDavid du Colombier 		if(!fileexists(a)) {
209*40d01547SDavid du Colombier 			diag("nonexistent $ccroot: %s", a);
210*40d01547SDavid du Colombier 			errorexit();
211*40d01547SDavid du Colombier 		}
212*40d01547SDavid du Colombier 	}else
213*40d01547SDavid du Colombier 		a = "";
214*40d01547SDavid du Colombier 	snprint(name, sizeof(name), "%s/%s/lib", a, thestring);
215*40d01547SDavid du Colombier 	addlibpath(name);
216*40d01547SDavid du Colombier }
217*40d01547SDavid du Colombier 
218*40d01547SDavid du Colombier void
addlib(char * obj)219*40d01547SDavid du Colombier addlib(char *obj)
220*40d01547SDavid du Colombier {
221*40d01547SDavid du Colombier 	char fn1[LIBNAMELEN], fn2[LIBNAMELEN], comp[LIBNAMELEN], *p, *name;
222*40d01547SDavid du Colombier 	int i, search;
223*40d01547SDavid du Colombier 
224*40d01547SDavid du Colombier 	if(histfrogp <= 0)
225*40d01547SDavid du Colombier 		return;
226*40d01547SDavid du Colombier 
227*40d01547SDavid du Colombier 	name = fn1;
228*40d01547SDavid du Colombier 	search = 0;
229*40d01547SDavid du Colombier 	if(histfrog[0]->name[1] == '/') {
230*40d01547SDavid du Colombier 		sprint(name, "");
231*40d01547SDavid du Colombier 		i = 1;
232*40d01547SDavid du Colombier 	} else if(histfrog[0]->name[1] == '.') {
233*40d01547SDavid du Colombier 		sprint(name, ".");
234*40d01547SDavid du Colombier 		i = 0;
235*40d01547SDavid du Colombier 	} else {
236*40d01547SDavid du Colombier 		sprint(name, "");
237*40d01547SDavid du Colombier 		i = 0;
238*40d01547SDavid du Colombier 		search = 1;
239*40d01547SDavid du Colombier 	}
240*40d01547SDavid du Colombier 
241*40d01547SDavid du Colombier 	for(; i<histfrogp; i++) {
242*40d01547SDavid du Colombier 		snprint(comp, sizeof comp, histfrog[i]->name+1);
243*40d01547SDavid du Colombier 		for(;;) {
244*40d01547SDavid du Colombier 			p = strstr(comp, "$O");
245*40d01547SDavid du Colombier 			if(p == 0)
246*40d01547SDavid du Colombier 				break;
247*40d01547SDavid du Colombier 			memmove(p+1, p+2, strlen(p+2)+1);
248*40d01547SDavid du Colombier 			p[0] = thechar;
249*40d01547SDavid du Colombier 		}
250*40d01547SDavid du Colombier 		for(;;) {
251*40d01547SDavid du Colombier 			p = strstr(comp, "$M");
252*40d01547SDavid du Colombier 			if(p == 0)
253*40d01547SDavid du Colombier 				break;
254*40d01547SDavid du Colombier 			if(strlen(comp)+strlen(thestring)-2+1 >= sizeof comp) {
255*40d01547SDavid du Colombier 				diag("library component too long");
256*40d01547SDavid du Colombier 				return;
257*40d01547SDavid du Colombier 			}
258*40d01547SDavid du Colombier 			memmove(p+strlen(thestring), p+2, strlen(p+2)+1);
259*40d01547SDavid du Colombier 			memmove(p, thestring, strlen(thestring));
260*40d01547SDavid du Colombier 		}
261*40d01547SDavid du Colombier 		if(strlen(fn1) + strlen(comp) + 3 >= sizeof(fn1)) {
262*40d01547SDavid du Colombier 			diag("library component too long");
263*40d01547SDavid du Colombier 			return;
264*40d01547SDavid du Colombier 		}
265*40d01547SDavid du Colombier 		if(i > 0 || !search)
266*40d01547SDavid du Colombier 			strcat(fn1, "/");
267*40d01547SDavid du Colombier 		strcat(fn1, comp);
268*40d01547SDavid du Colombier 	}
269*40d01547SDavid du Colombier 
270*40d01547SDavid du Colombier 	cleanname(name);
271*40d01547SDavid du Colombier 
272*40d01547SDavid du Colombier 	if(search){
273*40d01547SDavid du Colombier 		p = findlib(name);
274*40d01547SDavid du Colombier 		if(p != nil){
275*40d01547SDavid du Colombier 			snprint(fn2, sizeof(fn2), "%s/%s", p, name);
276*40d01547SDavid du Colombier 			name = fn2;
277*40d01547SDavid du Colombier 		}
278*40d01547SDavid du Colombier 	}
279*40d01547SDavid du Colombier 
280*40d01547SDavid du Colombier 	for(i=0; i<libraryp; i++)
281*40d01547SDavid du Colombier 		if(strcmp(name, library[i]) == 0)
282*40d01547SDavid du Colombier 			return;
283*40d01547SDavid du Colombier 	if(libraryp == nelem(library)){
284*40d01547SDavid du Colombier 		diag("too many autolibs; skipping %s", name);
285*40d01547SDavid du Colombier 		return;
286*40d01547SDavid du Colombier 	}
287*40d01547SDavid du Colombier 
288*40d01547SDavid du Colombier 	p = malloc(strlen(name) + 1);
289*40d01547SDavid du Colombier 	strcpy(p, name);
290*40d01547SDavid du Colombier 	library[libraryp] = p;
291*40d01547SDavid du Colombier 	p = malloc(strlen(obj) + 1);
292*40d01547SDavid du Colombier 	strcpy(p, obj);
293*40d01547SDavid du Colombier 	libraryobj[libraryp] = p;
294*40d01547SDavid du Colombier 	libraryp++;
295*40d01547SDavid du Colombier }
296*40d01547SDavid du Colombier 
297*40d01547SDavid du Colombier void
addhist(long line,int type)298*40d01547SDavid du Colombier addhist(long line, int type)
299*40d01547SDavid du Colombier {
300*40d01547SDavid du Colombier 	Auto *u;
301*40d01547SDavid du Colombier 	Sym *s;
302*40d01547SDavid du Colombier 	int i, j, k;
303*40d01547SDavid du Colombier 
304*40d01547SDavid du Colombier 	u = malloc(sizeof(Auto));
305*40d01547SDavid du Colombier 	s = malloc(sizeof(Sym));
306*40d01547SDavid du Colombier 	s->name = malloc(2*(histfrogp+1) + 1);
307*40d01547SDavid du Colombier 
308*40d01547SDavid du Colombier 	u->asym = s;
309*40d01547SDavid du Colombier 	u->type = type;
310*40d01547SDavid du Colombier 	u->aoffset = line;
311*40d01547SDavid du Colombier 	u->link = curhist;
312*40d01547SDavid du Colombier 	curhist = u;
313*40d01547SDavid du Colombier 
314*40d01547SDavid du Colombier 	j = 1;
315*40d01547SDavid du Colombier 	for(i=0; i<histfrogp; i++) {
316*40d01547SDavid du Colombier 		k = histfrog[i]->value;
317*40d01547SDavid du Colombier 		s->name[j+0] = k>>8;
318*40d01547SDavid du Colombier 		s->name[j+1] = k;
319*40d01547SDavid du Colombier 		j += 2;
320*40d01547SDavid du Colombier 	}
321*40d01547SDavid du Colombier }
322*40d01547SDavid du Colombier 
323*40d01547SDavid du Colombier void
histtoauto(void)324*40d01547SDavid du Colombier histtoauto(void)
325*40d01547SDavid du Colombier {
326*40d01547SDavid du Colombier 	Auto *l;
327*40d01547SDavid du Colombier 
328*40d01547SDavid du Colombier 	while((l = curhist) != nil) {
329*40d01547SDavid du Colombier 		curhist = l->link;
330*40d01547SDavid du Colombier 		l->link = curauto;
331*40d01547SDavid du Colombier 		curauto = l;
332*40d01547SDavid du Colombier 	}
333*40d01547SDavid du Colombier }
334*40d01547SDavid du Colombier 
335*40d01547SDavid du Colombier void
collapsefrog(Sym * s)336*40d01547SDavid du Colombier collapsefrog(Sym *s)
337*40d01547SDavid du Colombier {
338*40d01547SDavid du Colombier 	int i;
339*40d01547SDavid du Colombier 
340*40d01547SDavid du Colombier 	/*
341*40d01547SDavid du Colombier 	 * bad encoding of path components only allows
342*40d01547SDavid du Colombier 	 * MAXHIST components. if there is an overflow,
343*40d01547SDavid du Colombier 	 * first try to collapse xxx/..
344*40d01547SDavid du Colombier 	 */
345*40d01547SDavid du Colombier 	for(i=1; i<histfrogp; i++)
346*40d01547SDavid du Colombier 		if(strcmp(histfrog[i]->name+1, "..") == 0) {
347*40d01547SDavid du Colombier 			memmove(histfrog+i-1, histfrog+i+1,
348*40d01547SDavid du Colombier 				(histfrogp-i-1)*sizeof(histfrog[0]));
349*40d01547SDavid du Colombier 			histfrogp--;
350*40d01547SDavid du Colombier 			goto out;
351*40d01547SDavid du Colombier 		}
352*40d01547SDavid du Colombier 
353*40d01547SDavid du Colombier 	/*
354*40d01547SDavid du Colombier 	 * next try to collapse .
355*40d01547SDavid du Colombier 	 */
356*40d01547SDavid du Colombier 	for(i=0; i<histfrogp; i++)
357*40d01547SDavid du Colombier 		if(strcmp(histfrog[i]->name+1, ".") == 0) {
358*40d01547SDavid du Colombier 			memmove(histfrog+i, histfrog+i+1,
359*40d01547SDavid du Colombier 				(histfrogp-i-1)*sizeof(histfrog[0]));
360*40d01547SDavid du Colombier 			goto out;
361*40d01547SDavid du Colombier 		}
362*40d01547SDavid du Colombier 
363*40d01547SDavid du Colombier 	/*
364*40d01547SDavid du Colombier 	 * last chance, just truncate from front
365*40d01547SDavid du Colombier 	 */
366*40d01547SDavid du Colombier 	memmove(histfrog+0, histfrog+1,
367*40d01547SDavid du Colombier 		(histfrogp-1)*sizeof(histfrog[0]));
368*40d01547SDavid du Colombier 
369*40d01547SDavid du Colombier out:
370*40d01547SDavid du Colombier 	histfrog[histfrogp-1] = s;
371*40d01547SDavid du Colombier }
372*40d01547SDavid du Colombier 
373*40d01547SDavid du Colombier uchar*
readsome(int f,uchar * buf,uchar * good,uchar * stop,int max)374*40d01547SDavid du Colombier readsome(int f, uchar *buf, uchar *good, uchar *stop, int max)
375*40d01547SDavid du Colombier {
376*40d01547SDavid du Colombier 	int n;
377*40d01547SDavid du Colombier 
378*40d01547SDavid du Colombier 	n = stop - good;
379*40d01547SDavid du Colombier 	memmove(buf, good, stop - good);
380*40d01547SDavid du Colombier 	stop = buf + n;
381*40d01547SDavid du Colombier 	n = MAXIO - n;
382*40d01547SDavid du Colombier 	if(n > max)
383*40d01547SDavid du Colombier 		n = max;
384*40d01547SDavid du Colombier 	n = read(f, stop, n);
385*40d01547SDavid du Colombier 	if(n <= 0)
386*40d01547SDavid du Colombier 		return 0;
387*40d01547SDavid du Colombier 	return stop + n;
388*40d01547SDavid du Colombier }
389*40d01547SDavid du Colombier 
390*40d01547SDavid du Colombier Sym*
lookup(char * symb,int v)391*40d01547SDavid du Colombier lookup(char *symb, int v)
392*40d01547SDavid du Colombier {
393*40d01547SDavid du Colombier 	Sym *s;
394*40d01547SDavid du Colombier 	char *p;
395*40d01547SDavid du Colombier 	long h;
396*40d01547SDavid du Colombier 	int c, l;
397*40d01547SDavid du Colombier 
398*40d01547SDavid du Colombier 	h = v;
399*40d01547SDavid du Colombier 	for(p=symb; c = *p; p++)
400*40d01547SDavid du Colombier 		h = h+h+h + c;
401*40d01547SDavid du Colombier 	l = (p - symb) + 1;
402*40d01547SDavid du Colombier 	if(h < 0)
403*40d01547SDavid du Colombier 		h = ~h;
404*40d01547SDavid du Colombier 	h %= NHASH;
405*40d01547SDavid du Colombier 	for(s = hash[h]; s != S; s = s->link)
406*40d01547SDavid du Colombier 		if(s->version == v)
407*40d01547SDavid du Colombier 		if(memcmp(s->name, symb, l) == 0)
408*40d01547SDavid du Colombier 			return s;
409*40d01547SDavid du Colombier 
410*40d01547SDavid du Colombier 	s = halloc(sizeof(Sym));
411*40d01547SDavid du Colombier 	s->name = malloc(l + 1);
412*40d01547SDavid du Colombier 	memmove(s->name, symb, l);
413*40d01547SDavid du Colombier 
414*40d01547SDavid du Colombier 	s->link = hash[h];
415*40d01547SDavid du Colombier 	s->type = 0;
416*40d01547SDavid du Colombier 	s->version = v;
417*40d01547SDavid du Colombier 	s->value = 0;
418*40d01547SDavid du Colombier 	s->sig = 0;
419*40d01547SDavid du Colombier 	hash[h] = s;
420*40d01547SDavid du Colombier 	return s;
421*40d01547SDavid du Colombier }
422*40d01547SDavid du Colombier 
423*40d01547SDavid du Colombier int
find1(long l,int c)424*40d01547SDavid du Colombier find1(long l, int c)
425*40d01547SDavid du Colombier {
426*40d01547SDavid du Colombier 	char *p;
427*40d01547SDavid du Colombier 	int i;
428*40d01547SDavid du Colombier 
429*40d01547SDavid du Colombier 	p = (char*)&l;
430*40d01547SDavid du Colombier 	for(i=0; i<4; i++)
431*40d01547SDavid du Colombier 		if(*p++ == c)
432*40d01547SDavid du Colombier 			return i;
433*40d01547SDavid du Colombier 	return 0;
434*40d01547SDavid du Colombier }
435*40d01547SDavid du Colombier 
436*40d01547SDavid du Colombier long
ieeedtof(Ieee * ieeep)437*40d01547SDavid du Colombier ieeedtof(Ieee *ieeep)
438*40d01547SDavid du Colombier {
439*40d01547SDavid du Colombier 	int exp;
440*40d01547SDavid du Colombier 	long v;
441*40d01547SDavid du Colombier 
442*40d01547SDavid du Colombier 	if(ieeep->h == 0)
443*40d01547SDavid du Colombier 		return 0;
444*40d01547SDavid du Colombier 	exp = (ieeep->h>>20) & ((1L<<11)-1L);
445*40d01547SDavid du Colombier 	exp -= (1L<<10) - 2L;
446*40d01547SDavid du Colombier 	v = (ieeep->h & 0xfffffL) << 3;
447*40d01547SDavid du Colombier 	v |= (ieeep->l >> 29) & 0x7L;
448*40d01547SDavid du Colombier 	if((ieeep->l >> 28) & 1) {
449*40d01547SDavid du Colombier 		v++;
450*40d01547SDavid du Colombier 		if(v & 0x800000L) {
451*40d01547SDavid du Colombier 			v = (v & 0x7fffffL) >> 1;
452*40d01547SDavid du Colombier 			exp++;
453*40d01547SDavid du Colombier 		}
454*40d01547SDavid du Colombier 	}
455*40d01547SDavid du Colombier 	if(exp <= -126 || exp >= 130)
456*40d01547SDavid du Colombier 		diag("double fp to single fp overflow");
457*40d01547SDavid du Colombier 	v |= ((exp + 126) & 0xffL) << 23;
458*40d01547SDavid du Colombier 	v |= ieeep->h & 0x80000000L;
459*40d01547SDavid du Colombier 	return v;
460*40d01547SDavid du Colombier }
461*40d01547SDavid du Colombier 
462*40d01547SDavid du Colombier double
ieeedtod(Ieee * ieeep)463*40d01547SDavid du Colombier ieeedtod(Ieee *ieeep)
464*40d01547SDavid du Colombier {
465*40d01547SDavid du Colombier 	Ieee e;
466*40d01547SDavid du Colombier 	double fr;
467*40d01547SDavid du Colombier 	int exp;
468*40d01547SDavid du Colombier 
469*40d01547SDavid du Colombier 	if(ieeep->h & (1L<<31)) {
470*40d01547SDavid du Colombier 		e.h = ieeep->h & ~(1L<<31);
471*40d01547SDavid du Colombier 		e.l = ieeep->l;
472*40d01547SDavid du Colombier 		return -ieeedtod(&e);
473*40d01547SDavid du Colombier 	}
474*40d01547SDavid du Colombier 	if(ieeep->l == 0 && ieeep->h == 0)
475*40d01547SDavid du Colombier 		return 0;
476*40d01547SDavid du Colombier 	fr = ieeep->l & ((1L<<16)-1L);
477*40d01547SDavid du Colombier 	fr /= 1L<<16;
478*40d01547SDavid du Colombier 	fr += (ieeep->l>>16) & ((1L<<16)-1L);
479*40d01547SDavid du Colombier 	fr /= 1L<<16;
480*40d01547SDavid du Colombier 	fr += (ieeep->h & (1L<<20)-1L) | (1L<<20);
481*40d01547SDavid du Colombier 	fr /= 1L<<21;
482*40d01547SDavid du Colombier 	exp = (ieeep->h>>20) & ((1L<<11)-1L);
483*40d01547SDavid du Colombier 	exp -= (1L<<10) - 2L;
484*40d01547SDavid du Colombier 	return ldexp(fr, exp);
485*40d01547SDavid du Colombier }
486