⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 obj.c

📁 这是一个同样来自贝尔实验室的和UNIX有着渊源的操作系统, 其简洁的设计和实现易于我们学习和理解
💻 C
📖 第 1 页 / 共 2 页
字号:
#define	EXTERN#include	"l.h"#include	<ar.h>#ifndef	DEFAULT#define	DEFAULT	'9'#endifchar	*noname		= "<none>";char	symname[]	= SYMDEF;char	thechar		= 'q';char	*thestring 	= "power";/* *	-H0 -T0x200000 -R0		is boot *	-H1 -T0x100000 -R4		is Be boot *	-H2 -T4128 -R4096		is plan9 format *	-H3 -T0x02010000 -D0x00001000	is raw *	-H4 -T0x1000200 -D0x20000e00 -R4	is aix xcoff executable *	-H5 -T0x80010000 -t0x10000	ELF, phys = 10000, vaddr = 0x8001... */static intisobjfile(char *f){	int n, v;	Biobuf *b;	char buf1[5], buf2[SARMAG];	b = Bopen(f, OREAD);	if(b == nil)		return 0;	n = Bread(b, buf1, 5);	if(n == 5 && (buf1[2] == 1 && buf1[3] == '<' || buf1[3] == 1 && buf1[4] == '<'))		v = 1;	/* good enough for our purposes */	else{		Bseek(b, 0, 0);		n = Bread(b, buf2, SARMAG);		v = n == SARMAG && strncmp(buf2, ARMAG, SARMAG) == 0;	}	Bterm(b);	return v;}voidmain(int argc, char *argv[]){	int c;	char *a;	Binit(&bso, 1, OWRITE);	cout = -1;	listinit();	outfile = 0;	nerrors = 0;	curtext = P;	HEADTYPE = -1;	INITTEXT = -1;	INITDAT = -1;	INITRND = -1;	INITENTRY = 0;	ARGBEGIN {	default:		c = ARGC();		if(c >= 0 && c < sizeof(debug))			debug[c]++;		break;	case 'o':		outfile = ARGF();		break;	case 'E':		a = ARGF();		if(a)			INITENTRY = a;		break;	case 'T':		a = ARGF();		if(a)			INITTEXT = atolwhex(a);		break;	case 'D':		a = ARGF();		if(a)			INITDAT = atolwhex(a);		break;	case 'R':		a = ARGF();		if(a)			INITRND = atolwhex(a);		break;	case 'H':		a = ARGF();		if(a)			HEADTYPE = atolwhex(a);		break;	case 'x':	/* produce export table */		doexp = 1;		if(argv[1] != nil && argv[1][0] != '-' && !isobjfile(argv[1]))			readundefs(ARGF(), SEXPORT);		break;	case 'u':	/* produce dynamically loadable module */		dlm = 1;		if(argv[1] != nil && argv[1][0] != '-' && !isobjfile(argv[1]))			readundefs(ARGF(), SIMPORT);		break;	} ARGEND	USED(argc);	if(*argv == 0) {		diag("usage: ql [-options] objects");		errorexit();	}	if(!debug['9'] && !debug['U'] && !debug['B'])		debug[DEFAULT] = 1;	r0iszero = debug['0'] == 0;	if(HEADTYPE == -1) {		if(debug['U'])			HEADTYPE = 0;		if(debug['B'])			HEADTYPE = 1;		if(debug['9'])			HEADTYPE = 2;	}	switch(HEADTYPE) {	default:		diag("unknown -H option");		errorexit();	case 0:	/* boot */		HEADR = 32L;		if(INITTEXT == -1)			INITTEXT = 0x200000L;		if(INITDAT == -1)			INITDAT = 0;		if(INITRND == -1)			INITRND = 4096L;		break;	case 1:	/* Be boot format (PEF) */		HEADR = 208L;		if(INITTEXT == -1)			INITTEXT = 0x100000;		if(INITDAT == -1)			INITDAT = 0;		if(INITRND == -1)			INITRND = 4;		break;	case 2:	/* plan 9 */		HEADR = 32L;		if(INITTEXT == -1)			INITTEXT = 4128;		if(INITDAT == -1)			INITDAT = 0;		if(INITRND == -1)			INITRND = 4096;		break;	case 3:	/* raw */		HEADR = 0;		if(INITTEXT == -1)			INITTEXT = 4128;		if(INITDAT == -1) {			INITDAT = 0;			INITRND = 4;		}		if(INITRND == -1)			INITRND = 0;		break;	case 4:	/* aix unix xcoff executable */		HEADR = 20L+72L+3*40L;		if(INITTEXT == -1)			INITTEXT = 0x1000000L+HEADR;		if(INITDAT == -1)			INITDAT = 0x20000000;		if(INITRND == -1)			INITRND = 0;		break;	case 5:	/* elf executable */		HEADR = rnd(52L+3*32L, 16);		if(INITTEXT == -1)			INITTEXT = 0x00400000L+HEADR;		if(INITDAT == -1)			INITDAT = 0x10000000;		if(INITRND == -1)			INITRND = 0;		break;	}	if(INITDAT != 0 && INITRND != 0)		print("warning: -D0x%lux is ignored because of -R0x%lux\n",			INITDAT, INITRND);	if(debug['v'])		Bprint(&bso, "HEADER = -H0x%x -T0x%lux -D0x%lux -R0x%lux\n",			HEADTYPE, INITTEXT, INITDAT, INITRND);	Bflush(&bso);	zprg.as = AGOK;	zprg.reg = NREG;	zprg.from.name = D_NONE;	zprg.from.type = D_NONE;	zprg.from.reg = NREG;	zprg.from3 = zprg.from;	zprg.to = zprg.from;	buildop();	histgen = 0;	textp = P;	datap = P;	pc = 0;	dtype = 4;	if(outfile == 0)		outfile = "q.out";	cout = create(outfile, 1, 0775);	if(cout < 0) {		diag("%s: cannot create", outfile);		errorexit();	}	nuxiinit();	version = 0;	cbp = buf.cbuf;	cbc = sizeof(buf.cbuf);	firstp = prg();	lastp = firstp;	if(INITENTRY == 0) {		INITENTRY = "_main";		if(debug['p'])			INITENTRY = "_mainp";		if(!debug['l'])			lookup(INITENTRY, 0)->type = SXREF;	} else		lookup(INITENTRY, 0)->type = SXREF;	while(*argv)		objfile(*argv++);	if(!debug['l'])		loadlib();	firstp = firstp->link;	if(firstp == P)		goto out;	if(doexp || dlm){		EXPTAB = "_exporttab";		zerosig(EXPTAB);		zerosig("etext");		zerosig("edata");		zerosig("end");		if(dlm){			import();			HEADTYPE = 2;			INITTEXT = INITDAT = 0;			INITRND = 8;			INITENTRY = EXPTAB;		}		export();	}	patch();	if(debug['p'])		if(debug['1'])			doprof1();		else			doprof2();	dodata();	follow();	if(firstp == P)		goto out;	noops();	span();	asmb();	undef();out:	if(debug['v']) {		Bprint(&bso, "%5.2f cpu time\n", cputime());		Bprint(&bso, "%ld memory used\n", tothunk);		Bprint(&bso, "%d sizeof adr\n", sizeof(Adr));		Bprint(&bso, "%d sizeof prog\n", sizeof(Prog));	}	errorexit();}voidloadlib(void){	int i;	long h;	Sym *s;loop:	xrefresolv = 0;	for(i=0; i<libraryp; i++) {		if(debug['v'])			Bprint(&bso, "%5.2f autolib: %s (from %s)\n", cputime(), library[i], libraryobj[i]);		objfile(library[i]);	}	if(xrefresolv)	for(h=0; h<nelem(hash); h++)	for(s = hash[h]; s != S; s = s->link)		if(s->type == SXREF)			goto loop;}voiderrorexit(void){	Bflush(&bso);	if(nerrors) {		if(cout >= 0)			remove(outfile);		exits("error");	}	exits(0);}voidobjfile(char *file){	long off, esym, cnt, l;	int f, work;	Sym *s;	char magbuf[SARMAG];	char name[100], pname[150];	struct ar_hdr arhdr;	char *e, *start, *stop;	if(file[0] == '-' && file[1] == 'l') {		if(debug['9'])			sprint(name, "/%s/lib/lib", thestring);		else			sprint(name, "/usr/%clib/lib", thechar);		strcat(name, file+2);		strcat(name, ".a");		file = name;	}	if(debug['v'])		Bprint(&bso, "%5.2f ldobj: %s\n", cputime(), file);	Bflush(&bso);	f = open(file, 0);	if(f < 0) {		diag("cannot open file: %s", file);		errorexit();	}	l = read(f, magbuf, SARMAG);	if(l != SARMAG || strncmp(magbuf, ARMAG, SARMAG)){		/* load it as a regular file */		l = seek(f, 0L, 2);		seek(f, 0L, 0);		ldobj(f, l, file);		close(f);		return;	}	l = read(f, &arhdr, SAR_HDR);	if(l != SAR_HDR) {		diag("%s: short read on archive file symbol header", file);		goto out;	}	if(strncmp(arhdr.name, symname, strlen(symname))) {		diag("%s: first entry not symbol header", file);		goto out;	}	esym = SARMAG + SAR_HDR + atolwhex(arhdr.size);	off = SARMAG + SAR_HDR;	/*	 * just bang the whole symbol file into memory	 */	seek(f, off, 0);	cnt = esym - off;	start = malloc(cnt + 10);	cnt = read(f, start, cnt);	if(cnt <= 0){		close(f);		return;	}	stop = &start[cnt];	memset(stop, 0, 10);	work = 1;	while(work){		if(debug['v'])			Bprint(&bso, "%5.2f library pass: %s\n", cputime(), file);		Bflush(&bso);		work = 0;		for(e = start; e < stop; e = strchr(e+5, 0) + 1) {			s = lookup(e+5, 0);			if(s->type != SXREF)				continue;			sprint(pname, "%s(%s)", file, s->name);			if(debug['v'])				Bprint(&bso, "%5.2f library: %s\n", cputime(), pname);			Bflush(&bso);			l = e[1] & 0xff;			l |= (e[2] & 0xff) << 8;			l |= (e[3] & 0xff) << 16;			l |= (e[4] & 0xff) << 24;			seek(f, l, 0);			l = read(f, &arhdr, SAR_HDR);			if(l != SAR_HDR)				goto bad;			if(strncmp(arhdr.fmag, ARFMAG, sizeof(arhdr.fmag)))				goto bad;			l = atolwhex(arhdr.size);			ldobj(f, l, pname);			if(s->type == SXREF) {				diag("%s: failed to load: %s", file, s->name);				errorexit();			}			work = 1;			xrefresolv = 1;		}	}	return;bad:	diag("%s: bad or out of date archive", file);out:	close(f);}intzaddr(uchar *p, Adr *a, Sym *h[]){	int i, c;	long l;	Sym *s;	Auto *u;	c = p[2];	if(c < 0 || c > NSYM){		print("sym out of range: %d\n", c);		p[0] = AEND+1;		return 0;	}	a->type = p[0];	a->reg = p[1];	a->sym = h[c];	a->name = p[3];	c = 4;	if(a->reg > NREG) {		print("register out of range %d\n", a->reg);		p[0] = AEND+1;		return 0;	/*  force real diagnostic */	}	switch(a->type) {	default:		print("unknown type %d\n", a->type);		p[0] = AEND+1;		return 0;	/* force real diagnostic */	case D_NONE:	case D_REG:	case D_FREG:	case D_CREG:	case D_FPSCR:	case D_MSR:	case D_SREG:	case D_OPT:		break;	case D_SPR:	case D_DCR:	case D_BRANCH:	case D_OREG:	case D_CONST:		a->offset = p[4] | (p[5]<<8) |			(p[6]<<16) | (p[7]<<24);		c += 4;		break;	case D_SCONST:		memmove(a->sval, p+4, NSNAME);		c += NSNAME;		break;	case D_FCONST:		a->ieee.l = p[4] | (p[5]<<8) |			(p[6]<<16) | (p[7]<<24);		a->ieee.h = p[8] | (p[9]<<8) |			(p[10]<<16) | (p[11]<<24);		c += 8;		break;	}	s = a->sym;	if(s == S)		goto out;	i = a->name;	if(i != D_AUTO && i != D_PARAM)		goto out;	l = a->offset;	for(u=curauto; u; u=u->link)		if(u->sym == s)		if(u->type == i) {			if(u->aoffset > l)				u->aoffset = l;			goto out;		}	u = malloc(sizeof(Auto));	u->link = curauto;	curauto = u;	u->sym = s;	u->aoffset = l;	u->type = i;out:	return c;}voidaddlib(char *obj){	char name[1024], comp[256], *p;	int i;	if(histfrogp <= 0)		return;	if(histfrog[0]->name[1] == '/') {		sprint(name, "");		i = 1;	} else	if(histfrog[0]->name[1] == '.') {		sprint(name, ".");		i = 0;	} else {		if(debug['9'])			sprint(name, "/%s/lib", thestring);		else			sprint(name, "/usr/%clib", thechar);		i = 0;	}	for(; i<histfrogp; i++) {		snprint(comp, sizeof comp, histfrog[i]->name+1);		for(;;) {			p = strstr(comp, "$O");			if(p == 0)				break;			memmove(p+1, p+2, strlen(p+2)+1);			p[0] = thechar;		}		for(;;) {			p = strstr(comp, "$M");			if(p == 0)				break;			if(strlen(comp)+strlen(thestring)-2+1 >= sizeof comp) {				diag("library component too long");				return;			}			memmove(p+strlen(thestring), p+2, strlen(p+2)+1);			memmove(p, thestring, strlen(thestring));		}		if(strlen(name) + strlen(comp) + 3 >= sizeof(name)) {			diag("library component too long");			return;		}		strcat(name, "/");		strcat(name, comp);	}	for(i=0; i<libraryp; i++)		if(strcmp(name, library[i]) == 0)			return;	if(libraryp == nelem(library)){		diag("too many autolibs; skipping %s", name);		return;	}	p = malloc(strlen(name) + 1);	strcpy(p, name);	library[libraryp] = p;	p = malloc(strlen(obj) + 1);	strcpy(p, obj);	libraryobj[libraryp] = p;	libraryp++;}voidaddhist(long line, int type){	Auto *u;	Sym *s;	int i, j, k;	u = malloc(sizeof(Auto));	s = malloc(sizeof(Sym));	s->name = malloc(2*(histfrogp+1) + 1);	u->sym = s;	u->type = type;	u->aoffset = line;	u->link = curhist;	curhist = u;	j = 1;	for(i=0; i<histfrogp; i++) {		k = histfrog[i]->value;		s->name[j+0] = k>>8;		s->name[j+1] = k;		j += 2;	}}voidhisttoauto(void){	Auto *l;	while(l = curhist) {		curhist = l->link;		l->link = curauto;		curauto = l;	}}voidcollapsefrog(Sym *s){	int i;	/*	 * bad encoding of path components only allows	 * MAXHIST components. if there is an overflow,	 * first try to collapse xxx/..	 */	for(i=1; i<histfrogp; i++)		if(strcmp(histfrog[i]->name+1, "..") == 0) {			memmove(histfrog+i-1, histfrog+i+1,				(histfrogp-i-1)*sizeof(histfrog[0]));			histfrogp--;			goto out;		}	/*	 * next try to collapse .	 */	for(i=0; i<histfrogp; i++)		if(strcmp(histfrog[i]->name+1, ".") == 0) {			memmove(histfrog+i, histfrog+i+1,				(histfrogp-i-1)*sizeof(histfrog[0]));			goto out;		}	/*	 * last chance, just truncate from front	 */	memmove(histfrog+0, histfrog+1,		(histfrogp-1)*sizeof(histfrog[0]));out:	histfrog[histfrogp-1] = s;}voidnopout(Prog *p){	p->as = ANOP;	p->from.type = D_NONE;	p->to.type = D_NONE;}uchar*readsome(int f, uchar *buf, uchar *good, uchar *stop, int max){	int n;	n = stop - good;	memmove(buf, good, stop - good);	stop = buf + n;	n = MAXIO - n;	if(n > max)		n = max;	n = read(f, stop, n);	if(n <= 0)		return 0;	return stop + n;}voidldobj(int f, long c, char *pn){	Prog *p, *t;	Sym *h[NSYM], *s, *di;	int v, o, r, skip;	long ipc;	uchar *bloc, *bsize, *stop;	ulong sig;	static int files;	static char **filen;	char **nfilen;	if((files&15) == 0){		nfilen = malloc((files+16)*sizeof(char*));		memmove(nfilen, filen, files*sizeof(char*));		free(filen);		filen = nfilen;	}	filen[files++] = strdup(pn);	bsize = buf.xbuf;	bloc = buf.xbuf;	di = S;newloop:	memset(h, 0, sizeof(h));	histfrogp = 0;	version++;	ipc = pc;	skip = 0;loop:	if(c <= 0)		goto eof;	r = bsize - bloc;	if(r < 100 && r < c) {		/* enough for largest prog */		bsize = readsome(f, buf.xbuf, bloc, bsize, c);		if(bsize == 0)			goto eof;		bloc = buf.xbuf;		goto loop;	}	o = bloc[0];		/* as */	if(o <= 0 || o >= ALAST) {		diag("%s: opcode out of range %d", pn, o);		print("	probably not a .q file\n");		errorexit();	}	if(o == ANAME || o == ASIGNAME) {		sig = 0;		if(o == ASIGNAME) {			sig = bloc[1] | (bloc[2]<<8) | (bloc[3]<<16) | (bloc[4]<<24);			bloc += 4;			c -= 4;		}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -