661 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			661 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef NORCSID
 | |
| static char rcsid[] = "$Header$";
 | |
| #endif
 | |
| 
 | |
| #include "assert.h"
 | |
| #include "param.h"
 | |
| #include "tables.h"
 | |
| #include "types.h"
 | |
| #include <cg_pattern.h>
 | |
| #include "data.h"
 | |
| #include "result.h"
 | |
| #include "state.h"
 | |
| #include "equiv.h"
 | |
| #include "extern.h"
 | |
| 
 | |
| /*
 | |
|  * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | |
|  * See the copyright notice in the ACK home directory, in the file "Copyright".
 | |
|  *
 | |
|  * Author: Hans van Staveren
 | |
|  */
 | |
| 
 | |
| #define SHORTCUT	/* Stop searching at distance 0 */
 | |
| 
 | |
| #if NREGS >= MAXRULE
 | |
| #define MAXPOS NREGS
 | |
| #else
 | |
| #define MAXPOS MAXRULE
 | |
| #endif
 | |
| 
 | |
| #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();
 | |
| 
 | |
| #ifdef NDEBUG
 | |
| #define DEBUG(xxxxx)
 | |
| #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");goto doreturn;}
 | |
| #define CHKCOST() {if (totalcost>=costlimit) BROKE();}
 | |
| 
 | |
| unsigned codegen(codep,ply,toplevel,costlimit,forced) byte *codep; unsigned costlimit; {
 | |
| #ifndef NDEBUG
 | |
| 	byte *origcp=codep;
 | |
| 	static int level=0;
 | |
| #endif
 | |
| 	unsigned totalcost = 0;
 | |
| 	byte *bp;
 | |
| 	int n;
 | |
| 	unsigned mindistance,dist;
 | |
| 	register i;
 | |
| 	int cindex;
 | |
| 	int npos,npos2,pos[MAXPOS],pos2[MAXPOS];
 | |
| #ifdef STONSTACK
 | |
| 	state_t state;
 | |
| #define SAVEST	savestatus(&state)
 | |
| #define RESTST	restorestatus(&state)
 | |
| #define FREEST	/* nothing */
 | |
| #else
 | |
| 	state_p state;
 | |
| #define SAVEST	state=savestatus()
 | |
| #define RESTST	restorestatus(state)
 | |
| #define FREEST	freestatus(state)
 | |
| #endif
 | |
| 	unsigned mincost,t;
 | |
| 	int texpno,nodeno;
 | |
| 	token_p tp;
 | |
| 	tkdef_p tdp;
 | |
| 	int tinstno;
 | |
| 	struct reginfo *rp,**rpp;
 | |
| 	token_t token,mtoken,token2;
 | |
| 	int propno;
 | |
| 	int exactmatch;
 | |
| 	int j;
 | |
| 	int decision;
 | |
| 	int stringno;
 | |
| 	result_t result;
 | |
| 	cost_t cost;
 | |
| 	int size,lsize,repllen;
 | |
| 	int tokexp[MAXPATTERN];
 | |
| 	int nregneeded;
 | |
| 	token_p regtp[MAXCREG];
 | |
| 	c3_p regcp[MAXCREG];
 | |
| 	rl_p regls[MAXCREG];
 | |
| 	c3_p cp,findcoerc();
 | |
| 	int sret;
 | |
| 	token_t reptoken[MAXREPLLEN];
 | |
| 	int emrepllen,eminstr;
 | |
| 	int inscoerc=0;
 | |
| 	int stackpad;
 | |
| 	struct perm *tup,*ntup,*besttup,*tuples();
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| 	level++;
 | |
| 	DEBUG("Entering codegen");
 | |
| #endif
 | |
| 	for (;;) {
 | |
| 	switch( (*codep++)&037 ) {
 | |
|     default:
 | |
| 	assert(FALSE);
 | |
| 	/* NOTREACHED */
 | |
|     case DO_NEXTEM:
 | |
| 	DEBUG("NEXTEM");
 | |
| 	tokpatlen = 0;
 | |
| 	nallreg=0;
 | |
| 	if (toplevel) {
 | |
| 		garbage_collect();
 | |
| 		totalcost=0;
 | |
| 	} else {
 | |
| 		if (--ply <= 0)
 | |
| 			goto doreturn;
 | |
| 	}
 | |
| 	if (stackheight>MAXFSTACK-7)	
 | |
| 		totalcost += stackupto(&fakestack[6],ply,toplevel);
 | |
| 	bp = nextem(toplevel);
 | |
| 	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++;
 | |
| 		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) {
 | |
| #ifdef SHORTCUT
 | |
| 						if(dist==0)
 | |
| 							goto gotit;
 | |
| #endif
 | |
| 						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) {
 | |
| 					totalcost += mincost;
 | |
| 					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:
 | |
| 	DEBUG("XMATCH");
 | |
| 	tokpatlen=(codep[-1]>>5)&07;
 | |
| 	for (i=0;i<tokpatlen;i++)
 | |
| 		getint(tokexp[i],codep);
 | |
| 	tokexp[i]=0;
 | |
| 	break;	/* match already checked by distance() */
 | |
|     case DO_MATCH:
 | |
| 	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]]);
 | |
