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

📄 span.c

📁 这是一个同样来自贝尔实验室的和UNIX有着渊源的操作系统, 其简洁的设计和实现易于我们学习和理解
💻 C
📖 第 1 页 / 共 2 页
字号:
#include	"l.h"#define	r0iszero	1voidspan(void){	Prog *p;	Sym *setext;	Optab *o;	int m;	long c;	if(debug['v'])		Bprint(&bso, "%5.2f span\n", cputime());	Bflush(&bso);	c = INITTEXT;	for(p = firstp; p != P; p = p->link) {		p->pc = c;		o = oplook(p);		m = o->size;		if(m == 0) {			if(p->as == ATEXT) {				curtext = p;				autosize = p->to.offset + 4;				if(p->from3.type == D_CONST) {					if(p->from3.offset & 3)						diag("illegal origin\n%P", p);					if(c > p->from3.offset)						diag("passed origin (#%lux)\n%P", c, p);					else						c = p->from3.offset;					p->pc = c;				}				if(p->from.sym != S)					p->from.sym->value = c;				continue;			}			if(p->as != ANOP)				diag("zero-width instruction\n%P", p);			continue;		}		c += m;	}	c = rnd(c, 4);	setext = lookup("etext", 0);	if(setext != S) {		setext->value = c;		textsize = c - INITTEXT;	}	if(INITRND)		INITDAT = rnd(c, INITRND);	if(debug['v'])		Bprint(&bso, "tsize = %lux\n", textsize);	Bflush(&bso);}		voidxdefine(char *p, int t, long v){	Sym *s;	s = lookup(p, 0);	if(s->type == 0 || s->type == SXREF) {		s->type = t;		s->value = v;	}}longregoff(Adr *a){	instoffset = 0;	aclass(a);	return instoffset;}intaclass(Adr *a){	Sym *s;	int t;	switch(a->type) {	case D_NONE:		return C_NONE;	case D_REG:		return C_REG;	case D_FREG:		return C_FREG;	case D_CREG:		return C_CREG;	case D_SPR:		if(a->offset == D_LR)			return C_LR;		if(a->offset == D_XER)			return C_XER;		if(a->offset == D_CTR)			return C_CTR;		return C_SPR;	case D_DCR:		return C_SPR;	case D_SREG:		return C_SREG;	case D_FPSCR:		return C_FPSCR;	case D_MSR:		return C_MSR;	case D_OREG:		switch(a->name) {		case D_EXTERN:		case D_STATIC:			if(a->sym == S)				break;			t = a->sym->type;			if(t == 0 || t == SXREF) {				diag("undefined external: %s in %s",					a->sym->name, TNAME);				a->sym->type = SDATA;			}			if(dlm){				instoffset = a->sym->value + a->offset;				switch(a->sym->type){				case STEXT:				case SLEAF:				case SCONST:				case SUNDEF:					break;				default:					instoffset += INITDAT;				}				return C_ADDR;			}			instoffset = a->sym->value + a->offset - BIG;			if(instoffset >= -BIG && instoffset < BIG)				return C_SEXT;			return C_LEXT;		case D_AUTO:			instoffset = autosize + a->offset;			if(instoffset >= -BIG && instoffset < BIG)				return C_SAUTO;			return C_LAUTO;		case D_PARAM:			instoffset = autosize + a->offset + 4L;			if(instoffset >= -BIG && instoffset < BIG)				return C_SAUTO;			return C_LAUTO;		case D_NONE:			instoffset = a->offset;			if(instoffset == 0)				return C_ZOREG;			if(instoffset >= -BIG && instoffset < BIG)				return C_SOREG;			return C_LOREG;		}		return C_GOK;	case D_OPT:		instoffset = a->offset & 31L;		if(a->name == D_NONE)			return C_SCON;		return C_GOK;	case D_CONST:		switch(a->name) {		case D_NONE:			instoffset = a->offset;		consize:			if(instoffset >= 0) {				if(r0iszero && instoffset == 0)					return C_ZCON;				if(instoffset <= 0x7fff)					return C_SCON;				if(instoffset <= 0xffff)					return C_ANDCON;				if((instoffset & 0xffff) == 0)					return C_UCON;				return C_LCON;			}			if(instoffset >= -0x8000)				return C_ADDCON;			if((instoffset & 0xffff) == 0)				return C_UCON;			return C_LCON;		case D_EXTERN:		case D_STATIC:			s = a->sym;			if(s == S)				break;			t = s->type;			if(t == 0 || t == SXREF) {				diag("undefined external: %s in %s",					s->name, TNAME);				s->type = SDATA;			}			if(s->type == STEXT || s->type == SLEAF || s->type == SUNDEF) {				instoffset = s->value + a->offset;				return C_LCON;			}			if(s->type == SCONST) {				instoffset = s->value + a->offset;				if(dlm)					return C_LCON;				goto consize;			}			if(!dlm){				instoffset = s->value + a->offset - BIG;				if(instoffset >= -BIG && instoffset < BIG && instoffset != 0)					return C_SECON;			}			instoffset = s->value + a->offset + INITDAT;			if(dlm)				return C_LCON;			/* not sure why this barfs */			return C_LCON;		/*			if(instoffset == 0)				return C_ZCON;			if(instoffset >= -0x8000 && instoffset <= 0xffff)				return C_SCON;			if((instoffset & 0xffff) == 0)				return C_UCON;			return C_LCON;		*/		case D_AUTO:			instoffset = autosize + a->offset;			if(instoffset >= -BIG && instoffset < BIG)				return C_SACON;			return C_LACON;		case D_PARAM:			instoffset = autosize + a->offset + 4L;			if(instoffset >= -BIG && instoffset < BIG)				return C_SACON;			return C_LACON;		}		return C_GOK;	case D_BRANCH:		return C_SBRA;	}	return C_GOK;}Optab*oplook(Prog *p){	int a1, a2, a3, a4, r;	char *c1, *c3, *c4;	Optab *o, *e;	a1 = p->optab;	if(a1)		return optab+(a1-1);	a1 = p->from.class;	if(a1 == 0) {		a1 = aclass(&p->from) + 1;		p->from.class = a1;	}	a1--;	a3 = p->from3.class;	if(a3 == 0) {		a3 = aclass(&p->from3) + 1;		p->from3.class = a3;	}	a3--;	a4 = p->to.class;	if(a4 == 0) {		a4 = aclass(&p->to) + 1;		p->to.class = a4;	}	a4--;	a2 = C_NONE;	if(p->reg != NREG)		a2 = C_REG;	r = p->as;	o = oprange[r].start;	if(o == 0)		o = oprange[r].stop; /* just generate an error */	e = oprange[r].stop;	c1 = xcmp[a1];	c3 = xcmp[a3];	c4 = xcmp[a4];	for(; o<e; o++)		if(o->a2 == a2)		if(c1[o->a1])		if(c3[o->a3])		if(c4[o->a4]) {			p->optab = (o-optab)+1;			return o;		}	diag("illegal combination %A %R %R %R %R",		p->as, a1, a2, a3, a4);	if(1||!debug['a'])		prasm(p);	if(o == 0)		errorexit();	return o;}intcmp(int a, int b){	if(a == b)		return 1;	switch(a) {	case C_LCON:		if(b == C_ZCON || b == C_SCON || b == C_UCON || b == C_ADDCON || b == C_ANDCON)			return 1;		break;	case C_ADDCON:		if(b == C_ZCON || b == C_SCON)			return 1;		break;	case C_ANDCON:		if(b == C_ZCON || b == C_SCON)			return 1;		break;	case C_SPR:		if(b == C_LR || b == C_XER || b == C_CTR)			return 1;		break;	case C_UCON:		if(b == C_ZCON)			return 1;		break;	case C_SCON:		if(b == C_ZCON)			return 1;		break;	case C_LACON:		if(b == C_SACON)			return 1;		break;	case C_LBRA:		if(b == C_SBRA)			return 1;		break;	case C_LEXT:		if(b == C_SEXT)			return 1;		break;	case C_LAUTO:		if(b == C_SAUTO)			return 1;		break;	case C_REG:		if(r0iszero && b == C_ZCON)			return 1;		break;	case C_LOREG:		if(b == C_ZOREG || b == C_SOREG)			return 1;		break;	case C_SOREG:		if(b == C_ZOREG)			return 1;		break;	case C_ANY:		return 1;	}	return 0;}intocmp(void *a1, void *a2){	Optab *p1, *p2;	int n;	p1 = a1;	p2 = a2;	n = p1->as - p2->as;	if(n)		return n;	n = p1->a1 - p2->a1;	if(n)		return n;	n = p1->a2 - p2->a2;	if(n)		return n;	n = p1->a3 - p2->a3;	if(n)		return n;	n = p1->a4 - p2->a4;	if(n)		return n;	return 0;}voidbuildop(void){	int i, n, r;	for(i=0; i<C_NCLASS; i++)		for(n=0; n<C_NCLASS; n++)			xcmp[i][n] = cmp(n, i);	for(n=0; optab[n].as != AXXX; n++)		;	qsort(optab, n, sizeof(optab[0]), ocmp);	for(i=0; i<n; i++) {		r = optab[i].as;		oprange[r].start = optab+i;		while(optab[i].as == r)			i++;		oprange[r].stop = optab+i;		i--;				switch(r)		{		default:			diag("unknown op in build: %A", r);			errorexit();		case ADCBF:	/* unary indexed: op (b+a); op (b) */			oprange[ADCBI] = oprange[r];			oprange[ADCBST] = oprange[r];			oprange[ADCBT] = oprange[r];			oprange[ADCBTST] = oprange[r];			oprange[ADCBZ] = oprange[r];			oprange[AICBI] = oprange[r];			break;		case AECOWX:	/* indexed store: op s,(b+a); op s,(b) */			oprange[ASTWCCC] = oprange[r];			break;		case AREM:	/* macro */			oprange[AREMCC] = oprange[r];			oprange[AREMV] = oprange[r];			oprange[AREMVCC] = oprange[r];			oprange[AREMU] = oprange[r];			oprange[AREMUCC] = oprange[r];			oprange[AREMUV] = oprange[r];			oprange[AREMUVCC] = oprange[r];			break;		case ADIVW:	/* op Rb[,Ra],Rd */			oprange[AMULHW] = oprange[r];			oprange[AMULHWCC] = oprange[r];			oprange[AMULHWU] = oprange[r];			oprange[AMULHWUCC] = oprange[r];			oprange[AMULLWCC] = oprange[r];			oprange[AMULLWVCC] = oprange[r];			oprange[AMULLWV] = oprange[r];			oprange[ADIVWCC] = oprange[r];			oprange[ADIVWV] = oprange[r];			oprange[ADIVWVCC] = oprange[r];			oprange[ADIVWU] = oprange[r];			oprange[ADIVWUCC] = oprange[r];			oprange[ADIVWUV] = oprange[r];			oprange[ADIVWUVCC] = oprange[r];			oprange[AADDCC] = oprange[r];

⌨️ 快捷键说明

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