mirror of
				https://github.com/SEPPDROID/Digital-Research-Source-Code.git
				synced 2025-10-26 09:54:20 +00:00 
			
		
		
		
	Upload
Digital Research
This commit is contained in:
		
							
								
								
									
										389
									
								
								CPM OPERATING SYSTEMS/CPM 68K/1.0X SOURCES/v103/cgen/util.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										389
									
								
								CPM OPERATING SYSTEMS/CPM 68K/1.0X SOURCES/v103/cgen/util.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,389 @@ | ||||
| /* | ||||
| 	Copyright 1983 | ||||
| 	Alcyon Corporation | ||||
| 	8716 Production Ave. | ||||
| 	San Diego, Ca.  92121 | ||||
| */ | ||||
|  | ||||
| #include "cgen.h" | ||||
| #include "cskel.h" | ||||
|  | ||||
| char null[]; | ||||
| char *opname[]; | ||||
|  | ||||
| /* xnalloc - allocate address-indexed node*/ | ||||
| char *xnalloc(type,ar,off,xr,xt)	/* returns ptr to node alloced*/ | ||||
| int type;						/* data type*/ | ||||
| int ar;							/* address register*/ | ||||
| int off;						/* 8-bit offset*/ | ||||
| int xr;							/* index register*/ | ||||
| int xt;							/* index register type*/ | ||||
| { | ||||
| 	register struct indexnode *xp; | ||||
|  | ||||
| 	xp = talloc(sizeof(*xp)); | ||||
| 	xp->t_op = SYMBOL; | ||||
| 	xp->t_type = type; | ||||
| 	xp->t_sc = INDEXED; | ||||
| 	xp->t_reg = ar; | ||||
| 	xp->t_su = SU_ADDR; | ||||
| 	xp->t_offset = off; | ||||
| 	xp->t_xreg = xr; | ||||
| 	xp->t_xtype = xt; | ||||
| 	return(xp); | ||||
| } | ||||
|  | ||||
| /* tcopy - expression tree copy*/ | ||||
| char *tcopy(tp,autof)					/* returns ptr to copied tree*/ | ||||
| struct tnode *tp; | ||||
| int autof;								/* can do AUTOINC and AUTODEC? */ | ||||
| { | ||||
| 	register char *p; | ||||
| 	register short op; | ||||
|  | ||||
| 	op = tp->t_op; | ||||
| 	switch( op ) { | ||||
|  | ||||
| 	case SYMBOL: | ||||
| 		if( tp->t_sc == EXTERNAL || tp->t_sc == EXTOFF ) | ||||
| 			p = cenalloc(tp->t_type,tp->t_sc,tp->t_symbol); | ||||
| 		else { | ||||
| 			p = snalloc(tp->t_type,tp->t_sc,tp->t_offset,0,0); | ||||
| 			p->t_label = tp->t_label; | ||||
| 		} | ||||
| 		p->t_offset = tp->t_offset; | ||||
| 		p->t_reg = tp->t_reg; | ||||
| 		return(p); | ||||
|  | ||||
| 	case CINT: | ||||
| 		return(cnalloc(tp->t_type,tp->t_value)); | ||||
|  | ||||
| 	case CLONG: | ||||
| 		return(lcnalloc(tp->t_type,tp->t_lvalue)); | ||||
|  | ||||
| 	case CFLOAT:		/*[vlh] 3.4 */ | ||||
| 		return(fpcnalloc(tp->t_type,tp->t_lvalue)); | ||||
|  | ||||
| 	case DCLONG: | ||||
| 		p = lcnalloc(tp->t_type,tp->t_lvalue); | ||||
| 		p->t_op = DCLONG; | ||||
| 		return(p); | ||||
|  | ||||
| 	case AUTOINC:		/*[mac] 4.2*/ | ||||
| 	case AUTODEC: | ||||
| 		p = snalloc(tp->t_type,AUTO,0,0,0); | ||||
| 		if( autof == DOAUTO ) | ||||
| 			p->t_op = op; | ||||
| 		p->t_reg = tp->t_reg; | ||||
| 		return(p); | ||||
|  | ||||
| 	default: | ||||
| 		p = tnalloc(op,tp->t_type,0,0,null,null); | ||||
| 		if( ISASGOP(op) || op == PREINC || op == PREDEC )	/* [mac] 4.2 */ | ||||
| 			p->t_right = tcopy(tp->t_right,DOAUTO); | ||||
| 		else if( BINOP(op) ) | ||||
| 			p->t_right = tcopy(tp->t_right,autof); | ||||
| 		p->t_left = tcopy(tp->t_left,autof); | ||||
| 		return(p); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* outaexpr - output address expression*/ | ||||
| outaexpr(tp,flags)				/* returns - none*/ | ||||
| struct tnode *tp;				/* pointer to tree*/ | ||||
| int flags;						/* flags (IMMED,LOFFSET,...)*/ | ||||
| { | ||||
| 	register short off, reg, lab; | ||||
| 	long l; | ||||
|  | ||||
| 	if( tp->t_op == ADDR ) { | ||||
| 		tp = tp->t_left; | ||||
| 		putchar('#'); | ||||
| 	} | ||||
| 	off = tp->t_offset; | ||||
| 	reg = tp->t_reg; | ||||
| 	lab = tp->t_label; | ||||
| 	switch( tp->t_op ) { | ||||
| 	 | ||||
| 		case AUTOINC: | ||||
| 			printf("(R%d)+",reg); | ||||
| 			break; | ||||
| 		 | ||||
| 		case AUTODEC: | ||||
| 			printf("-(R%d)",reg); | ||||
| 			break; | ||||
| 	 | ||||
| 		case CINT: | ||||
| 			if( flags & IMMED ) | ||||
| 				putchar('#'); | ||||
| 			printf("%d",tp->t_value); | ||||
| 			break; | ||||
| 	 | ||||
| 		case DCLONG: | ||||
| 		case CLONG: | ||||
| 		case CFLOAT:		/*[vlh] 3.4 */ | ||||
| 			if( flags & IMMED ) | ||||
| 				putchar('#'); | ||||
| 			outlval(tp->t_lvalue); | ||||
| 			break; | ||||
| 	 | ||||
| 		case SYMBOL: | ||||
| 			if( off ) { | ||||
| 				switch( tp->t_sc ) { | ||||
| 		 | ||||
| 				default: | ||||
| 					printf("%d+",off); | ||||
| 					break; | ||||
|  | ||||
| 				case REGOFF: | ||||
| 					printf("%d",off); | ||||
| 				case CINDR: | ||||
| 				case CLINDR: | ||||
| 				case CFINDR:	/* [vlh] 3.4 */ | ||||
| 				case INDEXED: | ||||
| 					break; | ||||
|  | ||||
| 				case REGISTER: | ||||
| 					error("invalid register expression"); | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 			switch( tp->t_sc ) { | ||||
| 	 | ||||
| 				case REGISTER: | ||||
| 					printf("R%d",reg); | ||||
| 					break; | ||||
| 		 | ||||
| 				case REGOFF: | ||||
| 					printf("(R%d)",reg); | ||||
| 					break; | ||||
| 		 | ||||
| 				case EXTERNAL: | ||||
| 					printf("_%.8s",tp->t_symbol); | ||||
| 					break; | ||||
| 		 | ||||
| 				case EXTOFF: | ||||
| 					printf("_%.8s(R%d)",tp->t_symbol,reg); | ||||
| 					break; | ||||
| 		 | ||||
| 				case STATIC: | ||||
| 					printf("L%d",lab); | ||||
| 					break; | ||||
| 		 | ||||
| 				case STATOFF: | ||||
| 					printf("L%d(R%d)",lab,reg); | ||||
| 					break; | ||||
|  | ||||
| 				case INDEXED: | ||||
| 					printf("%d(R%d,R%d",off,reg,tp->t_xreg); | ||||
| 					outatype(tp->t_xtype); | ||||
| 					putchar(')'); | ||||
| 					break; | ||||
|  | ||||
| 				case CINDR: | ||||
| 					printf("%d",off); | ||||
| 					break; | ||||
| 		/* | ||||
| 		 * the following will work on: PDP-11, 68000, IBM-360, VAX, etc. | ||||
| 		 * it will not work on word machines or on machines where either | ||||
| 		 * longs two ints or two shorts. | ||||
| 		 */ | ||||
| 				case CLINDR: | ||||
| 				case CFINDR:	/* [vlh] 3.4 */ | ||||
| 					l.hiword = tp->t_offset; | ||||
| 					l.loword = tp->t_ssp; | ||||
| 					outlval(l); | ||||
| 					break; | ||||
| 		 | ||||
| 				default: | ||||
| 					error("invalid storage class %d\n",tp->t_sc); | ||||
| 					break; | ||||
| 			} | ||||
| 			break; | ||||
|  | ||||
| 		default: | ||||
| 			error("invalid operator %s\n",opname[tp->t_op]); | ||||
| 			break; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* outlval - output long value*/ | ||||
| /*		This is a big pain because the PDP-11 doesn't do long divides*/ | ||||
| /*		in hardware.*/ | ||||
| outlval(lval) | ||||
| long lval; | ||||
| { | ||||
| 	char digs[8]; | ||||
| 	register short i, c; | ||||
|  | ||||
| 	i = 0; | ||||
| 	do { | ||||
| 		digs[i++] = lval & 0xf; | ||||
| 		lval >>= 4; | ||||
| 		lval &= 0xfffffff; | ||||
| 	} while ( lval ); | ||||
| 	putchar('$'); | ||||
| 	while( --i >= 0 ) { | ||||
| 		c = digs[i]; | ||||
| 		putchar(c>=10?c+('a'-10):c+'0'); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* outtype - output 68000 type (null, .b, .l) depending on data type*/ | ||||
| outtype(type) | ||||
| int type; | ||||
| { | ||||
| 	if( ISFLOAT(type) || LONGORPTR(type) ) | ||||
| 		printf(".l"); | ||||
| 	else if( type == CHAR ) | ||||
| 		printf(".b"); | ||||
| } | ||||
|  | ||||
| /* outatype - output address type (.l or null) depending on data type*/ | ||||
| outatype(type) | ||||
| int type; | ||||
| { | ||||
| 	if( LONGORPTR(type) || ISFLOAT(type) ) | ||||
| 		printf(".l"); | ||||
| } | ||||
|  | ||||
| /* outextend - output register extension to long depending on type*/ | ||||
| outextend(tp,type,reg)						/* returns - none*/ | ||||
| struct tnode *tp;							/* tree to convert from*/ | ||||
| int type;									/* type to convert to*/ | ||||
| int reg;									/* register to convert*/ | ||||
| { | ||||
| 	if( (ISDREG(reg)) && !(LONGORPTR(tp->t_type)) && (LONGORPTR(type)) ) { | ||||
| 		if( UNSIGN(tp->t_type) ) | ||||
| 			OUTUEXT(reg); | ||||
| 		else | ||||
| 			OUTEXT(reg); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* outrr - output register to register instruction*/ | ||||
| outrr(ins,r1,r2,tp) | ||||
| char *ins; | ||||
| int r1; | ||||
| int r2; | ||||
| struct tnode *tp; | ||||
| { | ||||
| 	printf("%s",ins); | ||||
| 	if( ISAREG(r1) || ISAREG(r2) ) | ||||
| 		outatype(tp->t_type); | ||||
| 	else | ||||
| 		outtype(tp->t_type); | ||||
| 	printf(" R%d,R%d\n",r1,r2); | ||||
| } | ||||
|  | ||||
| /* outmovr - output "move[type] R1,R2" instruction*/ | ||||
| outmovr(r1,r2,tp) | ||||
| int r1; | ||||
| int r2; | ||||
| struct tnode *tp; | ||||
| { | ||||
| 	if( r1 != r2 ) | ||||
| 		outrr("move",r1,r2,tp); | ||||
| } | ||||
|  | ||||
| /* outcmpm - output "cmpm[type] (R1)+,(R2)+"*/ | ||||
| outcmpm(tp) | ||||
| struct tnode *tp; | ||||
| { | ||||
| 	register int lreg, rreg; | ||||
|  | ||||
| 	if (!m68010) {	/* 68000 version cmpm ok !!! */ | ||||
| 		printf("cmpm"); | ||||
| 		outtype(tp->t_left->t_type); | ||||
| 		printf(" (R%d)+,(R%d)+\n",tp->t_left->t_reg,tp->t_right->t_reg); | ||||
| 	} | ||||
| 	else	/* generating code for 68010, micro code is bad for cmpm */ | ||||
| 		fakecmpm(tp->t_left->t_type,tp->t_left->t_reg,tp->t_right->t_reg); | ||||
| } | ||||
|  | ||||
| /* fakecmpm - output 68010 fake cmpm for bad microcode [vlh] 4.2 */ | ||||
| fakecmpm(type,lreg,rreg) | ||||
| int type,lreg,rreg; | ||||
| { | ||||
| 	printf("cmp"); | ||||
| 	outtype(type); | ||||
| 	printf(" (R%d),(R%d)\n",lreg,rreg); | ||||
| 	if( ISFLOAT(type) || LONGORPTR(type) ) | ||||
| 		printf("addq #4,R%d\naddq #4,R%d\n",lreg,rreg); | ||||
| 	else if( type == CHAR ) | ||||
| 		printf("addq #1,R%d\naddq #1,R%d\n",lreg,rreg); | ||||
| 	else | ||||
| 		printf("addq #2,R%d\naddq #2,R%d\n",lreg,rreg); | ||||
| } | ||||
|  | ||||
| /* outcreg - output reference to compiler temp register*/ | ||||
| outcreg(reg) | ||||
| int reg; | ||||
| { | ||||
| 	if( (DREG(reg)) > HICREG ) | ||||
| 		error("expression too complex"); | ||||
| 	printf("R%d",reg); | ||||
| } | ||||
|  | ||||
| /* outcmp0 - output a compare with 0, special for address register*/ | ||||
| outcmp0(reg,tp) | ||||
| int reg; | ||||
| struct tnode *tp; | ||||
| { | ||||
| 	if( ISAREG(reg) ) { | ||||
| 		printf("cmp"); | ||||
| 		outatype(tp->t_type); | ||||
| 		printf(" #0,R%d\n",reg); | ||||
| 	} | ||||
| 	else { | ||||
| 		printf("tst"); | ||||
| 		outtype(tp->t_type); | ||||
| 		printf(" R%d\n",reg); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* outrpush - output "move[type] R1,[-](sp)"*/ | ||||
| outrpush(reg,tp,pflag) | ||||
| int reg; | ||||
| struct tnode *tp; | ||||
| int pflag; | ||||
| { | ||||
| 	printf("move"); | ||||
| 	outatype(tp->t_type); | ||||
| 	printf(" R%d,%c(sp)\n",reg,pflag?'-':'\0'); | ||||
| } | ||||
|  | ||||
| outdbra(reg,lab) | ||||
| int reg; | ||||
| int lab; | ||||
| { | ||||
| 	printf("dbra R%d,L%d\n",reg,lab); | ||||
| } | ||||
|  | ||||
| /* cenalloc - code generator external node allocation*/ | ||||
| /*		This may be coalesced into enalloc in parser.*/ | ||||
| char *cenalloc(type,sc,sym)		/* returns ptr to node alloced*/ | ||||
| int type;						/* type of symbol*/ | ||||
| int sc;							/* storage class*/ | ||||
| char *sym;						/* symbol name*/ | ||||
| { | ||||
| 	register struct extnode *ep; | ||||
|  | ||||
| 	ep = talloc(sizeof(*ep)); | ||||
| 	ep->t_op = SYMBOL; | ||||
| 	ep->t_type = type; | ||||
| 	ep->t_sc = sc; | ||||
| 	ep->t_su = 0; | ||||
| 	ep->t_offset = 0; | ||||
| 	symcopy(sym,ep->t_symbol); | ||||
| 	return(ep); | ||||
| } | ||||
|  | ||||
| /*popstack - clear off the stack after a call if necessary */ | ||||
| popstack(nb) | ||||
| { | ||||
| 	if (nb > 0 && nb <= 8) | ||||
| 		printf("addq.l #%d,sp\n",nb); | ||||
| 	else if (nb > 0) | ||||
| 		printf("adda.l #%d,sp\n",nb); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user