| 			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 {
 | |
| 					assert(nregneeded<MAXCREG);
 | |
| 					regtp[nregneeded] = tp;
 | |
| 					regcp[nregneeded] = cp;
 | |
| 					regls[nregneeded] = curreglist;
 | |
| 					nregneeded++;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		i++; tp--;
 | |
| 	}
 | |
| 	if (tokpatlen>stackheight) {
 | |
| 		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;
 | |
| 		tp = &fakestack[stackpad-1];
 | |
| 		while (i<tokpatlen && tp>=fakestack) {
 | |
| 			cp = findcoerc((token_p) 0, &machsets[tokexp[i]]);
 | |
| 			if (cp==0) {
 | |
| 				assert(!toplevel);
 | |
| 				for (j=0;j<nregneeded;j++)
 | |
| 					myfree(regls[j]);
 | |
| 				totalcost=INFINITY;
 | |
| 				BROKE();
 | |
| 			}
 | |
| 			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++;
 | |
| 			}
 | |
| 			i++; tp--;
 | |
| 		}
 | |
| 	} 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) {
 | |
| 		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)
 | |
| 			t += codegen(codep,ply,FALSE,mincost-t,0);
 | |
| 		if (t<mincost) {
 | |
| 			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);
 | |
| 		if (stackpad!=tokpatlen) {
 | |
| 			if (stackpad) {
 | |
| 				if (costlimit<MAXINT) {
 | |
| 					totalcost = costlimit+1;
 | |
| 					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;
 | |
| 		BROKE();
 | |
| 	}
 | |
| 	for (i=0;i<nregneeded;i++)
 | |
| 		totalcost += docoerc(regtp[i],regcp[i],ply,toplevel,besttup->p_rar[i]);
 | |
| 	myfree(besttup);
 | |
| 	break;
 | |
|     case DO_REMOVE:
 | |
| 	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+2;rp<machregs+NREGS;rp++)
 | |
| 		if (match(&rp->r_contents,&machsets[texpno],nodeno))
 | |
| 			rp->r_contents.t_token=0;
 | |
| 	break;
 | |
|     case DO_RREMOVE:	/* register remove */
 | |
| 	getint(nodeno,codep);
 | |
| 	result=compute(&enodes[nodeno]);
 | |
| 	assert(result.e_typ==EV_REG);
 | |
| 	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:
 | |
| 	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:
 | |
| 	DEBUG("REALLOCATE");
 | |
| 	for(rp=machregs;rp<machregs+NREGS;rp++)
 | |
| 		if(rp->r_tcount) {
 | |
| 			rp->r_refcount -= rp->r_tcount;
 | |
| 			rp->r_tcount = 0;
 | |
| 		}
 | |
| 	break;
 | |
|     case DO_ALLOCATE:
 | |
| 	DEBUG("ALLOCATE");
 | |
| 	if (codep[-1]&32) {
 | |
| 		getint(propno,codep);
 | |
| 		getint(tinstno,codep);
 | |
| 	} else {
 | |
| 		getint(propno,codep);
 | |
| 		tinstno=0;
 | |
| 	}
 | |
| 	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) {
 | |
| 						totalcost = INFINITY;
 | |
| 						BROKE();
 | |
| 					} else {
 | |
| 						fatal("No regs available");
 | |
| 					}
 | |
| 				}
 | |
| 				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) {
 | |
| 				totalcost = INFINITY;
 | |
| 				BROKE();
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		decision = forced;
 | |
| 		if (getrefcount(decision)!=0) {
 | |
| 			totalcost = INFINITY;
 | |
| 			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_LOUTPUT:
 | |
| 	DEBUG("LOUTPUT");
 | |
| 	getint(stringno,codep);
 | |
| 	getint(nodeno,codep);
 | |
| 	if (toplevel) {
 | |
| 		gencode(codestrings[stringno]);
 | |
| 		genexpr(nodeno);
 | |
| 	}
 | |
| 	break;
 | |
|     case DO_ROUTPUT:
 | |
| 	DEBUG("ROUTPUT");
 | |
| 	i=((codep[-1]>>5)&07);
 | |
| 	do {
 | |
| 		getint(stringno,codep);
 | |
| 		if (toplevel) {
 | |
| 			gencode(codestrings[stringno]);
 | |
| 			gennl();
 | |
| 		}
 | |
| 	} while (i--);
 | |
| 	break;
 | |
|     case DO_MOVE:
 | |
| 	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_ERASE:
 | |
| 	DEBUG("ERASE");
 | |
| 	getint(nodeno,codep);
 | |
| 	result=compute(&enodes[nodeno]);
 | |
| 	assert(result.e_typ==EV_REG);
 | |
| 	erasereg(result.e_v.e_reg);
 | |
| 	break;
 | |
|     case DO_TOKREPLACE:
 | |
| 	DEBUG("TOKREPLACE");
 | |
| 	assert(stackheight>=tokpatlen);
 | |
| 	repllen=(codep[-1]>>5)&07;
 | |
| 	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:
 | |
| 	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_STR:
 | |
| 			emp[i].em_optyp = OPSYMBOL;
 | |
| 			emp[i].em_soper = result.e_v.e_str;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!toplevel)
 | |
| 		ply += emrepllen;
 | |
| 	break;
 | |
|     case DO_COST:
 | |
| 	DEBUG("COST");
 | |
| 	getint(cost.c_size,codep);
 | |
| 	getint(cost.c_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);
 | |
| 	}
 | |
| 	}
 | |
| }
 |