881 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			881 lines
		
	
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef NORCSID
 | |
| static char rcsid[] = "$Header$";
 | |
| #endif
 | |
| 
 | |
| #include "assert.h"
 | |
| #include "param.h"
 | |
| #include "tables.h"
 | |
| #include "types.h"
 | |
| #include <cgg_cg.h>
 | |
| #include "data.h"
 | |
| #include "result.h"
 | |
| #include "state.h"
 | |
| #include "equiv.h"
 | |
| #include "extern.h"
 | |
| 
 | |
| /*
 | |
|  * (c) copyright 1983 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | |
|  *
 | |
|  *          This product is part of the Amsterdam Compiler Kit.
 | |
|  *
 | |
|  * Permission to use, sell, duplicate or disclose this software must be
 | |
|  * obtained in writing. Requests for such permissions may be sent to
 | |
|  *
 | |
|  *      Dr. Andrew S. Tanenbaum
 | |
|  *      Wiskundig Seminarium
 | |
|  *      Vrije Universiteit
 | |
|  *      Postbox 7161
 | |
|  *      1007 MC Amsterdam
 | |
|  *      The Netherlands
 | |
|  *
 | |
|  * Author: Hans van Staveren
 | |
|  */
 | |
| 
 | |
| #define ALLOW_NEXTEM	/* code generator is allowed new try of NEXTEM
 | |
| 			   in exceptional cases */
 | |
| 
 | |
| #define MAXPATTERN 5
 | |
| #define MAXREPLLEN 5    /* Max length of EM-replacement, should come from boot */
 | |
| 
 | |
| byte startupcode[] = { DO_NEXTEM };
 | |
| 
 | |
| byte *nextem();
 | |
| unsigned costcalc();
 | |
| unsigned docoerc();
 | |
| unsigned stackupto();
 | |
| string tostring();
 | |
| string ad2str();
 | |
| 
 | |
| #ifdef NDEBUG
 | |
| #define DEBUG(string)
 | |
| #else
 | |
| #include <stdio.h>
 | |
| #define DEBUG(string) {if(Debug) fprintf(stderr,"%-*d%s\n",4*level,level,string);}
 | |
| #endif
 | |
| 
 | |
| #define BROKE() {assert(origcp!=startupcode);DEBUG("BROKE");totalcost=INFINITY;goto doreturn;}
 | |
| #define CHKCOST() {if (totalcost>=costlimit) BROKE();}
 | |
| 
 | |
| #ifdef TABLEDEBUG
 | |
| int tablelines[MAXTDBUG];
 | |
| int ntableline;
 | |
| int set_fd,set_size;
 | |
| short *set_val;
 | |
| char *set_flag;
 | |
| #endif
 | |
| 
 | |
| unsigned codegen(codep,ply,toplevel,costlimit,forced) byte *codep; unsigned costlimit; {
 | |
| #ifndef NDEBUG
 | |
| 	byte *origcp=codep;
 | |
| 	static int level=0;
 | |
| #endif
 | |
| 	unsigned totalcost = 0;
 | |
| 	int inscoerc=0;
 | |
| 	int procarg[2];
 | |
| #ifdef ALLOW_NEXTEM
 | |
| 	int paniced;
 | |
| 	char *savebp;
 | |
| #endif
 | |
| 	state_t state;
 | |
| #define SAVEST	savestatus(&state)
 | |
| #define RESTST	restorestatus(&state)
 | |
| #define FREEST	/* nothing */
 | |
| #ifdef TABLEDEBUG
 | |
| 	extern char *tablename;
 | |
| #endif
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| 	level++;
 | |
| 	DEBUG("Entering codegen");
 | |
| #endif
 | |
| 	for (;;) {
 | |
| 	switch( (*codep++)&037 ) {
 | |
|     default:
 | |
| 	assert(FALSE);
 | |
| 	/* NOTREACHED */
 | |
| #ifdef TABLEDEBUG
 | |
|     case DO_DLINE: {
 | |
| 	int n;
 | |
| 
 | |
| 	getint(n,codep);
 | |
| 	tablelines[ntableline++] = n;
 | |
| 	if (ntableline>=MAXTDBUG)
 | |
| 		ntableline -= MAXTDBUG;
 | |
| 	if (set_fd)
 | |
| 		set_val[n>>4] &= ~(1<<(n&017));
 | |
| #ifndef NDEBUG
 | |
| 	if (Debug)
 | |
| 		fprintf(stderr,"code from \"%s\", line %d\n",tablename,n);
 | |
| #endif
 | |
| 	break;
 | |
|     }
 | |
| #endif
 | |
|     case DO_NEXTEM: {
 | |
| 	byte *bp;
 | |
| 	int n;
 | |
| 	unsigned mindistance,dist;
 | |
| 	register i;
 | |
| 	int cindex;
 | |
| 	int npos,pos[MAXRULE];
 | |
| 	unsigned mincost,t;
 | |
| 
 | |
| 	DEBUG("NEXTEM");
 | |
| 	tokpatlen = 0;
 | |
| 	nallreg=0;
 | |
| 	if (toplevel) {
 | |
| 		garbage_collect();
 | |
| 		totalcost=0;
 | |
| 	} else {
 | |
| 		if (--ply <= 0)
 | |
| 			goto doreturn;
 | |
| 	}
 | |
| 	if (stackheight>MAXFSTACK-7) {
 | |
| #ifndef NDEBUG
 | |
| 		if (Debug)
 | |
| 			fprintf(stderr,"Fakestack overflow threatens(%d), action ...\n",stackheight);
 | |
| #endif
 | |
| 		totalcost += stackupto(&fakestack[6],ply,toplevel);
 | |
| 	}
 | |
| #ifndef ALLOW_NEXTEM
 | |
| 	bp = nextem(toplevel);
 | |
| #else
 | |
| 	paniced=0;
 | |
| 	savebp = nextem(toplevel);
 | |
|     panic:
 | |
| 	bp = savebp;
 | |
| #endif
 | |
| 	if (bp == 0) {
 | |
| 		/*
 | |
| 		 * No pattern found, can be pseudo or error
 | |
| 		 * in table.
 | |
| 		 */
 | |
| 		if (toplevel) {
 | |
| 			codep--;
 | |
| 			DEBUG("pseudo");
 | |
| 			dopseudo();
 | |
| 		} else
 | |
| 			goto doreturn;
 | |
| 	} else {
 | |
| #ifndef NDEBUG
 | |
| 		chkregs();
 | |
| #endif
 | |
| 		n = *bp++;
 | |
| 		if (n==0) {	/* "procedure" */
 | |
| 			getint(i,bp);
 | |
| 			getint(procarg[0],bp);
 | |
| 			getint(procarg[1],bp);
 | |
| 			bp= &pattern[i];
 | |
| 			n = *bp++;
 | |
| 			DEBUG("PROC_CALL");
 | |
| 		}
 | |
| 		assert(n>0 && n<=MAXRULE);
 | |
| 		if (n>1) {
 | |
| 			mindistance = MAXINT; npos=0;
 | |
| 			for(i=0;i<n;i++) {
 | |
| 				getint(cindex,bp);
 | |
| 				dist=distance(cindex);
 | |
| #ifndef NDEBUG
 | |
| if (Debug)
 | |
| 	fprintf(stderr,"distance of pos %d is %u\n",i,dist);
 | |
| #endif
 | |
| 				if (dist<=mindistance) {
 | |
| 					if (dist<mindistance) {
 | |
| 						if(dist==0)
 | |
| 							goto gotit;
 | |
| 						npos=0;
 | |
| 						mindistance = dist;
 | |
| 					}
 | |
| 					pos[npos++] = cindex;
 | |
| 				}
 | |
| 			}
 | |
| 			assert(mindistance<MAXINT);
 | |
| 			if (npos>1) {
 | |
| 				/*
 | |
| 				 * More than 1 tokenpattern is a candidate.
 | |
| 				 * Decision has to be made by lookahead.
 | |
| 				 */
 | |
| 				SAVEST;
 | |
| 				mincost = costlimit-totalcost+1;
 | |
| 				for(i=0;i<npos;i++) {
 | |
| 					t=codegen(&coderules[pos[i]],ply,FALSE,mincost,0);
 | |
| #ifndef NDEBUG
 | |
| if (Debug)
 | |
| 	fprintf(stderr,"mincost %u,cost %u,pos %d\n",mincost,t,i);
 | |
| #endif
 | |
| 					if (t<mincost) {
 | |
| 						mincost = t;
 | |
| 						cindex = pos[i];
 | |
| 					}
 | |
| 					RESTST;
 | |
| 				}
 | |
| 				FREEST;
 | |
| 				if (totalcost+mincost>costlimit)
 | |
| 					BROKE();
 | |
| 			} else {
 | |
| 				cindex = pos[0];
 | |
| 			}
 | |
| 		} else {
 | |
| 			getint(cindex,bp);
 | |
| 		}
 | |
| 
 | |
| 	gotit:
 | |
| 		/*
 | |
| 		 * Now cindex contains the code-index of the best candidate
 | |
| 		 * so proceed to use it.
 | |
| 		 */
 | |
| 		codep = &coderules[cindex];
 | |
| 	}
 | |
| 	break;
 | |
|     }
 | |
|     case DO_COERC: {
 | |
| 	DEBUG("COERC");
 | |
| 	tokpatlen=1;
 | |
| 	inscoerc=1;
 | |
| 	break;
 | |
|     }
 | |
|     case DO_XXMATCH:
 | |
| 	DEBUG("XXMATCH");
 | |
|     case DO_XMATCH: {
 | |
| 	register i;
 | |
| 	int temp;
 | |
| 
 | |
| 	DEBUG("XMATCH");
 | |
| 	tokpatlen=(codep[-1]>>5)&07;
 | |
| 	for (i=0;i<tokpatlen;i++)
 | |
| 		getint(temp,codep);
 | |
| 	break;	/* match already checked by distance() */
 | |
|     }
 | |
|     case DO_MATCH: {
 | |
| 	register i;
 | |
| 	int j;
 | |
| 	unsigned mincost,t;
 | |
| 	token_p tp;
 | |
| 	int size,lsize;
 | |
| 	int tokexp[MAXPATTERN];
 | |
| 	int nregneeded;
 | |
| 	token_p regtp[MAXCREG];
 | |
| 	c3_p regcp[MAXCREG];
 | |
| 	rl_p regls[MAXCREG];
 | |
| 	c3_p cp,findcoerc();
 | |
| 	int sret;
 | |
| 	int stackpad;
 | |
| 	struct perm *tup,*ntup,*besttup,*tuples();
 | |
| 
 | |
| 	DEBUG("MATCH");
 | |
| 	tokpatlen=(codep[-1]>>5)&07;
 | |
| 	for(i=0;i<tokpatlen;i++)
 | |
| 		getint(tokexp[i],codep);
 | |
| 	tokexp[i] = 0;
 | |
| 	tp = &fakestack[stackheight-1];
 | |
| 	i=0;
 | |
| 	while (i<tokpatlen && tp>=fakestack) {
 | |
| 		size=tsize(tp);
 | |
| 		while (i<tokpatlen && (lsize=ssize(tokexp[i]))<=size) {
 | |
| 			size -= lsize;
 | |
| 			i++;
 | |
| 		}
 | |
| 		if (i<tokpatlen && size!=0) {
 | |
| 			totalcost += stackupto(tp,ply,toplevel);
 | |
| 			CHKCOST();
 | |
| 			break;
 | |
| 		}
 | |
| 		tp--;
 | |
| 	}
 | |
| 	tp = &fakestack[stackheight-1];
 | |
| 	i=0;
 | |
| 	while (i<tokpatlen && tp >= fakestack) {
 | |
| 		size = tsize(tp);
 | |
| 		lsize= ssize(tokexp[i]);
 | |
| 		if (size != lsize) {    /* find coercion */
 | |
| #ifdef MAXSPLIT
 | |
| 			sret = split(tp,&tokexp[i],ply,toplevel);
 | |
| 			if (sret==0) {
 | |
| #endif MAXSPLIT
 | |
| 				totalcost += stackupto(tp,ply,toplevel);
 | |
| 				CHKCOST();
 | |
| 				break;
 | |
| #ifdef MAXSPLIT
 | |
| 			}
 | |
| 			i += sret;
 | |
| #endif MAXSPLIT
 | |
| 		} else
 | |
| 			i += 1;
 | |
| 		tp--;
 | |
| 	}
 | |
|     nextmatch:
 | |
| 	tp = &fakestack[stackheight-1];
 | |
| 	i=0; nregneeded = 0;
 | |
| 	while (i<tokpatlen && tp>=fakestack) {
 | |
| 		if (!match(tp,&machsets[tokexp[i]],0)) {
 | |
| 			cp = findcoerc(tp, &machsets[tokexp[i]]);
 | |
| #ifndef NDEBUG
 | |
| if (Debug>1) fprintf(stderr,"findcoerc returns %d at position %d\n",cp,i);
 | |
| #endif
 | |
| 			if (cp==0) {
 | |
| 				for (j=0;j<nregneeded;j++)
 | |
| 					regtp[j] -= (tp-fakestack+1);
 | |
| 				totalcost += stackupto(tp,ply,toplevel);
 | |
| 				CHKCOST();
 | |
| 				break;
 | |
| 			} else {
 | |
| 				if (cp->c3_prop<0) {
 | |
| 					totalcost+=docoerc(tp,cp,ply,toplevel,0);
 | |
| 					CHKCOST();
 | |
| 				} else {
 | |
| #ifndef NDEBUG
 | |
| if(Debug>1) fprintf(stderr,"Register of type %d needed, remembering...\n",cp->c3_prop);
 | |
| #endif
 | |
| 					assert(nregneeded<MAXCREG);
 | |
| 					regtp[nregneeded] = tp;
 | |
| 					regcp[nregneeded] = cp;
 | |
| 					regls[nregneeded] = curreglist;
 | |
| 					nregneeded++;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		i++; tp--;
 | |
| 	}
 | |
| 	if (tokpatlen>stackheight) {
 | |
| #ifndef NDEBUG
 | |
| if(Debug>1) fprintf(stderr,"Pattern too long, %d with only %d items on stack\n",
 | |
| 		tokpatlen,stackheight);
 | |
| #endif
 | |
| 		stackpad = tokpatlen-stackheight;
 | |
| 		for (j=stackheight-1;j>=0;j--)
 | |
| 			fakestack[j+stackpad] = fakestack[j];
 | |
| 		for (j=0;j<stackpad;j++)
 | |
| 			fakestack[j].t_token=0;
 | |
| 		stackheight += stackpad;
 | |
| 		for (j=0;j<nregneeded;j++)
 | |
| 			regtp[j] += stackpad;
 | |
| 		for (tp = &fakestack[stackpad-1];i<tokpatlen && tp>=fakestack;i++,tp--) {
 | |
| 			cp = findcoerc((token_p) 0, &machsets[tokexp[i]]);
 | |
| 			if (cp==0) {
 | |
| 				for (j=0;j<nregneeded;j++)
 | |
| 					myfree(regls[j]);
 | |
| #ifndef ALLOW_NEXTEM
 | |
| 				assert(!toplevel);
 | |
| 				BROKE();
 | |
| #else
 | |
| 				assert(!(toplevel&&paniced));
 | |
| 				goto normalfailed;
 | |
| #endif
 | |
| 			}
 | |
| 			if (cp->c3_prop<0) {
 | |
| 				totalcost+=docoerc(tp,cp,ply,toplevel,0);
 | |
| 				CHKCOST();
 | |
| 			} else {
 | |
| 				assert(nregneeded<MAXCREG);
 | |
| 				regtp[nregneeded] = tp;
 | |
| 				regcp[nregneeded] = cp;
 | |
| 				regls[nregneeded] = curreglist;
 | |
| 				nregneeded++;
 | |
| 			}
 | |
| 		}
 | |
| 	} else
 | |
| 		stackpad=0;
 | |
| 	assert(i==tokpatlen);
 | |
| 	if (nregneeded==0)
 | |
| 		break;
 | |
| 	SAVEST;
 | |
| 	mincost=costlimit-totalcost+1;
 | |
| 	tup = tuples(regls,nregneeded);
 | |
| 	besttup=0;
 | |
| 	for (; tup != 0; tup = ntup) {
 | |
| #ifndef NDEBUG
 | |
| if(Debug>1) fprintf(stderr,"Next tuple %d,%d,%d,%d\n",
 | |
| 			tup->p_rar[0],
 | |
| 			tup->p_rar[1],
 | |
| 			tup->p_rar[2],
 | |
| 			tup->p_rar[3]);
 | |
| #endif
 | |
| 		ntup = tup->p_next;
 | |
| 		for (i=0,t=0;i<nregneeded && t<mincost; i++)
 | |
| 			t += docoerc(regtp[i],regcp[i],ply,FALSE,tup->p_rar[i]);
 | |
| 		if ( t<mincost && tokpatlen<=stackheight ) {
 | |
| #ifndef NDEBUG
 | |
| 			if (Debug>2)
 | |
| 				fprintf(stderr,"Continuing match after coercions\n");
 | |
| #endif
 | |
| 			t += codegen(codep,ply,FALSE,mincost-t,0);
 | |
| 		}
 | |
| 		if ( t<mincost && tokpatlen<=stackheight ) {
 | |
| 			mincost = t;
 | |
| 			besttup = tup;
 | |
| 		} else
 | |
| 			myfree(tup);
 | |
| 		RESTST;
 | |
| 	}
 | |
| 	FREEST;
 | |
| 	for (i=0;i<nregneeded;i++)
 | |
| 		myfree(regls[i]);
 | |
| 	if (totalcost+mincost>costlimit) {
 | |
| 		if (besttup)
 | |
| 			myfree(besttup);
 | |
| normalfailed:	if (stackpad!=tokpatlen) {
 | |
| 			if (stackpad) {
 | |
| 				if (costlimit<MAXINT)
 | |
| 					BROKE();
 | |
| 				for (i=0;i<stackheight-stackpad;i++)
 | |
| 					fakestack[i] = fakestack[i+stackpad];
 | |
| 				stackheight -= stackpad;
 | |
| 				totalcost += stackupto(&fakestack[stackheight-1],ply,toplevel);
 | |
| 			} else
 | |
| 				totalcost += stackupto(fakestack,ply,toplevel);
 | |
| 			CHKCOST();
 | |
| 			goto nextmatch;
 | |
| 		}
 | |
| 		totalcost += mincost;
 | |
| #ifndef ALLOW_NEXTEM
 | |
| 		BROKE();
 | |
| #else
 | |
| 		if (toplevel && !paniced) {
 | |
| 			stackheight=0;
 | |
| 			paniced++;
 | |
| 			DEBUG("PANIC!");
 | |
| 			goto panic;
 | |
| 		} else
 | |
| 			BROKE();
 | |
| #endif
 | |
| 	}
 | |
| 	for (i=0;i<nregneeded;i++)
 | |
| 		totalcost += docoerc(regtp[i],regcp[i],ply,toplevel,besttup->p_rar[i]);
 | |
| 	myfree(besttup);
 | |
| 	break;
 | |
|     }
 | |
|     case DO_REMOVE: {
 | |
| 	int texpno,nodeno;
 | |
| 	token_p tp;
 | |
| 	struct reginfo *rp;
 | |
| 
 | |
| 	DEBUG("REMOVE");
 | |
| 	if (codep[-1]&32) {
 | |
| 		getint(texpno,codep);
 | |
| 		getint(nodeno,codep);
 | |
| 	} else {
 | |
| 		getint(texpno,codep);
 | |
| 		nodeno=0;
 | |
| 	}
 | |
| 	for (tp= &fakestack[stackheight-tokpatlen-1];tp>=&fakestack[0];tp--)
 | |
| 		if (match(tp,&machsets[texpno],nodeno)) {
 | |
| 			/* investigate possible coercion to register */
 | |
| 			totalcost += stackupto(tp,ply,toplevel);
 | |
| 			CHKCOST();
 | |
| 			break;
 | |
| 		}
 | |
| 	for (rp=machregs;rp<machregs+NREGS;rp++)
 | |
| 		if (match(&rp->r_contents,&machsets[texpno],nodeno))
 | |
| 			rp->r_contents.t_token=0;
 | |
| 	break;
 | |
|     }
 | |
|     case DO_RREMOVE: {	/* register remove */
 | |
| 	register i;
 | |
| 	int nodeno;
 | |
| 	token_p tp;
 | |
| 	tkdef_p tdp;
 | |
| 	result_t result;
 | |
| 
 | |
| 	DEBUG("RREMOVE");
 | |
| 	getint(nodeno,codep);
 | |
| 	result=compute(&enodes[nodeno]);
 | |
| 	if (result.e_typ!=EV_REG)
 | |
| 		break;
 | |
| 	if ( in_stack(result.e_v.e_reg) ) BROKE() ; /* Check aside-stack */
 | |
| 	for (tp= &fakestack[stackheight-tokpatlen-1];tp>=&fakestack[0];tp--)
 | |
| 		if (tp->t_token==-1) {
 | |
| 			if(tp->t_att[0].ar==result.e_v.e_reg)
 | |
| 				goto gotone;
 | |
| 		} else {
 | |
| 			tdp = &tokens[tp->t_token];
 | |
| 			for(i=0;i<TOKENSIZE;i++)
 | |
| 				if (tdp->t_type[i]==EV_REG &&
 | |
| 				    tp->t_att[i].ar==result.e_v.e_reg)
 | |
| 					goto gotone;
 | |
| 		}
 | |
| 	break;
 | |
|     gotone:
 | |
| 	/* investigate possible coercion to register */
 | |
| 	totalcost += stackupto(tp,ply,toplevel);
 | |
| 	CHKCOST();
 | |
| 	break;
 | |
|     }
 | |
|     case DO_DEALLOCATE: {
 | |
| 	register i;
 | |
| 	tkdef_p tdp;
 | |
| 	int tinstno;
 | |
| 	token_t token;
 | |
| 
 | |
| 	DEBUG("DEALLOCATE");
 | |
| 	getint(tinstno,codep);
 | |
| 	instance(tinstno,&token);
 | |
| 	if (token.t_token==-1)
 | |
| 		chrefcount(token.t_att[0].ar,-1,TRUE);
 | |
| 	else {
 | |
| 		tdp= &tokens[token.t_token];
 | |
| 		for (i=0;i<TOKENSIZE;i++)
 | |
| 			if (tdp->t_type[i]==EV_REG)
 | |
| 				chrefcount(token.t_att[i].ar,-1,TRUE);
 | |
| 	}
 | |
| 	break;
 | |
|     }
 | |
|     case DO_REALLOCATE: {
 | |
| 	struct reginfo *rp;
 | |
| 
 | |
| 	DEBUG("REALLOCATE");
 | |
| 	for(rp=machregs+1;rp<machregs+NREGS;rp++)
 | |
| 		if(rp->r_tcount) {
 | |
| 			rp->r_refcount -= rp->r_tcount;
 | |
| 			rp->r_tcount = 0;
 | |
| 		}
 | |
| 	break;
 | |
|     }
 | |
|     case DO_ALLOCATE: {
 | |
| 	register i;
 | |
| 	int j;
 | |
| 	int tinstno;
 | |
| 	int npos,npos2,pos[NREGS],pos2[NREGS];
 | |
| 	unsigned mincost,t;
 | |
| 	struct reginfo *rp,**rpp;
 | |
| 	token_t token,mtoken,token2;
 | |
| 	int propno;
 | |
| 	int exactmatch;
 | |
| 	int decision;
 | |
| 
 | |
| 	if (codep[-1]&32) {
 | |
| 		getint(propno,codep);
 | |
| 		getint(tinstno,codep);
 | |
| 		DEBUG("ALLOCATE,INIT");
 | |
| 	} else {
 | |
| 		getint(propno,codep);
 | |
| 		tinstno=0;
 | |
| 		DEBUG("ALLOCATE,EMPTY");
 | |
| 	}
 | |
| 	instance(tinstno,&token);
 | |
| 	if (!forced) {
 | |
| 		do {
 | |
| 			npos=exactmatch=0;
 | |
| 			for(rpp=reglist[propno];rp= *rpp; rpp++)
 | |
| 				if (getrefcount(rp-machregs)==0) {
 | |
| 					pos[npos++] = rp-machregs;
 | |
| 					if (eqtoken(&rp->r_contents,&token))
 | |
| 						exactmatch++;
 | |
| 				}
 | |
| 			/*
 | |
| 			 * Now pos[] contains all free registers with desired
 | |
| 			 * property. If none then some stacking has to take place.
 | |
| 			 */
 | |
| 			if (npos==0) {
 | |
|   				if (stackheight<=tokpatlen) {
 | |
|   					if (!toplevel) {
 | |
|   						BROKE();
 | |
|  					} else {
 | |
|  						if (paniced)
 | |
|  							fatal("No regs available");
 | |
|  						totalcost += stackupto( &fakestack[0],ply,toplevel);
 | |
|  						goto panic;
 | |
|  					}
 | |
|   				}
 | |
|   				totalcost += stackupto( &fakestack[0],ply,toplevel);
 | |
|   				CHKCOST();
 | |
| 			}
 | |
| 		} while (npos==0);
 | |
| 		if (!exactmatch) {
 | |
| 			npos2=npos;
 | |
| 			for(i=0;i<npos;i++)
 | |
| 				pos2[i]=pos[i];
 | |
| 		} else {
 | |
| 			/*
 | |
| 			 * Now we are reducing the number of possible registers.
 | |
| 			 * We take only one equally likely register out of every
 | |
| 			 * equivalence class as given by set of properties.
 | |
| 			 */
 | |
| 			mtoken = token;
 | |
| 			npos2=0;
 | |
| 			for(i=0;i<npos;i++)
 | |
| 				if (eqtoken(&machregs[pos[i]].r_contents,&mtoken)) {
 | |
| 					pos2[npos2++] = pos[i];
 | |
| 					for(j=0;j<npos2-1;j++)
 | |
| 						if (eqregclass(pos2[j],pos[i])) {
 | |
| 							npos2--;
 | |
| 							break;
 | |
| 						}
 | |
| 				}
 | |
| 		}
 | |
| 		/*
 | |
| 		 * Now pos2[] contains all possibilities to try, if more than
 | |
| 		 * one, lookahead is necessary.
 | |
| 		 */
 | |
| 		token2.t_token= -1;
 | |
| 		for (i=1;i<TOKENSIZE;i++)
 | |
| 			token2.t_att[i].aw=0;
 | |
| 		if (npos2==1)
 | |
| 			decision=pos2[0];
 | |
| 		else {
 | |
| 			SAVEST;
 | |
| 			mincost=costlimit-totalcost+1;
 | |
| 			for(j=0;j<npos2;j++) {
 | |
| 				chrefcount(pos2[j],1,FALSE);
 | |
| 				token2.t_att[0].ar=pos2[j];
 | |
| 				allreg[nallreg++] = pos2[j];
 | |
| 				if (token.t_token != 0)
 | |
| 					t=move(&token,&token2,ply,FALSE,mincost);
 | |
| 				else {
 | |
| 					t = 0;
 | |
| 					erasereg(pos2[j]);
 | |
| 				}
 | |
| 				if (t<mincost)
 | |
| 					t += codegen(codep,ply,FALSE,mincost-t,0);
 | |
| 				if (t<mincost) {
 | |
| 					mincost=t;
 | |
| 					decision=pos2[j];
 | |
| 				}
 | |
| 				RESTST;
 | |
| 			}
 | |
| 			FREEST;
 | |
| 			if (totalcost+mincost>costlimit)
 | |
| 				BROKE();
 | |
| 		}
 | |
| 	} else {
 | |
| 		decision = forced;
 | |
| 		if (getrefcount(decision)!=0)
 | |
| 			BROKE();
 | |
| 		token2.t_token = -1;
 | |
| 	}
 | |
| 	chrefcount(decision,1,FALSE);
 | |
| 	token2.t_att[0].ar=decision;
 | |
| 	if (token.t_token != 0) {
 | |
| 		totalcost+=move(&token,&token2,ply,toplevel,MAXINT);
 | |
| 		CHKCOST();
 | |
| 	} else
 | |
| 		erasereg(decision);
 | |
| 	allreg[nallreg++]=decision;
 | |
| 	break;
 | |
|     }
 | |
|     case DO_INSTR: {
 | |
| 	register i;
 | |
| 	int n;
 | |
| 	int tinstno;
 | |
| 	token_t token;
 | |
| 	int stringno;
 | |
| 
 | |
|     	DEBUG("INSTR");
 | |
| 	n=((codep[-1]>>5)&07);
 | |
| 	getint(stringno,codep);
 | |
| 	if (toplevel) {
 | |
| 		swtxt();
 | |
| 		if (stringno>10000) {
 | |
| 			assert(stringno== 10001 || stringno== 10002);
 | |
| 			genstr(procarg[stringno-10001]);
 | |
| 		} else
 | |
| 			genstr(stringno);
 | |
| 	}
 | |
| 	for(i=0;i<n;i++) {
 | |
| 		getint(tinstno,codep);
 | |
| 		instance(tinstno,&token);
 | |
| 		if (toplevel)
 | |
| 			prtoken(&token,i==0 ? ' ' : ',');
 | |
| 		if (token.t_token>0)
 | |
| 			totalcost += tokens[token.t_token].t_cost.ct_space;
 | |
| 	}
 | |
| 	if (toplevel)
 | |
| 		gennl();
 | |
| 	break;		
 | |
|     }
 | |
|     case DO_MOVE: {
 | |
| 	int tinstno;
 | |
| 	token_t token,token2;
 | |
| 
 | |
| 	DEBUG("MOVE");
 | |
| 	getint(tinstno,codep);
 | |
| 	instance(tinstno,&token);
 | |
| 	getint(tinstno,codep);
 | |
| 	instance(tinstno,&token2);
 | |
| 	totalcost += move(&token,&token2,ply,toplevel,costlimit-totalcost+1);
 | |
| 	CHKCOST();
 | |
| 	break;
 | |
|     }
 | |
|     case DO_TEST: {
 | |
| 	int tinstno;
 | |
| 	token_t token;
 | |
| 
 | |
| 	DEBUG("TEST");
 | |
| 	getint(tinstno,codep);
 | |
| 	instance(tinstno,&token);
 | |
| 	totalcost += test(&token,ply,toplevel,costlimit-totalcost+1);
 | |
| 	CHKCOST();
 | |
| 	break;
 | |
|     }
 | |
|     case DO_SETCC: {
 | |
| 	int tinstno;
 | |
| 	token_t token;
 | |
| 
 | |
|     	DEBUG("SETCC");
 | |
| 	getint(tinstno,codep);
 | |
| 	instance(tinstno,&token);
 | |
| 	setcc(&token);
 | |
| 	break;
 | |
|     }
 | |
|     case DO_ERASE: {
 | |
| 	int nodeno;
 | |
| 	result_t result;
 | |
| 
 | |
| 	DEBUG("ERASE");
 | |
| 	getint(nodeno,codep);
 | |
| 	result=compute(&enodes[nodeno]);
 | |
| 	assert(result.e_typ!=EV_INT && result.e_typ!=EV_ADDR);
 | |
| 	if (result.e_typ==EV_REG)
 | |
| 		erasereg(result.e_v.e_reg);
 | |
| 	break;
 | |
|     }
 | |
|     case DO_TOKREPLACE: {
 | |
| 	register i;
 | |
| 	int tinstno;
 | |
| 	int repllen;
 | |
| 	token_t reptoken[MAXREPLLEN];
 | |
| 
 | |
| 	DEBUG("TOKREPLACE");
 | |
| 	assert(stackheight>=tokpatlen);
 | |
| 	repllen=(codep[-1]>>5)&07;
 | |
| #ifndef NDEBUG
 | |
| 	if (Debug>2)
 | |
| 		fprintf(stderr,"Stackheight=%d, tokpatlen=%d, repllen=%d %s\n",
 | |
| 			stackheight,tokpatlen,repllen,inscoerc ? "(inscoerc)":"");
 | |
| #endif
 | |
| 	for(i=0;i<repllen;i++) {
 | |
| 		getint(tinstno,codep);
 | |
| 		instance(tinstno,&reptoken[i]);
 | |
| 		tref(&reptoken[i],1);
 | |
| 	}
 | |
| 	for(i=0;i<tokpatlen;i++) {
 | |
| 		if (!inscoerc)
 | |
| 			tref(&fakestack[stackheight-1],-1);
 | |
| 		stackheight--;
 | |
| 	}
 | |
| 	for (i=0;i<repllen;i++) {
 | |
| 		assert(stackheight<MAXFSTACK);
 | |
| 		fakestack[stackheight++] = reptoken[i];
 | |
| 	}
 | |
| 	for(i=0;i<nallreg;i++)
 | |
| 		chrefcount(allreg[i],-1,FALSE);
 | |
| 	break;
 | |
|     }
 | |
|     case DO_EMREPLACE: {
 | |
| 	register i;
 | |
| 	int j;
 | |
| 	int nodeno;
 | |
| 	result_t result;
 | |
| 	int emrepllen,eminstr;
 | |
| 
 | |
| 	DEBUG("EMREPLACE");
 | |
| 	emrepllen=(codep[-1]>>5)&07;
 | |
| 	j=emp-emlines;
 | |
| 	if (emrepllen>j) {
 | |
| 		assert(nemlines+emrepllen-j<MAXEMLINES);
 | |
| 		for (i=nemlines;i>=0;i--)
 | |
| 			emlines[i+emrepllen-j] = emlines[i];
 | |
| 		nemlines += emrepllen-j;
 | |
| 		emp += emrepllen-j;
 | |
| 	}
 | |
| 	emp -= emrepllen;
 | |
| 	for (i=0;i<emrepllen;i++) {
 | |
| 		getint(eminstr,codep);
 | |
| 		getint(nodeno,codep);
 | |
| 		emp[i].em_instr = eminstr;
 | |
| 		result = compute(&enodes[nodeno]);
 | |
| 		switch(result.e_typ) {
 | |
| 		default:
 | |
| 			assert(FALSE);
 | |
| 		case 0:
 | |
| 			emp[i].em_optyp = OPNO;
 | |
| 			emp[i].em_soper = 0;
 | |
| 			break;
 | |
| 		case EV_INT:
 | |
| 			emp[i].em_optyp = OPINT;
 | |
| 			emp[i].em_soper = tostring(result.e_v.e_con);
 | |
| 			emp[i].em_u.em_ioper = result.e_v.e_con;
 | |
| 			break;
 | |
| 		case EV_ADDR:
 | |
| 			emp[i].em_optyp = OPSYMBOL;
 | |
| 			emp[i].em_soper = ad2str(result.e_v.e_addr);
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!toplevel)
 | |
| 		ply += emrepllen;
 | |
| 	break;
 | |
|     }
 | |
|     case DO_COST: {
 | |
| 	cost_t cost;
 | |
| 
 | |
| 	DEBUG("COST");
 | |
| 	getint(cost.ct_space,codep);
 | |
| 	getint(cost.ct_time,codep);
 | |
| 	totalcost += costcalc(cost);
 | |
| 	CHKCOST();
 | |
| 	break;
 | |
|     }
 | |
| #ifdef REGVARS
 | |
|     case DO_PRETURN: {
 | |
| 	if (toplevel) {
 | |
| 		swtxt();
 | |
| 		regreturn();	/* in mach.c */
 | |
| 	}
 | |
| 	break;
 | |
|     }
 | |
| #endif
 | |
|     case DO_RETURN:
 | |
| 	DEBUG("RETURN");
 | |
| 	assert(origcp!=startupcode);
 | |
|     doreturn:
 | |
| #ifndef NDEBUG
 | |
| 	level--;
 | |
| #endif
 | |
| 	return(totalcost);
 | |
| 	}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| readcodebytes() {
 | |
| #ifndef CODEINC
 | |
| 	register fd;
 | |
| 	extern int ncodebytes;
 | |
| 
 | |
| 	if ((fd=open("code",0))<0) {
 | |
| 		error("Can't open code");
 | |
| 	}
 | |
| 	if (read(fd,coderules,ncodebytes)!=ncodebytes) {
 | |
| 		error("Short read from code");
 | |
| 	}
 | |
| 	close(fd);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #ifdef TABLEDEBUG
 | |
| initlset(f) char *f; {
 | |
| 	extern char *myalloc();
 | |
| 
 | |
| 	set_flag = f;
 | |
| 	if ((set_fd=open(f+1,2))<0)
 | |
| 		error("Can't open %s rw",f+1);
 | |
| 	read(set_fd,&set_size,sizeof(int));
 | |
| 	set_val=( short *) myalloc(set_size);
 | |
| 	read(set_fd,set_val,set_size);
 | |
| }
 | |
| 
 | |
| termlset() {
 | |
| 
 | |
| 	if (set_fd) {
 | |
| 		lseek(set_fd,(long) sizeof(int),0);
 | |
| 		write(set_fd,set_val,set_size);
 | |
| 		close(set_fd);
 | |
| 		if (set_flag[0]=='u') {
 | |
| 			register i;
 | |
| 			
 | |
| 			fprintf(stderr,"Unused code rules:\n\n");
 | |
| 			for(i=0;i<8*set_size;i++)
 | |
| 				if(set_val[i>>4]&(1<<(i&017)))
 | |
| 					fprintf(stderr,"\"%s\", line %d\n",tablename,i);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| #endif
 |