mirror of
				https://github.com/SEPPDROID/Digital-Research-Source-Code.git
				synced 2025-10-25 01:14:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			312 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			312 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| 	Copyright 1982
 | |
| 	Alcyon Corporation
 | |
| 	8716 Production Ave.
 | |
| 	San Diego, Ca.  92121
 | |
| */
 | |
| 
 | |
| #include "parser.h"
 | |
| 
 | |
| int bol 1;
 | |
| int inittype;
 | |
| int begseq;
 | |
| /* int onepass; */
 | |
| 
 | |
| /*
 | |
| 	This interfaces the Parser and the Code Generator, note that these
 | |
| 	allow you to link together the Parser and the Code Generator.
 | |
| */
 | |
| 
 | |
| /* outbdata - set up for byte data*/
 | |
| outbdata()									/* returns - none*/
 | |
| {
 | |
| 	inittype = CHAR;
 | |
| 	printf("\t.dc.b ");
 | |
| }
 | |
| 
 | |
| /* outc - output a constant*/
 | |
| outc(type,value)							/* returns - none*/
 | |
| int type;
 | |
| int value;
 | |
| {
 | |
| 	if( type == CHAR )
 | |
| 		outbdata();
 | |
| 	else
 | |
| 		outwdata();
 | |
| 	printf("%d\n",value);
 | |
| }
 | |
| 
 | |
| /* outwdata - set up for word data*/
 | |
| outwdata()									/* returns - none*/
 | |
| {
 | |
| 	inittype = INT;
 | |
| 	printf("\t.dc.w ");
 | |
| }
 | |
| 
 | |
| /* outdata - set up for data output*/
 | |
| outdata()								/* returns - none*/
 | |
| {
 | |
| 	inittype = INT;
 | |
| 	printf("\t.data\n");
 | |
| }
 | |
| 
 | |
| /* outldata - set up for long data output*/
 | |
| outldata()								/* returns - none*/
 | |
| {
 | |
| 	inittype = LONG;
 | |
| 	printf("\t.data\n");
 | |
| }
 | |
| 
 | |
| /* outfpdata - set up for floating point data output*/
 | |
| outfpdata()								/*[vlh] 3.4 returns - none*/
 | |
| {
 | |
| 	inittype = FLOAT;
 | |
| 	printf("\t.data\n");
 | |
| }
 | |
| 
 | |
| /* outbentry - outputs block/function entry code*/
 | |
| outbentry(nlocs,nds,nas)			/* returns - none*/
 | |
| int nlocs;							/* local size*/
 | |
| int nds;							/* number of D registers*/
 | |
| int nas;							/* number of A registers*/
 | |
