|
|
1.1 root 1: /*
2: * Copyright (c) 1980 Regents of the University of California.
3: * All rights reserved. The Berkeley software License Agreement
4: * specifies the terms and conditions for redistribution.
5: */
6:
7: #ifndef lint
8: char copyright[] =
9: "@(#) Copyright (c) 1980 Regents of the University of California.\n\
10: All rights reserved.\n";
11: #endif not lint
12:
13: #ifndef lint
14: static char sccsid[] = "@(#)ld.c 5.4 (Berkeley) 11/26/85";
15: #endif not lint
16:
17: /*
18: * ld - string table version for VAX
19: */
20:
21: #include <sys/param.h>
22: #include <signal.h>
23: #include <stdio.h>
24: #include <ctype.h>
25: #include <ar.h>
26: #include <a.out.h>
27: #include <ranlib.h>
28: #include <sys/stat.h>
29: #include <sys/file.h>
30:
31: /*
32: * Basic strategy:
33: *
34: * The loader takes a number of files and libraries as arguments.
35: * A first pass examines each file in turn. Normal files are
36: * unconditionally loaded, and the (external) symbols they define and require
37: * are noted in the symbol table. Libraries are searched, and the
38: * library members which define needed symbols are remembered
39: * in a special data structure so they can be selected on the second
40: * pass. Symbols defined and required by library members are also
41: * recorded.
42: *
43: * After the first pass, the loader knows the size of the basic text
44: * data, and bss segments from the sum of the sizes of the modules which
45: * were required. It has computed, for each ``common'' symbol, the
46: * maximum size of any reference to it, and these symbols are then assigned
47: * storage locations after their sizes are appropriately rounded.
48: * The loader now knows all sizes for the eventual output file, and
49: * can determine the final locations of external symbols before it
50: * begins a second pass.
51: *
52: * On the second pass each normal file and required library member
53: * is processed again. The symbol table for each such file is
54: * reread and relevant parts of it are placed in the output. The offsets
55: * in the local symbol table for externally defined symbols are recorded
56: * since relocation information refers to symbols in this way.
57: * Armed with all necessary information, the text and data segments
58: * are relocated and the result is placed in the output file, which
59: * is pasted together, ``in place'', by writing to it in several
60: * different places concurrently.
61: */
62:
63: /*
64: * Internal data structures
65: *
66: * All internal data structures are segmented and dynamically extended.
67: * The basic structures hold 1103 (NSYM) symbols, ~~200 (NROUT)
68: * referenced library members, and 100 (NSYMPR) private (local) symbols
69: * per object module. For large programs and/or modules, these structures
70: * expand to be up to 40 (NSEG) times as large as this as necessary.
71: */
72: #define NSEG 40 /* Number of segments, each data structure */
73: #define NSYM 1103 /* Number of symbols per segment */
74: #define NROUT 250 /* Number of library references per segment */
75: #define NSYMPR 100 /* Number of private symbols per segment */
76:
77: /*
78: * Structure describing each symbol table segment.
79: * Each segment has its own hash table. We record the first
80: * address in and first address beyond both the symbol and hash
81: * tables, for use in the routine symx and the lookup routine respectively.
82: * The symfree routine also understands this structure well as it used
83: * to back out symbols from modules we decide that we don't need in pass 1.
84: *
85: * Csymseg points to the current symbol table segment;
86: * csymseg->sy_first[csymseg->sy_used] is the next symbol slot to be allocated,
87: * (unless csymseg->sy_used == NSYM in which case we will allocate another
88: * symbol table segment first.)
89: */
90: struct symseg {
91: struct nlist *sy_first; /* base of this alloc'ed segment */
92: struct nlist *sy_last; /* end of this segment, for n_strx */
93: int sy_used; /* symbols used in this seg */
94: struct nlist **sy_hfirst; /* base of hash table, this seg */
95: struct nlist **sy_hlast; /* end of hash table, this seg */
96: } symseg[NSEG], *csymseg;
97:
98: /*
99: * The lookup routine uses quadratic rehash. Since a quadratic rehash
100: * only probes 1/2 of the buckets in the table, and since the hash
101: * table is segmented the same way the symbol table is, we make the
102: * hash table have twice as many buckets as there are symbol table slots
103: * in the segment. This guarantees that the quadratic rehash will never
104: * fail to find an empty bucket if the segment is not full and the
105: * symbol is not there.
106: */
107: #define HSIZE (NSYM*2)
108:
109: /*
110: * Xsym converts symbol table indices (ala x) into symbol table pointers.
111: * Symx (harder, but never used in loops) inverts pointers into the symbol
112: * table into indices using the symseg[] structure.
113: */
114: #define xsym(x) (symseg[(x)/NSYM].sy_first+((x)%NSYM))
115: /* symx() is a function, defined below */
116:
117: struct nlist cursym; /* current symbol */
118: struct nlist *lastsym; /* last symbol entered */
119: struct nlist *nextsym; /* next available symbol table entry */
120: struct nlist *addsym; /* first sym defined during incr load */
121: int nsym; /* pass2: number of local symbols in a.out */
122: /* nsym + symx(nextsym) is the symbol table size during pass2 */
123:
124: struct nlist **lookup(), **slookup();
125: struct nlist *p_etext, *p_edata, *p_end, *entrypt;
126:
127: /*
128: * Definitions of segmentation for library member table.
129: * For each library we encounter on pass 1 we record pointers to all
130: * members which we will load on pass 2. These are recorded as offsets
131: * into the archive in the library member table. Libraries are
132: * separated in the table by the special offset value -1.
133: */
134: off_t li_init[NROUT];
135: struct libseg {
136: off_t *li_first;
137: int li_used;
138: int li_used2;
139: } libseg[NSEG] = {
140: li_init, 0, 0,
141: }, *clibseg = libseg;
142:
143: /*
144: * In processing each module on pass 2 we must relocate references
145: * relative to external symbols. These references are recorded
146: * in the relocation information as relative to local symbol numbers
147: * assigned to the external symbols when the module was created.
148: * Thus before relocating the module in pass 2 we create a table
149: * which maps these internal numbers to symbol table entries.
150: * A hash table is constructed, based on the local symbol table indices,
151: * for quick lookup of these symbols.
152: */
153: #define LHSIZ 31
154: struct local {
155: int l_index; /* index to symbol in file */
156: struct nlist *l_symbol; /* ptr to symbol table */
157: struct local *l_link; /* hash link */
158: } *lochash[LHSIZ], lhinit[NSYMPR];
159: struct locseg {
160: struct local *lo_first;
161: int lo_used;
162: } locseg[NSEG] = {
163: lhinit, 0
164: }, *clocseg;
165:
166: /*
167: * Libraries are typically built with a table of contents,
168: * which is the first member of a library with special file
169: * name __.SYMDEF and contains a list of symbol names
170: * and with each symbol the offset of the library member which defines
171: * it. The loader uses this table to quickly tell which library members
172: * are (potentially) useful. The alternative, examining the symbol
173: * table of each library member, is painfully slow for large archives.
174: *
175: * See <ranlib.h> for the definition of the ranlib structure and an
176: * explanation of the __.SYMDEF file format.
177: */
178: int tnum; /* number of symbols in table of contents */
179: int ssiz; /* size of string table for table of contents */
180: struct ranlib *tab; /* the table of contents (dynamically allocated) */
181: char *tabstr; /* string table for table of contents */
182:
183: /*
184: * We open each input file or library only once, but in pass2 we
185: * (historically) read from such a file at 2 different places at the
186: * same time. These structures are remnants from those days,
187: * and now serve only to catch ``Premature EOF''.
188: * In order to make I/O more efficient, we provide routines which
189: * use the optimal block size returned by stat().
190: */
191: #define BLKSIZE 1024
192: typedef struct {
193: short *fakeptr;
194: int bno;
195: int nibuf;
196: int nuser;
197: char *buff;
198: int bufsize;
199: } PAGE;
200:
201: PAGE page[2];
202: int p_blksize;
203: int p_blkshift;
204: int p_blkmask;
205:
206: struct {
207: short *fakeptr;
208: int bno;
209: int nibuf;
210: int nuser;
211: } fpage;
212:
213: typedef struct {
214: char *ptr;
215: int bno;
216: int nibuf;
217: long size;
218: long pos;
219: PAGE *pno;
220: } STREAM;
221:
222: STREAM text;
223: STREAM reloc;
224:
225: /*
226: * Header from the a.out and the archive it is from (if any).
227: */
228: struct exec filhdr;
229: struct ar_hdr archdr;
230: #define OARMAG 0177545
231:
232: /*
233: * Options.
234: */
235: int trace;
236: int xflag; /* discard local symbols */
237: int Xflag; /* discard locals starting with 'L' */
238: int Sflag; /* discard all except locals and globals*/
239: int rflag; /* preserve relocation bits, don't define common */
240: int arflag; /* original copy of rflag */
241: int sflag; /* discard all symbols */
242: int Mflag; /* print rudimentary load map */
243: int nflag; /* pure procedure */
244: int dflag; /* define common even with rflag */
245: int zflag; /* demand paged */
246: long hsize; /* size of hole at beginning of data to be squashed */
247: int Aflag; /* doing incremental load */
248: int Nflag; /* want impure a.out */
249: int funding; /* reading fundamental file for incremental load */
250: int yflag; /* number of symbols to be traced */
251: char **ytab; /* the symbols */
252:
253: /*
254: * These are the cumulative sizes, set in pass 1, which
255: * appear in the a.out header when the loader is finished.
256: */
257: off_t tsize, dsize, bsize, trsize, drsize, ssize;
258:
259: /*
260: * Symbol relocation: c?rel is a scale factor which is
261: * added to an old relocation to convert it to new units;
262: * i.e. it is the difference between segment origins.
263: * (Thus if we are loading from a data segment which began at location
264: * 4 in a .o file into an a.out where it will be loaded starting at
265: * 1024, cdrel will be 1020.)
266: */
267: long ctrel, cdrel, cbrel;
268:
269: /*
270: * Textbase is the start address of all text, 0 unless given by -T.
271: * Database is the base of all data, computed before and used during pass2.
272: */
273: long textbase, database;
274:
275: /*
276: * The base addresses for the loaded text, data and bss from the
277: * current module during pass2 are given by torigin, dorigin and borigin.
278: */
279: long torigin, dorigin, borigin;
280:
281: /*
282: * Errlev is nonzero when errors have occured.
283: * Delarg is an implicit argument to the routine delexit
284: * which is called on error. We do ``delarg = errlev'' before normal
285: * exits, and only if delarg is 0 (i.e. errlev was 0) do we make the
286: * result file executable.
287: */
288: int errlev;
289: int delarg = 4;
290:
291: /*
292: * The biobuf structure and associated routines are used to write
293: * into one file at several places concurrently. Calling bopen
294: * with a biobuf structure sets it up to write ``biofd'' starting
295: * at the specified offset. You can then use ``bwrite'' and/or ``bputc''
296: * to stuff characters in the stream, much like ``fwrite'' and ``fputc''.
297: * Calling bflush drains all the buffers and MUST be done before exit.
298: */
299: struct biobuf {
300: short b_nleft; /* Number free spaces left in b_buf */
301: /* Initialize to be less than b_bufsize initially, to boundary align in file */
302: char *b_ptr; /* Next place to stuff characters */
303: char *b_buf; /* Pointer to the buffer */
304: int b_bufsize; /* Size of the buffer */
305: off_t b_off; /* Current file offset */
306: struct biobuf *b_link; /* Link in chain for bflush() */
307: } *biobufs;
308: #define bputc(c,b) ((b)->b_nleft ? (--(b)->b_nleft, *(b)->b_ptr++ = (c)) \
309: : bflushc(b, c))
310: int biofd;
311: off_t boffset;
312: struct biobuf *tout, *dout, *trout, *drout, *sout, *strout;
313:
314: /*
315: * Offset is the current offset in the string file.
316: * Its initial value reflects the fact that we will
317: * eventually stuff the size of the string table at the
318: * beginning of the string table (i.e. offset itself!).
319: */
320: off_t offset = sizeof (off_t);
321:
322: int ofilfnd; /* -o given; otherwise move l.out to a.out */
323: char *ofilename = "l.out";
324: int ofilemode; /* respect umask even for unsucessful ld's */
325: int infil; /* current input file descriptor */
326: char *filname; /* and its name */
327:
328: #define NDIRS 25
329: #define NDEFDIRS 3 /* number of default directories in dirs[] */
330: char *dirs[NDIRS]; /* directories for library search */
331: int ndir; /* number of directories */
332:
333: /*
334: * Base of the string table of the current module (pass1 and pass2).
335: */
336: char *curstr;
337:
338: /*
339: * System software page size, as returned by getpagesize.
340: */
341: int pagesize;
342:
343: char get();
344: int delexit();
345: char *savestr();
346: char *malloc();
347:
348: main(argc, argv)
349: char **argv;
350: {
351: register int c, i;
352: int num;
353: register char *ap, **p;
354: char save;
355:
356: if (signal(SIGINT, SIG_IGN) != SIG_IGN) {
357: signal(SIGINT, delexit);
358: signal(SIGTERM, delexit);
359: }
360: if (argc == 1)
361: exit(4);
362: pagesize = getpagesize();
363:
364: /*
365: * Pull out search directories.
366: */
367: for (c = 1; c < argc; c++) {
368: ap = argv[c];
369: if (ap[0] == '-' && ap[1] == 'L') {
370: if (ap[2] == 0)
371: error(1, "-L: pathname missing");
372: if (ndir >= NDIRS - NDEFDIRS)
373: error(1, "-L: too many directories");
374: dirs[ndir++] = &ap[2];
375: }
376: }
377: /* add default search directories */
378: dirs[ndir++] = "/lib";
379: dirs[ndir++] = "/usr/lib";
380: dirs[ndir++] = "/usr/local/lib";
381:
382: p = argv+1;
383: /*
384: * Scan files once to find where symbols are defined.
385: */
386: for (c=1; c<argc; c++) {
387: if (trace)
388: printf("%s:\n", *p);
389: filname = 0;
390: ap = *p++;
391: if (*ap != '-') {
392: load1arg(ap);
393: continue;
394: }
395: for (i=1; ap[i]; i++) switch (ap[i]) {
396:
397: case 'o':
398: if (++c >= argc)
399: error(1, "-o where?");
400: ofilename = *p++;
401: ofilfnd++;
402: continue;
403: case 'u':
404: case 'e':
405: if (++c >= argc)
406: error(1, "-u or -c: arg missing");
407: enter(slookup(*p++));
408: if (ap[i]=='e')
409: entrypt = lastsym;
410: continue;
411: case 'H':
412: if (++c >= argc)
413: error(1, "-H: arg missing");
414: if (tsize!=0)
415: error(1, "-H: too late, some text already loaded");
416: hsize = atoi(*p++);
417: continue;
418: case 'A':
419: if (++c >= argc)
420: error(1, "-A: arg missing");
421: if (Aflag)
422: error(1, "-A: only one base file allowed");
423: Aflag = 1;
424: nflag = 0;
425: funding = 1;
426: load1arg(*p++);
427: trsize = drsize = tsize = dsize = bsize = 0;
428: ctrel = cdrel = cbrel = 0;
429: funding = 0;
430: addsym = nextsym;
431: continue;
432: case 'D':
433: if (++c >= argc)
434: error(1, "-D: arg missing");
435: num = htoi(*p++);
436: if (dsize > num)
437: error(1, "-D: too small");
438: dsize = num;
439: continue;
440: case 'T':
441: if (++c >= argc)
442: error(1, "-T: arg missing");
443: if (tsize!=0)
444: error(1, "-T: too late, some text already loaded");
445: textbase = htoi(*p++);
446: continue;
447: case 'l':
448: save = ap[--i];
449: ap[i]='-';
450: load1arg(&ap[i]);
451: ap[i]=save;
452: goto next;
453: case 'M':
454: Mflag++;
455: continue;
456: case 'x':
457: xflag++;
458: continue;
459: case 'X':
460: Xflag++;
461: continue;
462: case 'S':
463: Sflag++;
464: continue;
465: case 'r':
466: rflag++;
467: arflag++;
468: continue;
469: case 's':
470: sflag++;
471: xflag++;
472: continue;
473: case 'n':
474: nflag++;
475: Nflag = zflag = 0;
476: continue;
477: case 'N':
478: Nflag++;
479: nflag = zflag = 0;
480: continue;
481: case 'd':
482: dflag++;
483: continue;
484: case 'i':
485: printf("ld: -i ignored\n");
486: continue;
487: case 't':
488: trace++;
489: continue;
490: case 'y':
491: if (ap[i+1] == 0)
492: error(1, "-y: symbol name missing");
493: if (yflag == 0) {
494: ytab = (char **)calloc(argc, sizeof (char **));
495: if (ytab == 0)
496: error(1, "ran out of memory (-y)");
497: }
498: ytab[yflag++] = &ap[i+1];
499: goto next;
500: case 'z':
501: zflag++;
502: Nflag = nflag = 0;
503: continue;
504: case 'L':
505: goto next;
506: default:
507: filname = savestr("-x"); /* kludge */
508: filname[1] = ap[i]; /* kludge */
509: archdr.ar_name[0] = 0; /* kludge */
510: error(1, "bad flag");
511: }
512: next:
513: ;
514: }
515: if (rflag == 0 && Nflag == 0 && nflag == 0)
516: zflag++;
517: endload(argc, argv);
518: exit(0);
519: }
520:
521: /*
522: * Convert a ascii string which is a hex number.
523: * Used by -T and -D options.
524: */
525: htoi(p)
526: register char *p;
527: {
528: register int c, n;
529:
530: n = 0;
531: while (c = *p++) {
532: n <<= 4;
533: if (isdigit(c))
534: n += c - '0';
535: else if (c >= 'a' && c <= 'f')
536: n += 10 + (c - 'a');
537: else if (c >= 'A' && c <= 'F')
538: n += 10 + (c - 'A');
539: else
540: error(1, "badly formed hex number");
541: }
542: return (n);
543: }
544:
545: delexit()
546: {
547: struct stat stbuf;
548: long size;
549: char c = 0;
550:
551: bflush();
552: unlink("l.out");
553: /*
554: * We have to insure that the last block of the data segment
555: * is allocated a full pagesize block. If the underlying
556: * file system allocates frags that are smaller than pagesize,
557: * a full zero filled pagesize block needs to be allocated so
558: * that when it is demand paged, the paged in block will be
559: * appropriately filled with zeros.
560: */
561: fstat(biofd, &stbuf);
562: size = round(stbuf.st_size, pagesize);
563: if (!rflag && size > stbuf.st_size) {
564: lseek(biofd, size - 1, 0);
565: if (write(biofd, &c, 1) != 1)
566: delarg |= 4;
567: }
568: if (delarg==0 && Aflag==0)
569: (void) chmod(ofilename, ofilemode);
570: exit (delarg);
571: }
572:
573: endload(argc, argv)
574: int argc;
575: char **argv;
576: {
577: register int c, i;
578: long dnum;
579: register char *ap, **p;
580:
581: clibseg = libseg;
582: filname = 0;
583: middle();
584: setupout();
585: p = argv+1;
586: for (c=1; c<argc; c++) {
587: ap = *p++;
588: if (trace)
589: printf("%s:\n", ap);
590: if (*ap != '-') {
591: load2arg(ap);
592: continue;
593: }
594: for (i=1; ap[i]; i++) switch (ap[i]) {
595:
596: case 'D':
597: dnum = htoi(*p);
598: if (dorigin < dnum)
599: while (dorigin < dnum)
600: bputc(0, dout), dorigin++;
601: /* fall into ... */
602: case 'T':
603: case 'u':
604: case 'e':
605: case 'o':
606: case 'H':
607: ++c;
608: ++p;
609: /* fall into ... */
610: default:
611: continue;
612: case 'A':
613: funding = 1;
614: load2arg(*p++);
615: funding = 0;
616: c++;
617: continue;
618: case 'y':
619: case 'L':
620: goto next;
621: case 'l':
622: ap[--i]='-';
623: load2arg(&ap[i]);
624: goto next;
625: }
626: next:
627: ;
628: }
629: finishout();
630: }
631:
632: /*
633: * Scan file to find defined symbols.
634: */
635: load1arg(cp)
636: register char *cp;
637: {
638: register struct ranlib *tp;
639: off_t nloc;
640: int kind;
641:
642: kind = getfile(cp);
643: if (Mflag)
644: printf("%s\n", filname);
645: switch (kind) {
646:
647: /*
648: * Plain file.
649: */
650: case 0:
651: load1(0, 0L);
652: break;
653:
654: /*
655: * Archive without table of contents.
656: * (Slowly) process each member.
657: */
658: case 1:
659: error(-1,
660: "warning: archive has no table of contents; add one using ranlib(1)");
661: nloc = SARMAG;
662: while (step(nloc))
663: nloc += sizeof(archdr) +
664: round(atol(archdr.ar_size), sizeof (short));
665: break;
666:
667: /*
668: * Archive with table of contents.
669: * Read the table of contents and its associated string table.
670: * Pass through the library resolving symbols until nothing changes
671: * for an entire pass (i.e. you can get away with backward references
672: * when there is a table of contents!)
673: */
674: case 2:
675: nloc = SARMAG + sizeof (archdr);
676: dseek(&text, nloc, sizeof (tnum));
677: mget((char *)&tnum, sizeof (tnum), &text);
678: nloc += sizeof (tnum);
679: tab = (struct ranlib *)malloc(tnum);
680: if (tab == 0)
681: error(1, "ran out of memory (toc)");
682: dseek(&text, nloc, tnum);
683: mget((char *)tab, tnum, &text);
684: nloc += tnum;
685: tnum /= sizeof (struct ranlib);
686: dseek(&text, nloc, sizeof (ssiz));
687: mget((char *)&ssiz, sizeof (ssiz), &text);
688: nloc += sizeof (ssiz);
689: tabstr = (char *)malloc(ssiz);
690: if (tabstr == 0)
691: error(1, "ran out of memory (tocstr)");
692: dseek(&text, nloc, ssiz);
693: mget((char *)tabstr, ssiz, &text);
694: for (tp = &tab[tnum]; --tp >= tab;) {
695: if (tp->ran_un.ran_strx < 0 ||
696: tp->ran_un.ran_strx >= ssiz)
697: error(1, "mangled archive table of contents");
698: tp->ran_un.ran_name = tabstr + tp->ran_un.ran_strx;
699: }
700: while (ldrand())
701: continue;
702: free((char *)tab);
703: free(tabstr);
704: nextlibp(-1);
705: break;
706:
707: /*
708: * Table of contents is out of date, so search
709: * as a normal library (but skip the __.SYMDEF file).
710: */
711: case 3:
712: error(-1,
713: "warning: table of contents for archive is out of date; rerun ranlib(1)");
714: nloc = SARMAG;
715: do
716: nloc += sizeof(archdr) +
717: round(atol(archdr.ar_size), sizeof(short));
718: while (step(nloc));
719: break;
720: }
721: close(infil);
722: }
723:
724: /*
725: * Advance to the next archive member, which
726: * is at offset nloc in the archive. If the member
727: * is useful, record its location in the liblist structure
728: * for use in pass2. Mark the end of the archive in libilst with a -1.
729: */
730: step(nloc)
731: off_t nloc;
732: {
733:
734: dseek(&text, nloc, (long) sizeof archdr);
735: if (text.size <= 0) {
736: nextlibp(-1);
737: return (0);
738: }
739: getarhdr();
740: if (load1(1, nloc + (sizeof archdr)))
741: nextlibp(nloc);
742: return (1);
743: }
744:
745: /*
746: * Record the location of a useful archive member.
747: * Recording -1 marks the end of files from an archive.
748: * The liblist data structure is dynamically extended here.
749: */
750: nextlibp(val)
751: off_t val;
752: {
753:
754: if (clibseg->li_used == NROUT) {
755: if (++clibseg == &libseg[NSEG])
756: error(1, "too many files loaded from libraries");
757: clibseg->li_first = (off_t *)malloc(NROUT * sizeof (off_t));
758: if (clibseg->li_first == 0)
759: error(1, "ran out of memory (nextlibp)");
760: }
761: clibseg->li_first[clibseg->li_used++] = val;
762: if (val != -1 && Mflag)
763: printf("\t%s\n", archdr.ar_name);
764: }
765:
766: /*
767: * One pass over an archive with a table of contents.
768: * Remember the number of symbols currently defined,
769: * then call step on members which look promising (i.e.
770: * that define a symbol which is currently externally undefined).
771: * Indicate to our caller whether this process netted any more symbols.
772: */
773: ldrand()
774: {
775: register struct nlist *sp, **hp;
776: register struct ranlib *tp, *tplast;
777: off_t loc;
778: int nsymt = symx(nextsym);
779:
780: tplast = &tab[tnum-1];
781: for (tp = tab; tp <= tplast; tp++) {
782: if ((hp = slookup(tp->ran_un.ran_name)) == 0 || *hp == 0)
783: continue;
784: sp = *hp;
785: if (sp->n_type != N_EXT+N_UNDF)
786: continue;
787: step(tp->ran_off);
788: loc = tp->ran_off;
789: while (tp < tplast && (tp+1)->ran_off == loc)
790: tp++;
791: }
792: return (symx(nextsym) != nsymt);
793: }
794:
795: /*
796: * Examine a single file or archive member on pass 1.
797: */
798: load1(libflg, loc)
799: off_t loc;
800: {
801: register struct nlist *sp;
802: struct nlist *savnext;
803: int ndef, nlocal, type, size, nsymt;
804: register int i;
805: off_t maxoff;
806: struct stat stb;
807:
808: readhdr(loc);
809: if (filhdr.a_syms == 0) {
810: if (filhdr.a_text+filhdr.a_data == 0)
811: return (0);
812: error(1, "no namelist");
813: }
814: if (libflg)
815: maxoff = atol(archdr.ar_size);
816: else {
817: fstat(infil, &stb);
818: maxoff = stb.st_size;
819: }
820: if (N_STROFF(filhdr) + sizeof (off_t) >= maxoff)
821: error(1, "too small (old format .o?)");
822: ctrel = tsize; cdrel += dsize; cbrel += bsize;
823: ndef = 0;
824: nlocal = sizeof(cursym);
825: savnext = nextsym;
826: loc += N_SYMOFF(filhdr);
827: dseek(&text, loc, filhdr.a_syms);
828: dseek(&reloc, loc + filhdr.a_syms, sizeof(off_t));
829: mget(&size, sizeof (size), &reloc);
830: dseek(&reloc, loc + filhdr.a_syms+sizeof (off_t), size-sizeof (off_t));
831: curstr = (char *)malloc(size);
832: if (curstr == NULL)
833: error(1, "no space for string table");
834: mget(curstr+sizeof(off_t), size-sizeof(off_t), &reloc);
835: while (text.size > 0) {
836: mget((char *)&cursym, sizeof(struct nlist), &text);
837: if (cursym.n_un.n_strx) {
838: if (cursym.n_un.n_strx<sizeof(size) ||
839: cursym.n_un.n_strx>=size)
840: error(1, "bad string table index (pass 1)");
841: cursym.n_un.n_name = curstr + cursym.n_un.n_strx;
842: }
843: type = cursym.n_type;
844: if ((type&N_EXT)==0) {
845: if (Xflag==0 || cursym.n_un.n_name[0]!='L' ||
846: type & N_STAB)
847: nlocal += sizeof cursym;
848: continue;
849: }
850: symreloc();
851: if (enter(lookup()))
852: continue;
853: if ((sp = lastsym)->n_type != N_EXT+N_UNDF)
854: continue;
855: if (cursym.n_type == N_EXT+N_UNDF) {
856: if (cursym.n_value > sp->n_value)
857: sp->n_value = cursym.n_value;
858: continue;
859: }
860: if (sp->n_value != 0 && cursym.n_type == N_EXT+N_TEXT)
861: continue;
862: ndef++;
863: sp->n_type = cursym.n_type;
864: sp->n_value = cursym.n_value;
865: }
866: if (libflg==0 || ndef) {
867: tsize += filhdr.a_text;
868: dsize += round(filhdr.a_data, sizeof (long));
869: bsize += round(filhdr.a_bss, sizeof (long));
870: ssize += nlocal;
871: trsize += filhdr.a_trsize;
872: drsize += filhdr.a_drsize;
873: if (funding)
874: textbase = (*slookup("_end"))->n_value;
875: nsymt = symx(nextsym);
876: for (i = symx(savnext); i < nsymt; i++) {
877: sp = xsym(i);
878: sp->n_un.n_name = savestr(sp->n_un.n_name);
879: }
880: free(curstr);
881: return (1);
882: }
883: /*
884: * No symbols defined by this library member.
885: * Rip out the hash table entries and reset the symbol table.
886: */
887: symfree(savnext);
888: free(curstr);
889: return(0);
890: }
891:
892: middle()
893: {
894: register struct nlist *sp;
895: long csize, t, corigin, ocsize;
896: int nund, rnd;
897: char s;
898: register int i;
899: int nsymt;
900:
901: torigin = 0;
902: dorigin = 0;
903: borigin = 0;
904:
905: p_etext = *slookup("_etext");
906: p_edata = *slookup("_edata");
907: p_end = *slookup("_end");
908: /*
909: * If there are any undefined symbols, save the relocation bits.
910: */
911: nsymt = symx(nextsym);
912: if (rflag==0) {
913: for (i = 0; i < nsymt; i++) {
914: sp = xsym(i);
915: if (sp->n_type==N_EXT+N_UNDF && sp->n_value==0 &&
916: sp!=p_end && sp!=p_edata && sp!=p_etext) {
917: rflag++;
918: dflag = 0;
919: break;
920: }
921: }
922: }
923: if (rflag)
924: sflag = zflag = 0;
925: /*
926: * Assign common locations.
927: */
928: csize = 0;
929: if (!Aflag)
930: addsym = symseg[0].sy_first;
931: database = round(tsize+textbase,
932: (nflag||zflag? pagesize : sizeof (long)));
933: database += hsize;
934: if (dflag || rflag==0) {
935: ldrsym(p_etext, tsize, N_EXT+N_TEXT);
936: ldrsym(p_edata, dsize, N_EXT+N_DATA);
937: ldrsym(p_end, bsize, N_EXT+N_BSS);
938: for (i = symx(addsym); i < nsymt; i++) {
939: sp = xsym(i);
940: if ((s=sp->n_type)==N_EXT+N_UNDF &&
941: (t = sp->n_value)!=0) {
942: if (t >= sizeof (double))
943: rnd = sizeof (double);
944: else if (t >= sizeof (long))
945: rnd = sizeof (long);
946: else
947: rnd = sizeof (short);
948: csize = round(csize, rnd);
949: sp->n_value = csize;
950: sp->n_type = N_EXT+N_COMM;
951: ocsize = csize;
952: csize += t;
953: }
954: if (s&N_EXT && (s&N_TYPE)==N_UNDF && s&N_STAB) {
955: sp->n_value = ocsize;
956: sp->n_type = (s&N_STAB) | (N_EXT+N_COMM);
957: }
958: }
959: }
960: /*
961: * Now set symbols to their final value
962: */
963: csize = round(csize, sizeof (long));
964: torigin = textbase;
965: dorigin = database;
966: corigin = dorigin + dsize;
967: borigin = corigin + csize;
968: nund = 0;
969: nsymt = symx(nextsym);
970: for (i = symx(addsym); i<nsymt; i++) {
971: sp = xsym(i);
972: switch (sp->n_type & (N_TYPE+N_EXT)) {
973:
974: case N_EXT+N_UNDF:
975: if (arflag == 0)
976: errlev |= 01;
977: if ((arflag==0 || dflag) && sp->n_value==0) {
978: if (sp==p_end || sp==p_etext || sp==p_edata)
979: continue;
980: if (nund==0)
981: printf("Undefined:\n");
982: nund++;
983: printf("%s\n", sp->n_un.n_name);
984: }
985: continue;
986: case N_EXT+N_ABS:
987: default:
988: continue;
989: case N_EXT+N_TEXT:
990: sp->n_value += torigin;
991: continue;
992: case N_EXT+N_DATA:
993: sp->n_value += dorigin;
994: continue;
995: case N_EXT+N_BSS:
996: sp->n_value += borigin;
997: continue;
998: case N_EXT+N_COMM:
999: sp->n_type = (sp->n_type & N_STAB) | (N_EXT+N_BSS);
1000: sp->n_value += corigin;
1001: continue;
1002: }
1003: }
1004: if (sflag || xflag)
1005: ssize = 0;
1006: bsize += csize;
1007: nsym = ssize / (sizeof cursym);
1008: if (Aflag) {
1009: fixspec(p_etext,torigin);
1010: fixspec(p_edata,dorigin);
1011: fixspec(p_end,borigin);
1012: }
1013: }
1014:
1015: fixspec(sym,offset)
1016: struct nlist *sym;
1017: long offset;
1018: {
1019:
1020: if(symx(sym) < symx(addsym) && sym!=0)
1021: sym->n_value += offset;
1022: }
1023:
1024: ldrsym(sp, val, type)
1025: register struct nlist *sp;
1026: long val;
1027: {
1028:
1029: if (sp == 0)
1030: return;
1031: if ((sp->n_type != N_EXT+N_UNDF || sp->n_value) && !Aflag) {
1032: printf("%s: ", sp->n_un.n_name);
1033: error(0, "user attempt to redfine loader-defined symbol");
1034: return;
1035: }
1036: sp->n_type = type;
1037: sp->n_value = val;
1038: }
1039:
1040: off_t wroff;
1041: struct biobuf toutb;
1042:
1043: setupout()
1044: {
1045: int bss;
1046: struct stat stbuf;
1047: extern char *sys_errlist[];
1048: extern int errno;
1049:
1050: ofilemode = 0777 & ~umask(0);
1051: biofd = creat(ofilename, 0666 & ofilemode);
1052: if (biofd < 0) {
1053: filname = ofilename; /* kludge */
1054: archdr.ar_name[0] = 0; /* kludge */
1055: error(1, sys_errlist[errno]); /* kludge */
1056: }
1057: fstat(biofd, &stbuf); /* suppose file exists, wrong*/
1058: if (stbuf.st_mode & 0111) { /* mode, ld fails? */
1059: chmod(ofilename, stbuf.st_mode & 0666);
1060: ofilemode = stbuf.st_mode;
1061: }
1062: filhdr.a_magic = nflag ? NMAGIC : (zflag ? ZMAGIC : OMAGIC);
1063: filhdr.a_text = nflag ? tsize :
1064: round(tsize, zflag ? pagesize : sizeof (long));
1065: filhdr.a_data = zflag ? round(dsize, pagesize) : dsize;
1066: bss = bsize - (filhdr.a_data - dsize);
1067: if (bss < 0)
1068: bss = 0;
1069: filhdr.a_bss = bss;
1070: filhdr.a_trsize = trsize;
1071: filhdr.a_drsize = drsize;
1072: filhdr.a_syms = sflag? 0: (ssize + (sizeof cursym)*symx(nextsym));
1073: if (entrypt) {
1074: if (entrypt->n_type!=N_EXT+N_TEXT)
1075: error(0, "entry point not in text");
1076: else
1077: filhdr.a_entry = entrypt->n_value;
1078: } else
1079: filhdr.a_entry = 0;
1080: filhdr.a_trsize = (rflag ? trsize:0);
1081: filhdr.a_drsize = (rflag ? drsize:0);
1082: tout = &toutb;
1083: bopen(tout, 0, stbuf.st_blksize);
1084: bwrite((char *)&filhdr, sizeof (filhdr), tout);
1085: if (zflag)
1086: bseek(tout, pagesize);
1087: wroff = N_TXTOFF(filhdr) + filhdr.a_text;
1088: outb(&dout, filhdr.a_data, stbuf.st_blksize);
1089: if (rflag) {
1090: outb(&trout, filhdr.a_trsize, stbuf.st_blksize);
1091: outb(&drout, filhdr.a_drsize, stbuf.st_blksize);
1092: }
1093: if (sflag==0 || xflag==0) {
1094: outb(&sout, filhdr.a_syms, stbuf.st_blksize);
1095: wroff += sizeof (offset);
1096: outb(&strout, 0, stbuf.st_blksize);
1097: }
1098: }
1099:
1100: outb(bp, inc, bufsize)
1101: register struct biobuf **bp;
1102: {
1103:
1104: *bp = (struct biobuf *)malloc(sizeof (struct biobuf));
1105: if (*bp == 0)
1106: error(1, "ran out of memory (outb)");
1107: bopen(*bp, wroff, bufsize);
1108: wroff += inc;
1109: }
1110:
1111: load2arg(acp)
1112: char *acp;
1113: {
1114: register char *cp;
1115: off_t loc;
1116:
1117: cp = acp;
1118: if (getfile(cp) == 0) {
1119: while (*cp)
1120: cp++;
1121: while (cp >= acp && *--cp != '/');
1122: mkfsym(++cp);
1123: load2(0L);
1124: } else { /* scan archive members referenced */
1125: for (;;) {
1126: if (clibseg->li_used2 == clibseg->li_used) {
1127: if (clibseg->li_used < NROUT)
1128: error(1, "libseg botch");
1129: clibseg++;
1130: }
1131: loc = clibseg->li_first[clibseg->li_used2++];
1132: if (loc == -1)
1133: break;
1134: dseek(&text, loc, (long)sizeof(archdr));
1135: getarhdr();
1136: mkfsym(archdr.ar_name);
1137: load2(loc + (long)sizeof(archdr));
1138: }
1139: }
1140: close(infil);
1141: }
1142:
1143: load2(loc)
1144: long loc;
1145: {
1146: int size;
1147: register struct nlist *sp;
1148: register struct local *lp;
1149: register int symno, i;
1150: int type;
1151:
1152: readhdr(loc);
1153: if (!funding) {
1154: ctrel = torigin;
1155: cdrel += dorigin;
1156: cbrel += borigin;
1157: }
1158: /*
1159: * Reread the symbol table, recording the numbering
1160: * of symbols for fixing external references.
1161: */
1162: for (i = 0; i < LHSIZ; i++)
1163: lochash[i] = 0;
1164: clocseg = locseg;
1165: clocseg->lo_used = 0;
1166: symno = -1;
1167: loc += N_TXTOFF(filhdr);
1168: dseek(&text, loc+filhdr.a_text+filhdr.a_data+
1169: filhdr.a_trsize+filhdr.a_drsize+filhdr.a_syms, sizeof(off_t));
1170: mget(&size, sizeof(size), &text);
1171: dseek(&text, loc+filhdr.a_text+filhdr.a_data+
1172: filhdr.a_trsize+filhdr.a_drsize+filhdr.a_syms+sizeof(off_t),
1173: size - sizeof(off_t));
1174: curstr = (char *)malloc(size);
1175: if (curstr == NULL)
1176: error(1, "out of space reading string table (pass 2)");
1177: mget(curstr+sizeof(off_t), size-sizeof(off_t), &text);
1178: dseek(&text, loc+filhdr.a_text+filhdr.a_data+
1179: filhdr.a_trsize+filhdr.a_drsize, filhdr.a_syms);
1180: while (text.size > 0) {
1181: symno++;
1182: mget((char *)&cursym, sizeof(struct nlist), &text);
1183: if (cursym.n_un.n_strx) {
1184: if (cursym.n_un.n_strx<sizeof(size) ||
1185: cursym.n_un.n_strx>=size)
1186: error(1, "bad string table index (pass 2)");
1187: cursym.n_un.n_name = curstr + cursym.n_un.n_strx;
1188: }
1189: /* inline expansion of symreloc() */
1190: switch (cursym.n_type & 017) {
1191:
1192: case N_TEXT:
1193: case N_EXT+N_TEXT:
1194: cursym.n_value += ctrel;
1195: break;
1196: case N_DATA:
1197: case N_EXT+N_DATA:
1198: cursym.n_value += cdrel;
1199: break;
1200: case N_BSS:
1201: case N_EXT+N_BSS:
1202: cursym.n_value += cbrel;
1203: break;
1204: case N_EXT+N_UNDF:
1205: break;
1206: default:
1207: if (cursym.n_type&N_EXT)
1208: cursym.n_type = N_EXT+N_ABS;
1209: }
1210: /* end inline expansion of symreloc() */
1211: type = cursym.n_type;
1212: if (yflag && cursym.n_un.n_name)
1213: for (i = 0; i < yflag; i++)
1214: /* fast check for 2d character! */
1215: if (ytab[i][1] == cursym.n_un.n_name[1] &&
1216: !strcmp(ytab[i], cursym.n_un.n_name)) {
1217: tracesym();
1218: break;
1219: }
1220: if ((type&N_EXT) == 0) {
1221: if (!sflag&&!xflag&&
1222: (!Xflag||cursym.n_un.n_name[0]!='L'||type&N_STAB))
1223: symwrite(&cursym, sout);
1224: continue;
1225: }
1226: if (funding)
1227: continue;
1228: if ((sp = *lookup()) == 0)
1229: error(1, "internal error: symbol not found");
1230: if (cursym.n_type == N_EXT+N_UNDF) {
1231: if (clocseg->lo_used == NSYMPR) {
1232: if (++clocseg == &locseg[NSEG])
1233: error(1, "local symbol overflow");
1234: clocseg->lo_used = 0;
1235: }
1236: if (clocseg->lo_first == 0) {
1237: clocseg->lo_first = (struct local *)
1238: malloc(NSYMPR * sizeof (struct local));
1239: if (clocseg->lo_first == 0)
1240: error(1, "out of memory (clocseg)");
1241: }
1242: lp = &clocseg->lo_first[clocseg->lo_used++];
1243: lp->l_index = symno;
1244: lp->l_symbol = sp;
1245: lp->l_link = lochash[symno % LHSIZ];
1246: lochash[symno % LHSIZ] = lp;
1247: continue;
1248: }
1249: if (cursym.n_type & N_STAB)
1250: continue;
1251: if (cursym.n_type!=sp->n_type || cursym.n_value!=sp->n_value) {
1252: printf("%s: ", cursym.n_un.n_name);
1253: error(0, "multiply defined");
1254: }
1255: }
1256: if (funding)
1257: return;
1258: dseek(&text, loc, filhdr.a_text);
1259: dseek(&reloc, loc+filhdr.a_text+filhdr.a_data, filhdr.a_trsize);
1260: load2td(ctrel, torigin - textbase, tout, trout);
1261: dseek(&text, loc+filhdr.a_text, filhdr.a_data);
1262: dseek(&reloc, loc+filhdr.a_text+filhdr.a_data+filhdr.a_trsize,
1263: filhdr.a_drsize);
1264: load2td(cdrel, dorigin - database, dout, drout);
1265: while (filhdr.a_data & (sizeof(long)-1)) {
1266: bputc(0, dout);
1267: filhdr.a_data++;
1268: }
1269: torigin += filhdr.a_text;
1270: dorigin += round(filhdr.a_data, sizeof (long));
1271: borigin += round(filhdr.a_bss, sizeof (long));
1272: free(curstr);
1273: }
1274:
1275: struct tynames {
1276: int ty_value;
1277: char *ty_name;
1278: } tynames[] = {
1279: N_UNDF, "undefined",
1280: N_ABS, "absolute",
1281: N_TEXT, "text",
1282: N_DATA, "data",
1283: N_BSS, "bss",
1284: N_COMM, "common",
1285: 0, 0,
1286: };
1287:
1288: tracesym()
1289: {
1290: register struct tynames *tp;
1291:
1292: if (cursym.n_type & N_STAB)
1293: return;
1294: printf("%s", filname);
1295: if (archdr.ar_name[0])
1296: printf("(%s)", archdr.ar_name);
1297: printf(": ");
1298: if ((cursym.n_type&N_TYPE) == N_UNDF && cursym.n_value) {
1299: printf("definition of common %s size %d\n",
1300: cursym.n_un.n_name, cursym.n_value);
1301: return;
1302: }
1303: for (tp = tynames; tp->ty_name; tp++)
1304: if (tp->ty_value == (cursym.n_type&N_TYPE))
1305: break;
1306: printf((cursym.n_type&N_TYPE) ? "definition of" : "reference to");
1307: if (cursym.n_type&N_EXT)
1308: printf(" external");
1309: if (tp->ty_name)
1310: printf(" %s", tp->ty_name);
1311: printf(" %s\n", cursym.n_un.n_name);
1312: }
1313:
1314: /*
1315: * This routine relocates the single text or data segment argument.
1316: * Offsets from external symbols are resolved by adding the value
1317: * of the external symbols. Non-external reference are updated to account
1318: * for the relative motion of the segments (ctrel, cdrel, ...). If
1319: * a relocation was pc-relative, then we update it to reflect the
1320: * change in the positioning of the segments by adding the displacement
1321: * of the referenced segment and subtracting the displacement of the
1322: * current segment (creloc).
1323: *
1324: * If we are saving the relocation information, then we increase
1325: * each relocation datum address by our base position in the new segment.
1326: */
1327: load2td(creloc, position, b1, b2)
1328: long creloc, offset;
1329: struct biobuf *b1, *b2;
1330: {
1331: register struct nlist *sp;
1332: register struct local *lp;
1333: long tw;
1334: register struct relocation_info *rp, *rpend;
1335: struct relocation_info *relp;
1336: char *codep;
1337: register char *cp;
1338: int relsz, codesz;
1339:
1340: relsz = reloc.size;
1341: relp = (struct relocation_info *)malloc(relsz);
1342: codesz = text.size;
1343: codep = (char *)malloc(codesz);
1344: if (relp == 0 || codep == 0)
1345: error(1, "out of memory (load2td)");
1346: mget((char *)relp, relsz, &reloc);
1347: rpend = &relp[relsz / sizeof (struct relocation_info)];
1348: mget(codep, codesz, &text);
1349: for (rp = relp; rp < rpend; rp++) {
1350: cp = codep + rp->r_address;
1351: /*
1352: * Pick up previous value at location to be relocated.
1353: */
1354: switch (rp->r_length) {
1355:
1356: case 0: /* byte */
1357: tw = *cp;
1358: break;
1359:
1360: case 1: /* word */
1361: tw = *(short *)cp;
1362: break;
1363:
1364: case 2: /* long */
1365: tw = *(long *)cp;
1366: break;
1367:
1368: default:
1369: error(1, "load2td botch: bad length");
1370: }
1371: /*
1372: * If relative to an external which is defined,
1373: * resolve to a simpler kind of reference in the
1374: * result file. If the external is undefined, just
1375: * convert the symbol number to the number of the
1376: * symbol in the result file and leave it undefined.
1377: */
1378: if (rp->r_extern) {
1379: /*
1380: * Search the hash table which maps local
1381: * symbol numbers to symbol tables entries
1382: * in the new a.out file.
1383: */
1384: lp = lochash[rp->r_symbolnum % LHSIZ];
1385: while (lp->l_index != rp->r_symbolnum) {
1386: lp = lp->l_link;
1387: if (lp == 0)
1388: error(1, "local symbol botch");
1389: }
1390: sp = lp->l_symbol;
1391: if (sp->n_type == N_EXT+N_UNDF)
1392: rp->r_symbolnum = nsym+symx(sp);
1393: else {
1394: rp->r_symbolnum = sp->n_type & N_TYPE;
1395: tw += sp->n_value;
1396: rp->r_extern = 0;
1397: }
1398: } else switch (rp->r_symbolnum & N_TYPE) {
1399: /*
1400: * Relocation is relative to the loaded position
1401: * of another segment. Update by the change in position
1402: * of that segment.
1403: */
1404: case N_TEXT:
1405: tw += ctrel;
1406: break;
1407: case N_DATA:
1408: tw += cdrel;
1409: break;
1410: case N_BSS:
1411: tw += cbrel;
1412: break;
1413: case N_ABS:
1414: break;
1415: default:
1416: error(1, "relocation format botch (symbol type))");
1417: }
1418: /*
1419: * Relocation is pc relative, so decrease the relocation
1420: * by the amount the current segment is displaced.
1421: * (E.g if we are a relative reference to a text location
1422: * from data space, we added the increase in the text address
1423: * above, and subtract the increase in our (data) address
1424: * here, leaving the net change the relative change in the
1425: * positioning of our text and data segments.)
1426: */
1427: if (rp->r_pcrel)
1428: tw -= creloc;
1429: /*
1430: * Put the value back in the segment,
1431: * while checking for overflow.
1432: */
1433: switch (rp->r_length) {
1434:
1435: case 0: /* byte */
1436: if (tw < -128 || tw > 127)
1437: error(0, "byte displacement overflow");
1438: *cp = tw;
1439: break;
1440: case 1: /* word */
1441: if (tw < -32768 || tw > 32767)
1442: error(0, "word displacement overflow");
1443: *(short *)cp = tw;
1444: break;
1445: case 2: /* long */
1446: *(long *)cp = tw;
1447: break;
1448: }
1449: /*
1450: * If we are saving relocation information,
1451: * we must convert the address in the segment from
1452: * the old .o file into an address in the segment in
1453: * the new a.out, by adding the position of our
1454: * segment in the new larger segment.
1455: */
1456: if (rflag)
1457: rp->r_address += position;
1458: }
1459: bwrite(codep, codesz, b1);
1460: if (rflag)
1461: bwrite(relp, relsz, b2);
1462: free((char *)relp);
1463: free(codep);
1464: }
1465:
1466: finishout()
1467: {
1468: register int i;
1469: int nsymt;
1470:
1471: if (sflag==0) {
1472: nsymt = symx(nextsym);
1473: for (i = 0; i < nsymt; i++)
1474: symwrite(xsym(i), sout);
1475: bwrite(&offset, sizeof offset, sout);
1476: }
1477: if (!ofilfnd) {
1478: unlink("a.out");
1479: if (link("l.out", "a.out") < 0)
1480: error(1, "cannot move l.out to a.out");
1481: ofilename = "a.out";
1482: }
1483: delarg = errlev;
1484: delexit();
1485: }
1486:
1487: mkfsym(s)
1488: char *s;
1489: {
1490:
1491: if (sflag || xflag)
1492: return;
1493: cursym.n_un.n_name = s;
1494: cursym.n_type = N_TEXT;
1495: cursym.n_value = torigin;
1496: symwrite(&cursym, sout);
1497: }
1498:
1499: getarhdr()
1500: {
1501: register char *cp;
1502:
1503: mget((char *)&archdr, sizeof archdr, &text);
1504: for (cp=archdr.ar_name; cp<&archdr.ar_name[sizeof(archdr.ar_name)];)
1505: if (*cp++ == ' ') {
1506: cp[-1] = 0;
1507: return;
1508: }
1509: }
1510:
1511: mget(loc, n, sp)
1512: register STREAM *sp;
1513: register char *loc;
1514: {
1515: register char *p;
1516: register int take;
1517:
1518: top:
1519: if (n == 0)
1520: return;
1521: if (sp->size && sp->nibuf) {
1522: p = sp->ptr;
1523: take = sp->size;
1524: if (take > sp->nibuf)
1525: take = sp->nibuf;
1526: if (take > n)
1527: take = n;
1528: n -= take;
1529: sp->size -= take;
1530: sp->nibuf -= take;
1531: sp->pos += take;
1532: do
1533: *loc++ = *p++;
1534: while (--take > 0);
1535: sp->ptr = p;
1536: goto top;
1537: }
1538: if (n > p_blksize) {
1539: take = n - n % p_blksize;
1540: lseek(infil, (sp->bno+1)<<p_blkshift, 0);
1541: if (take > sp->size || read(infil, loc, take) != take)
1542: error(1, "premature EOF");
1543: loc += take;
1544: n -= take;
1545: sp->size -= take;
1546: sp->pos += take;
1547: dseek(sp, (sp->bno+1+(take>>p_blkshift))<<p_blkshift, -1);
1548: goto top;
1549: }
1550: *loc++ = get(sp);
1551: --n;
1552: goto top;
1553: }
1554:
1555: symwrite(sp, bp)
1556: struct nlist *sp;
1557: struct biobuf *bp;
1558: {
1559: register int len;
1560: register char *str;
1561:
1562: str = sp->n_un.n_name;
1563: if (str) {
1564: sp->n_un.n_strx = offset;
1565: len = strlen(str) + 1;
1566: bwrite(str, len, strout);
1567: offset += len;
1568: }
1569: bwrite(sp, sizeof (*sp), bp);
1570: sp->n_un.n_name = str;
1571: }
1572:
1573: dseek(sp, loc, s)
1574: register STREAM *sp;
1575: long loc, s;
1576: {
1577: register PAGE *p;
1578: register b, o;
1579: int n;
1580:
1581: b = loc>>p_blkshift;
1582: o = loc&p_blkmask;
1583: if (o&01)
1584: error(1, "loader error; odd offset");
1585: --sp->pno->nuser;
1586: if ((p = &page[0])->bno!=b && (p = &page[1])->bno!=b)
1587: if (p->nuser==0 || (p = &page[0])->nuser==0) {
1588: if (page[0].nuser==0 && page[1].nuser==0)
1589: if (page[0].bno < page[1].bno)
1590: p = &page[0];
1591: p->bno = b;
1592: lseek(infil, loc & ~(long)p_blkmask, 0);
1593: if ((n = read(infil, p->buff, p_blksize)) < 0)
1594: n = 0;
1595: p->nibuf = n;
1596: } else
1597: error(1, "botch: no pages");
1598: ++p->nuser;
1599: sp->bno = b;
1600: sp->pno = p;
1601: if (s != -1) {sp->size = s; sp->pos = 0;}
1602: sp->ptr = (char *)(p->buff + o);
1603: if ((sp->nibuf = p->nibuf-o) <= 0)
1604: sp->size = 0;
1605: }
1606:
1607: char
1608: get(asp)
1609: STREAM *asp;
1610: {
1611: register STREAM *sp;
1612:
1613: sp = asp;
1614: if ((sp->nibuf -= sizeof(char)) < 0) {
1615: dseek(sp, ((long)(sp->bno+1)<<p_blkshift), (long)-1);
1616: sp->nibuf -= sizeof(char);
1617: }
1618: if ((sp->size -= sizeof(char)) <= 0) {
1619: if (sp->size < 0)
1620: error(1, "premature EOF");
1621: ++fpage.nuser;
1622: --sp->pno->nuser;
1623: sp->pno = (PAGE *) &fpage;
1624: }
1625: sp->pos += sizeof(char);
1626: return(*sp->ptr++);
1627: }
1628:
1629: getfile(acp)
1630: char *acp;
1631: {
1632: register int c;
1633: char arcmag[SARMAG+1];
1634: struct stat stb;
1635:
1636: archdr.ar_name[0] = '\0';
1637: filname = acp;
1638: if (filname[0] == '-' && filname[1] == 'l')
1639: infil = libopen(filname + 2, O_RDONLY);
1640: else
1641: infil = open(filname, O_RDONLY);
1642: if (infil < 0)
1643: error(1, "cannot open");
1644: fstat(infil, &stb);
1645: page[0].bno = page[1].bno = -1;
1646: page[0].nuser = page[1].nuser = 0;
1647: c = stb.st_blksize;
1648: if (c == 0 || (c & (c - 1)) != 0) {
1649: /* use default size if not a power of two */
1650: c = BLKSIZE;
1651: }
1652: if (p_blksize != c) {
1653: p_blksize = c;
1654: p_blkmask = c - 1;
1655: for (p_blkshift = 0; c > 1 ; p_blkshift++)
1656: c >>= 1;
1657: if (page[0].buff != NULL)
1658: free(page[0].buff);
1659: page[0].buff = (char *)malloc(p_blksize);
1660: if (page[0].buff == NULL)
1661: error(1, "ran out of memory (getfile)");
1662: if (page[1].buff != NULL)
1663: free(page[1].buff);
1664: page[1].buff = (char *)malloc(p_blksize);
1665: if (page[1].buff == NULL)
1666: error(1, "ran out of memory (getfile)");
1667: }
1668: text.pno = reloc.pno = (PAGE *) &fpage;
1669: fpage.nuser = 2;
1670: dseek(&text, 0L, SARMAG);
1671: if (text.size <= 0)
1672: error(1, "premature EOF");
1673: mget((char *)arcmag, SARMAG, &text);
1674: arcmag[SARMAG] = 0;
1675: if (strcmp(arcmag, ARMAG))
1676: return (0);
1677: dseek(&text, SARMAG, sizeof archdr);
1678: if (text.size <= 0)
1679: return (1);
1680: getarhdr();
1681: if (strncmp(archdr.ar_name, "__.SYMDEF", sizeof(archdr.ar_name)) != 0)
1682: return (1);
1683: return (stb.st_mtime > atol(archdr.ar_date) ? 3 : 2);
1684: }
1685:
1686: /*
1687: * Search for a library with given name
1688: * using the directory search array.
1689: */
1690: libopen(name, oflags)
1691: char *name;
1692: int oflags;
1693: {
1694: register char *p, *cp;
1695: register int i;
1696: static char buf[MAXPATHLEN+1];
1697: int fd = -1;
1698:
1699: if (*name == '\0') /* backwards compat */
1700: name = "a";
1701: for (i = 0; i < ndir && fd == -1; i++) {
1702: p = buf;
1703: for (cp = dirs[i]; *cp; *p++ = *cp++)
1704: ;
1705: *p++ = '/';
1706: for (cp = "lib"; *cp; *p++ = *cp++)
1707: ;
1708: for (cp = name; *cp; *p++ = *cp++)
1709: ;
1710: cp = ".a";
1711: while (*p++ = *cp++)
1712: ;
1713: fd = open(buf, oflags);
1714: }
1715: if (fd != -1)
1716: filname = buf;
1717: return (fd);
1718: }
1719:
1720: struct nlist **
1721: lookup()
1722: {
1723: register int sh;
1724: register struct nlist **hp;
1725: register char *cp, *cp1;
1726: register struct symseg *gp;
1727: register int i;
1728:
1729: sh = 0;
1730: for (cp = cursym.n_un.n_name; *cp;)
1731: sh = (sh<<1) + *cp++;
1732: sh = (sh & 0x7fffffff) % HSIZE;
1733: for (gp = symseg; gp < &symseg[NSEG]; gp++) {
1734: if (gp->sy_first == 0) {
1735: gp->sy_first = (struct nlist *)
1736: calloc(NSYM, sizeof (struct nlist));
1737: gp->sy_hfirst = (struct nlist **)
1738: calloc(HSIZE, sizeof (struct nlist *));
1739: if (gp->sy_first == 0 || gp->sy_hfirst == 0)
1740: error(1, "ran out of space for symbol table");
1741: gp->sy_last = gp->sy_first + NSYM;
1742: gp->sy_hlast = gp->sy_hfirst + HSIZE;
1743: }
1744: if (gp > csymseg)
1745: csymseg = gp;
1746: hp = gp->sy_hfirst + sh;
1747: i = 1;
1748: do {
1749: if (*hp == 0) {
1750: if (gp->sy_used == NSYM)
1751: break;
1752: return (hp);
1753: }
1754: cp1 = (*hp)->n_un.n_name;
1755: for (cp = cursym.n_un.n_name; *cp == *cp1++;)
1756: if (*cp++ == 0)
1757: return (hp);
1758: hp += i;
1759: i += 2;
1760: if (hp >= gp->sy_hlast)
1761: hp -= HSIZE;
1762: } while (i < HSIZE);
1763: if (i > HSIZE)
1764: error(1, "hash table botch");
1765: }
1766: error(1, "symbol table overflow");
1767: /*NOTREACHED*/
1768: }
1769:
1770: symfree(saved)
1771: struct nlist *saved;
1772: {
1773: register struct symseg *gp;
1774: register struct nlist *sp;
1775:
1776: for (gp = csymseg; gp >= symseg; gp--, csymseg--) {
1777: sp = gp->sy_first + gp->sy_used;
1778: if (sp == saved) {
1779: nextsym = sp;
1780: return;
1781: }
1782: for (sp--; sp >= gp->sy_first; sp--) {
1783: gp->sy_hfirst[sp->n_hash] = 0;
1784: gp->sy_used--;
1785: if (sp == saved) {
1786: nextsym = sp;
1787: return;
1788: }
1789: }
1790: }
1791: if (saved == 0)
1792: return;
1793: error(1, "symfree botch");
1794: }
1795:
1796: struct nlist **
1797: slookup(s)
1798: char *s;
1799: {
1800:
1801: cursym.n_un.n_name = s;
1802: cursym.n_type = N_EXT+N_UNDF;
1803: cursym.n_value = 0;
1804: return (lookup());
1805: }
1806:
1807: enter(hp)
1808: register struct nlist **hp;
1809: {
1810: register struct nlist *sp;
1811:
1812: if (*hp==0) {
1813: if (hp < csymseg->sy_hfirst || hp >= csymseg->sy_hlast)
1814: error(1, "enter botch");
1815: *hp = lastsym = sp = csymseg->sy_first + csymseg->sy_used;
1816: csymseg->sy_used++;
1817: sp->n_un.n_name = cursym.n_un.n_name;
1818: sp->n_type = cursym.n_type;
1819: sp->n_hash = hp - csymseg->sy_hfirst;
1820: sp->n_value = cursym.n_value;
1821: nextsym = lastsym + 1;
1822: return(1);
1823: } else {
1824: lastsym = *hp;
1825: return(0);
1826: }
1827: }
1828:
1829: symx(sp)
1830: struct nlist *sp;
1831: {
1832: register struct symseg *gp;
1833:
1834: if (sp == 0)
1835: return (0);
1836: for (gp = csymseg; gp >= symseg; gp--)
1837: /* <= is sloppy so nextsym will always work */
1838: if (sp >= gp->sy_first && sp <= gp->sy_last)
1839: return ((gp - symseg) * NSYM + sp - gp->sy_first);
1840: error(1, "symx botch");
1841: /*NOTREACHED*/
1842: }
1843:
1844: symreloc()
1845: {
1846: if(funding) return;
1847: switch (cursym.n_type & 017) {
1848:
1849: case N_TEXT:
1850: case N_EXT+N_TEXT:
1851: cursym.n_value += ctrel;
1852: return;
1853:
1854: case N_DATA:
1855: case N_EXT+N_DATA:
1856: cursym.n_value += cdrel;
1857: return;
1858:
1859: case N_BSS:
1860: case N_EXT+N_BSS:
1861: cursym.n_value += cbrel;
1862: return;
1863:
1864: case N_EXT+N_UNDF:
1865: return;
1866:
1867: default:
1868: if (cursym.n_type&N_EXT)
1869: cursym.n_type = N_EXT+N_ABS;
1870: return;
1871: }
1872: }
1873:
1874: error(n, s)
1875: char *s;
1876: {
1877:
1878: if (errlev==0)
1879: printf("ld:");
1880: if (filname) {
1881: printf("%s", filname);
1882: if (n != -1 && archdr.ar_name[0])
1883: printf("(%s)", archdr.ar_name);
1884: printf(": ");
1885: }
1886: printf("%s\n", s);
1887: if (n == -1)
1888: return;
1889: if (n)
1890: delexit();
1891: errlev = 2;
1892: }
1893:
1894: readhdr(loc)
1895: off_t loc;
1896: {
1897:
1898: dseek(&text, loc, (long)sizeof(filhdr));
1899: mget((short *)&filhdr, sizeof(filhdr), &text);
1900: if (N_BADMAG(filhdr)) {
1901: if (filhdr.a_magic == OARMAG)
1902: error(1, "old archive");
1903: error(1, "bad magic number");
1904: }
1905: if (filhdr.a_text&01 || filhdr.a_data&01)
1906: error(1, "text/data size odd");
1907: if (filhdr.a_magic == NMAGIC || filhdr.a_magic == ZMAGIC) {
1908: cdrel = -round(filhdr.a_text, pagesize);
1909: cbrel = cdrel - filhdr.a_data;
1910: } else if (filhdr.a_magic == OMAGIC) {
1911: cdrel = -filhdr.a_text;
1912: cbrel = cdrel - filhdr.a_data;
1913: } else
1914: error(1, "bad format");
1915: }
1916:
1917: round(v, r)
1918: int v;
1919: u_long r;
1920: {
1921:
1922: r--;
1923: v += r;
1924: v &= ~(long)r;
1925: return(v);
1926: }
1927:
1928: #define NSAVETAB 8192
1929: char *savetab;
1930: int saveleft;
1931:
1932: char *
1933: savestr(cp)
1934: register char *cp;
1935: {
1936: register int len;
1937:
1938: len = strlen(cp) + 1;
1939: if (len > saveleft) {
1940: saveleft = NSAVETAB;
1941: if (len > saveleft)
1942: saveleft = len;
1943: savetab = malloc(saveleft);
1944: if (savetab == 0)
1945: error(1, "ran out of memory (savestr)");
1946: }
1947: strncpy(savetab, cp, len);
1948: cp = savetab;
1949: savetab += len;
1950: saveleft -= len;
1951: return (cp);
1952: }
1953:
1954: bopen(bp, off, bufsize)
1955: register struct biobuf *bp;
1956: {
1957:
1958: bp->b_ptr = bp->b_buf = malloc(bufsize);
1959: if (bp->b_ptr == (char *)0)
1960: error(1, "ran out of memory (bopen)");
1961: bp->b_bufsize = bufsize;
1962: bp->b_nleft = bufsize - (off % bufsize);
1963: bp->b_off = off;
1964: bp->b_link = biobufs;
1965: biobufs = bp;
1966: }
1967:
1968: int bwrerror;
1969:
1970: bwrite(p, cnt, bp)
1971: register char *p;
1972: register int cnt;
1973: register struct biobuf *bp;
1974: {
1975: register int put;
1976: register char *to;
1977:
1978: top:
1979: if (cnt == 0)
1980: return;
1981: if (bp->b_nleft) {
1982: put = bp->b_nleft;
1983: if (put > cnt)
1984: put = cnt;
1985: bp->b_nleft -= put;
1986: to = bp->b_ptr;
1987: bcopy(p, to, put);
1988: bp->b_ptr += put;
1989: p += put;
1990: cnt -= put;
1991: goto top;
1992: }
1993: if (cnt >= bp->b_bufsize) {
1994: if (bp->b_ptr != bp->b_buf)
1995: bflush1(bp);
1996: put = cnt - cnt % bp->b_bufsize;
1997: if (boffset != bp->b_off)
1998: lseek(biofd, bp->b_off, 0);
1999: if (write(biofd, p, put) != put) {
2000: bwrerror = 1;
2001: error(1, "output write error");
2002: }
2003: bp->b_off += put;
2004: boffset = bp->b_off;
2005: p += put;
2006: cnt -= put;
2007: goto top;
2008: }
2009: bflush1(bp);
2010: goto top;
2011: }
2012:
2013: bflush()
2014: {
2015: register struct biobuf *bp;
2016:
2017: if (bwrerror)
2018: return;
2019: for (bp = biobufs; bp; bp = bp->b_link)
2020: bflush1(bp);
2021: }
2022:
2023: bflush1(bp)
2024: register struct biobuf *bp;
2025: {
2026: register int cnt = bp->b_ptr - bp->b_buf;
2027:
2028: if (cnt == 0)
2029: return;
2030: if (boffset != bp->b_off)
2031: lseek(biofd, bp->b_off, 0);
2032: if (write(biofd, bp->b_buf, cnt) != cnt) {
2033: bwrerror = 1;
2034: error(1, "output write error");
2035: }
2036: bp->b_off += cnt;
2037: boffset = bp->b_off;
2038: bp->b_ptr = bp->b_buf;
2039: bp->b_nleft = bp->b_bufsize;
2040: }
2041:
2042: bflushc(bp, c)
2043: register struct biobuf *bp;
2044: {
2045:
2046: bflush1(bp);
2047: bputc(c, bp);
2048: }
2049:
2050: bseek(bp, off)
2051: register struct biobuf *bp;
2052: register off_t off;
2053: {
2054: bflush1(bp);
2055:
2056: bp->b_nleft = bp->b_bufsize - (off % bp->b_bufsize);
2057: bp->b_off = off;
2058: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.