| {
 | |
| 	if( !nds && !nas )		/* adjust for 1 arg*/
 | |
| 		nlocs =+ 4;
 | |
| 	printf("\tlink R14,#%d\n",-nlocs);
 | |
| 	if( nds || nas ) {
 | |
| 		printf("\tmovem.l R%d-R7",7-nds);	/*7 for one arg*/
 | |
| 		if( nas ) {
 | |
| 			putchar('/');
 | |
| 			printf("R%d-R13",14-nas);
 | |
| 		}
 | |
| 		printf(",-(sp)\n");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* outbexit - output function exit code*/
 | |
| outbexit(nds,nas)						/* returns - none*/
 | |
| int nds;								/* number of D registers*/
 | |
| int nas;								/* number of A registers*/
 | |
| {
 | |
| 	if( nds || nas ) {
 | |
| 		printf("\ttst.l (sp)+\n\tmovem.l (sp)+,");	/*1 arg stuff*/
 | |
| 		if( nds ) {
 | |
| 			printf("R%d-R7",8-nds);
 | |
| 			if( nas )
 | |
| 				putchar('/');
 | |
| 		}
 | |
| 		if( nas )
 | |
| 			printf("R%d-R13",14-nas);
 | |
| 		putchar('\n');
 | |
| 	}
 | |
| 	printf("\tunlk R14\n\trts\n");
 | |
| }
 | |
| 
 | |
| /* outlocal - output local symbol for debugger*/
 | |
| outlocal(type,sc,sym,val)
 | |
| int type;								/* local name type*/
 | |
| int sc;									/* storage type*/
 | |
| char *sym;								/* symbol name*/
 | |
| int val;
 | |
| {
 | |
| 	switch( sc ) {
 | |
| 
 | |
| 	case STATIC:
 | |
| 		if( notfunction(type) )
 | |
| 			printf("\t~%.8s=L%d\n",sym,val);
 | |
| 		break;
 | |
| 
 | |
| 	case REGISTER:
 | |
| 		printf("\t~%.8s=R%d\n",sym,val);
 | |
| 		break;
 | |
| 
 | |
| 	case AUTO:
 | |
| 		printf("\t~%.8s=%d\n",sym,val);
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* outswitch - output switch table info*/
 | |
| outswitch(ncases,deflab,sp)		/* returns - none*/
 | |
| int ncases;							/* number of cases in switch*/
 | |
| int deflab;								/* default label*/
 | |
| struct swtch *sp;						/* switch table pointer*/
 | |
| {
 | |
| 	register int vdif, val, hval, i, tlab;
 | |
| 	register struct swtch *s;
 | |
| 
 | |
| 	val = sp->sw_value;
 | |
| 	hval = sp[ncases-1].sw_value;
 | |
| 	vdif = hval - val;
 | |
| 	if( ncases <= 4 ) {
 | |
| 	/*
 | |
| 	 *simple switch, do compares and brances, followed by branch to default
 | |
| 	 */
 | |
| 		for( s = sp; --ncases >= 0; s++ ) {
 | |
| 			if( !s->sw_value )
 | |
| 				printf("\ttst R0\n");
 | |
| 			else
 | |
| 				printf("\tcmp #%d,R0\n",s->sw_value);
 | |
| 			printf("\tbeq L%d\n",s->sw_label);
 | |
| 		}
 | |
| 		outgoto(deflab);
 | |
| 	}
 | |
| 	else if( vdif > 0 && vdif <= ncases*3 ) {
 | |
| 
 | |
| 	/*jump switch, uses value in R0 to index into table of labels*/
 | |
| 
 | |
| 		if( val )
 | |
| 			printf("\tsub #%d,R0\n",val);
 | |
| 		tlab = nextlabel++;
 | |
| 		printf("\tcmp #%d,R0\n\tbhi L%d\n",vdif,deflab);	/*check for max*/
 | |
| 		printf("\tasl #2,R0\n\tmove R0,R8\n\tadd.l #L%d,R8\n",tlab);
 | |
| 		printf("\tmove.l (R8),R8\n\tjmp (R8)\n");
 | |
| 		outdata();
 | |
| 		outlab(tlab);
 | |
| 		for( s = sp; val <= hval; val++ ) {
 | |
| 			if( val == s->sw_value ) {
 | |
| 				outclab(s->sw_label);
 | |
| 				s++;
 | |
| 			}
 | |
| 			else
 | |
| 				outclab(deflab);
 | |
| 		}
 | |
| 		outtext();
 | |
| 	}
 | |
| 	else {
 | |
| 	/*
 | |
| 	 * direct switch, searches down table of values for match, if match
 | |
| 	 * found, branches to corresponding label in label table.
 | |
| 	 */
 | |
| 		tlab = nextlabel++;
 | |
| 		printf("\text.l R0\n\tmove.l #L%d,R8\n\tmove #%d,R1\n",tlab,ncases);
 | |
| 		i = nextlabel++;
 | |
| 		outlab(i);					/*loop label*/
 | |
| 		printf("\tcmp.l (R8)+,R0\n\tdbeq R1,L%d\n",i);
 | |
| 		printf("\tmove.l %d(R8),R8\n\tjmp (R8)\n",ncases*4);
 | |
| 		outdata();
 | |
| 		outlab(tlab);
 | |
| 		for( s = sp, i = ncases; --i >= 0; s++ )
 | |
| 			outlcon(s->sw_value);
 | |
| 		outlcon(0);						/* mark for default label*/
 | |
| 		for( s = sp, i = ncases; --i >= 0; s++ )
 | |
| 			outclab(s->sw_label);
 | |
| 		outclab(deflab);
 | |
| 		outtext();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| outeof()
 | |
| {
 | |
| 	register int c;
 | |
| 
 | |
| 	v6flush(&sbuf);
 | |
| 	v6flush(&obuf);
 | |
| }
 | |
| 
 | |
| /* copysfile - copy string file to end of output file*/
 | |
| copysfile(fname)
 | |
| char *fname;
 | |
| {
 | |
| 	register int c;
 | |
| 
 | |
| 	close(sbuf.io_fd);
 | |
| 	if( fopen(fname,&sbuf,0) < 0 )		/* 3rd arg for versados */
 | |
| 		ferror("can't copy %s",fname);
 | |
| 	while( (c=getc(&sbuf)) > 0 )
 | |
| 		putc(c,&obuf);
 | |
| 	v6flush(&obuf);
 | |
| }
 | |
| 
 | |
| /* outword - output a word of data*/
 | |
| outword(w)								/* word expression*/
 | |
| int w;
 | |
| {
 | |
| 	if( begseq )
 | |
| 		putchar(',');
 | |
| 	begseq++;
 | |
| 	printf("%d",w);
 | |
| }
 | |
| 
 | |
| /* outlong - output a long data*/
 | |
| outlong(l)								/* returns - none*/
 | |
| long l;									/* long data to output*/
 | |
| {
 | |
| 	outwdata();
 | |
| 	outword(l.hiword);
 | |
| 	outword(l.loword);
 | |
| 	outendseq();
 | |
| }
 | |
| 
 | |
| /* outfp - output floating point data*/
 | |
| outfp(l)						/*[vlh] 3.4 returns - none*/
 | |
| long l;							/* floating point data to output*/
 | |
| {
 | |
| 	outwdata();
 | |
| 	outword(l.hiword);
 | |
| 	outword(l.loword);
 | |
| 	outendseq();
 | |
| }
 | |
| 
 | |
| outendseq()								/* returns - none*/
 | |
| {
 | |
| 	begseq = 0;
 | |
| 	putchar('\n');
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * outtstr - output text string
 | |
|  *		This outputs a string to the string file, this is used wherever
 | |
|  *		you cannot output the string directly to data space, such as in
 | |
|  *		the middle of expressions.
 | |
|  */
 | |
| outtstr(lab)
 | |
| int lab;
 | |
| {
 | |
| 	char *savep;
 | |
| 	int sbol;
 | |
| 
 | |
| 	savep = obp;					/*save to restore later...*/
 | |
| 	obp = &sbuf;
 | |
| 	sbol = bol;
 | |
| 	bol = 1;
 | |
| 	printf("\tL%d:",lab);
 | |
| 	outstr();
 | |
| 	obp = savep;
 | |
| 	bol = sbol;
 | |
| }
 | |
| 
 | |
| /* outstr - output a string as a sequence of bytes*/
 | |
| /*		Outputs ".dc.b <byte1>,<byte2>,...,<0>*/
 | |
| outstr()
 | |
| {
 | |
| 	register char *s;
 | |
| 	register int i;
 | |
| 
 | |
| 	outbdata();
 | |
| 	for( s = cstr, i = cstrsize; i > 0; i-- )
 | |
| 		outword(*s++ & 0xff);
 | |
| 	outendseq();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * putchar - handle outputting to intermediate or error files
 | |
|  *		This catches tabs to allow for the integration of the parser
 | |
|  *		and code generator into one pass.  By merely throwing away the
 | |
|  *		tabs here, the output will be OK for the assembler.
 | |
|  */
 | |
| putchar(c)
 | |
| char c;
 | |
| {
 | |
| 	if( !obp )
 | |
| 		write(1,&c,1);
 | |
| 	else if( c == '\t' ) {
 | |
| 		if( bol ) /* not used  && !onepass ) */
 | |
| 			putc('(',obp);					/*for code generator*/
 | |
| 	}
 | |
| 	else {
 | |
| 		bol = (c == '\n');
 | |
| 		putc(c,obp);
 | |
| 	}
 | |
| }
 |