fd7e9f9046
--HG-- branch : default-branch
823 lines
18 KiB
C
823 lines
18 KiB
C
#ifndef NORCSID
|
|
static char rcsid[] = "$Id$";
|
|
#endif
|
|
|
|
#include "param.h"
|
|
#include "types.h"
|
|
#include "tes.h"
|
|
#include "assert.h"
|
|
#include "line.h"
|
|
#include "lookup.h"
|
|
#include "proinf.h"
|
|
#include "alloc.h"
|
|
#include "pattern.h"
|
|
#include <em_spec.h>
|
|
#include <em_mnem.h>
|
|
#include "optim.h"
|
|
#include "ext.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
|
|
*/
|
|
|
|
#undef CHK_HASH /* print numbers patterns are hashed to */
|
|
#ifdef CHK_HASH
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#define ILLHASH 0177777
|
|
short pathash[256]; /* table of indices into pattern[] */
|
|
|
|
int opind = 0; /* second index of next matrix */
|
|
byte transl[op_plast-op_pfirst+1][3] = {
|
|
/* LLP */ { op_LLP, op_lol, op_ldl },
|
|
/* LEP */ { op_LEP, op_loe, op_lde },
|
|
/* SLP */ { op_SLP, op_stl, op_sdl },
|
|
/* SEP */ { op_SEP, op_ste, op_sde }
|
|
};
|
|
|
|
opcheck(bp) register byte *bp; {
|
|
|
|
if (((*bp)&BMASK) >= op_pfirst)
|
|
*bp = transl[((*bp)&BMASK)-op_pfirst][opind];
|
|
}
|
|
|
|
/*
|
|
* The hashing method used is believed to be reasonably efficient.
|
|
* A minor speed improvement could be obtained by keeping a boolean
|
|
* array telling which opcode has any patterns starting with it.
|
|
* Currently only about one third of the opcodes actually have a
|
|
* pattern starting with it, but they are the most common ones.
|
|
* Estimated improvement possible: about 2%
|
|
*/
|
|
|
|
hashpatterns() {
|
|
short index;
|
|
register byte *bp,*tp;
|
|
register short i;
|
|
unsigned short hashvalue;
|
|
byte *save;
|
|
int patlen;
|
|
|
|
if (pointersize == wordsize)
|
|
opind=1;
|
|
else if (pointersize == 2*wordsize)
|
|
opind=2;
|
|
index = lastind; /* set by mktab */
|
|
while (index != 0) {
|
|
bp = &pattern[index];
|
|
tp = &bp[PO_MATCH];
|
|
i = *tp++&BMASK;
|
|
if (i==BMASK) {
|
|
i = *tp++&BMASK;
|
|
i |= (*tp++&BMASK)<<8;
|
|
}
|
|
save = tp;
|
|
patlen = i;
|
|
while (i--)
|
|
opcheck(tp++);
|
|
if ((*tp++&BMASK)==BMASK)
|
|
tp += 2;
|
|
i = *tp++&BMASK;
|
|
if (i==BMASK) {
|
|
i = *tp++&BMASK;
|
|
i |= (*tp++&BMASK)<<8;
|
|
}
|
|
while (i--) {
|
|
opcheck(tp++);
|
|
if ((*tp++&BMASK)==BMASK)
|
|
tp += 2;
|
|
}
|
|
|
|
/*
|
|
* Now the special opcodes are filled
|
|
* in properly, we can hash the pattern
|
|
*/
|
|
|
|
hashvalue = 0;
|
|
tp = save;
|
|
switch(patlen) {
|
|
default: /* 3 or more */
|
|
hashvalue = (hashvalue<<4)^(*tp++&BMASK);
|
|
case 2:
|
|
hashvalue = (hashvalue<<4)^(*tp++&BMASK);
|
|
case 1:
|
|
hashvalue = (hashvalue<<4)^(*tp++&BMASK);
|
|
}
|
|
assert(hashvalue!= ILLHASH);
|
|
i=index;
|
|
index = (bp[PO_NEXT]&BMASK)|(bp[PO_NEXT+1]<<8);
|
|
bp[PO_HASH] = hashvalue>>8;
|
|
hashvalue &= BMASK;
|
|
bp[PO_NEXT] = pathash[hashvalue]&BMASK;
|
|
bp[PO_NEXT+1] = pathash[hashvalue]>>8;
|
|
pathash[hashvalue] = i;
|
|
#ifdef CHK_HASH
|
|
fprintf(stderr,"%d\n",hashvalue);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
peephole() {
|
|
static bool phashed = FALSE;
|
|
|
|
if (!phashed) {
|
|
hashpatterns();
|
|
phashed=TRUE;
|
|
}
|
|
return optimize();
|
|
}
|
|
|
|
optimize() {
|
|
register num_p *npp,np;
|
|
register instr;
|
|
bool madeopt;
|
|
|
|
madeopt = basicblock(&instrs);
|
|
for (npp=curpro.numhash;npp< &curpro.numhash[NNUMHASH]; npp++)
|
|
for (np = *npp; np != (num_p) 0; np=np->n_next) {
|
|
if (! np->n_line) continue;
|
|
if(np->n_line->l_next == (line_p) 0)
|
|
continue;
|
|
instr = np->n_line->l_next->l_instr&BMASK;
|
|
if (instr == op_lab || instr == op_bra)
|
|
np->n_repl = np->n_line->l_next->l_a.la_np;
|
|
else
|
|
if (basicblock(&np->n_line->l_next))
|
|
madeopt = TRUE;
|
|
}
|
|
return madeopt;
|
|
}
|
|
|
|
offset oabs(off) offset off; {
|
|
|
|
return(off >= 0 ? off : -off);
|
|
}
|
|
|
|
line_p repline(ev,patlen) eval_t ev; {
|
|
register line_p lp;
|
|
register iarg_p iap;
|
|
register sym_p sp;
|
|
offset diff,newdiff;
|
|
|
|
assert(ev.e_typ != EV_UNDEF);
|
|
switch(ev.e_typ) {
|
|
case EV_CONST:
|
|
if ((short) ev.e_v.e_con == ev.e_v.e_con) {
|
|
if (CANMINI((short) ev.e_v.e_con))
|
|
lp = newline((short) (ev.e_v.e_con)+Z_OPMINI);
|
|
else {
|
|
lp = newline(OPSHORT);
|
|
lp->l_a.la_short = (short) ev.e_v.e_con;
|
|
}
|
|
} else {
|
|
lp = newline(OPOFFSET);
|
|
lp->l_a.la_offset = ev.e_v.e_con;
|
|
}
|
|
return(lp);
|
|
case EV_NUMLAB:
|
|
lp = newline(OPNUMLAB);
|
|
lp->l_a.la_np = ev.e_v.e_np;
|
|
return(lp);
|
|
default: /* fragment + offset */
|
|
/*
|
|
* There is a slight problem here, because we have to
|
|
* map fragment+offset to symbol+offset.
|
|
* Fortunately the fragment we have must be the fragment
|
|
* of one of the symbols in the matchpattern.
|
|
* So a short search should do the job.
|
|
*/
|
|
sp = (sym_p) 0;
|
|
for (iap= &iargs[patlen-1]; iap >= iargs; iap--)
|
|
if (iap->ia_ev.e_typ == ev.e_typ) {
|
|
/*
|
|
* Although lint complains, diff is not used
|
|
* before set.
|
|
*
|
|
* The proof is left as an exercise to the
|
|
* reader.
|
|
*/
|
|
newdiff = oabs(iap->ia_sp->s_value-ev.e_v.e_con);
|
|
if (sp==(sym_p) 0 || newdiff < diff) {
|
|
sp = iap->ia_sp;
|
|
diff = newdiff;
|
|
}
|
|
}
|
|
assert(sp != (sym_p) 0);
|
|
if (diff == 0) {
|
|
lp = newline(OPSYMBOL);
|
|
lp->l_a.la_sp = sp;
|
|
} else {
|
|
diff = ev.e_v.e_con - sp->s_value;
|
|
if ((short) diff == diff) {
|
|
lp = newline(OPSVAL);
|
|
lp->l_a.la_sval.lasv_short = (short) diff;
|
|
lp->l_a.la_sval.lasv_sp = sp;
|
|
} else {
|
|
lp = newline(OPLVAL);
|
|
lp->l_a.la_lval.lalv_offset = diff;
|
|
lp->l_a.la_lval.lalv_sp = sp;
|
|
}
|
|
}
|
|
return(lp);
|
|
}
|
|
}
|
|
|
|
offset rotate(w,amount) offset w,amount; {
|
|
offset highmask,lowmask;
|
|
|
|
#ifndef LONGOFF
|
|
assert(wordsize<=4);
|
|
#endif
|
|
highmask = (offset)(-1) << amount;
|
|
lowmask = ~highmask;
|
|
if (wordsize != 4)
|
|
highmask &= wordsize==2 ? 0xFFFF : 0xFF;
|
|
return(((w<<amount)&highmask)|((w>>(8*wordsize-amount))&lowmask));
|
|
}
|
|
|
|
eval_t undefres = { EV_UNDEF };
|
|
|
|
eval_t compute(pexp) register expr_p pexp; {
|
|
eval_t leaf1,leaf2,res;
|
|
register i;
|
|
register sym_p sp;
|
|
offset mask;
|
|
|
|
switch(nparam[pexp->ex_operator]) {
|
|
default:
|
|
assert(FALSE);
|
|
case 2:
|
|
leaf2 = compute(&enodes[pexp->ex_rnode]);
|
|
if (leaf2.e_typ == EV_UNDEF ||
|
|
nonumlab[pexp->ex_operator] && leaf2.e_typ == EV_NUMLAB ||
|
|
onlyconst[pexp->ex_operator] && leaf2.e_typ != EV_CONST)
|
|
return(undefres);
|
|
case 1:
|
|
leaf1 = compute(&enodes[pexp->ex_lnode]);
|
|
if (leaf1.e_typ == EV_UNDEF ||
|
|
nonumlab[pexp->ex_operator] && leaf1.e_typ == EV_NUMLAB ||
|
|
onlyconst[pexp->ex_operator] && leaf1.e_typ != EV_CONST)
|
|
return(undefres);
|
|
case 0:
|
|
break;
|
|
}
|
|
|
|
res.e_typ = EV_CONST;
|
|
res.e_v.e_con = 0;
|
|
switch(pexp->ex_operator) {
|
|
default:
|
|
assert(FALSE);
|
|
case EX_CON:
|
|
res.e_v.e_con = (offset) pexp->ex_lnode;
|
|
break;
|
|
case EX_ARG:
|
|
return(iargs[pexp->ex_lnode - 1].ia_ev);
|
|
case EX_CMPEQ:
|
|
if (leaf1.e_typ != leaf2.e_typ)
|
|
return(undefres);
|
|
if (leaf1.e_typ == EV_NUMLAB) {
|
|
if (leaf1.e_v.e_np == leaf2.e_v.e_np)
|
|
res.e_v.e_con = 1;
|
|
break;
|
|
}
|
|
if (leaf1.e_v.e_con == leaf2.e_v.e_con)
|
|
res.e_v.e_con = 1;
|
|
break;
|
|
case EX_CMPNE:
|
|
if (leaf1.e_typ != leaf2.e_typ) {
|
|
res.e_v.e_con = 1;
|
|
break;
|
|
}
|
|
if (leaf1.e_typ == EV_NUMLAB) {
|
|
if (leaf1.e_v.e_np != leaf2.e_v.e_np)
|
|
res.e_v.e_con = 1;
|
|
break;
|
|
}
|
|
if (leaf1.e_v.e_con != leaf2.e_v.e_con)
|
|
res.e_v.e_con = 1;
|
|
break;
|
|
case EX_CMPGT:
|
|
if (leaf1.e_typ != leaf2.e_typ)
|
|
return(undefres);
|
|
res.e_v.e_con = leaf1.e_v.e_con > leaf2.e_v.e_con;
|
|
break;
|
|
case EX_CMPGE:
|
|
if (leaf1.e_typ != leaf2.e_typ)
|
|
return(undefres);
|
|
res.e_v.e_con = leaf1.e_v.e_con >= leaf2.e_v.e_con;
|
|
break;
|
|
case EX_CMPLT:
|
|
if (leaf1.e_typ != leaf2.e_typ)
|
|
return(undefres);
|
|
res.e_v.e_con = leaf1.e_v.e_con < leaf2.e_v.e_con;
|
|
break;
|
|
case EX_CMPLE:
|
|
if (leaf1.e_typ != leaf2.e_typ)
|
|
return(undefres);
|
|
res.e_v.e_con = leaf1.e_v.e_con <= leaf2.e_v.e_con;
|
|
break;
|
|
case EX_OR2:
|
|
if (leaf1.e_v.e_con != 0)
|
|
return(leaf1);
|
|
leaf2 = compute(&enodes[pexp->ex_rnode]);
|
|
if (leaf2.e_typ != EV_CONST)
|
|
return(undefres);
|
|
return(leaf2);
|
|
case EX_AND2:
|
|
if (leaf1.e_v.e_con == 0)
|
|
return(leaf1);
|
|
leaf2 = compute(&enodes[pexp->ex_rnode]);
|
|
if (leaf2.e_typ != EV_CONST)
|
|
return(undefres);
|
|
return(leaf2);
|
|
case EX_OR1:
|
|
res.e_v.e_con = leaf1.e_v.e_con | leaf2.e_v.e_con;
|
|
break;
|
|
case EX_XOR1:
|
|
res.e_v.e_con = leaf1.e_v.e_con ^ leaf2.e_v.e_con;
|
|
break;
|
|
case EX_AND1:
|
|
res.e_v.e_con = leaf1.e_v.e_con & leaf2.e_v.e_con;
|
|
break;
|
|
case EX_TIMES:
|
|
res.e_v.e_con = leaf1.e_v.e_con * leaf2.e_v.e_con;
|
|
break;
|
|
case EX_DIVIDE:
|
|
res.e_v.e_con = leaf1.e_v.e_con / leaf2.e_v.e_con;
|
|
break;
|
|
case EX_MOD:
|
|
res.e_v.e_con = leaf1.e_v.e_con % leaf2.e_v.e_con;
|
|
break;
|
|
case EX_LSHIFT:
|
|
res.e_v.e_con = leaf1.e_v.e_con << leaf2.e_v.e_con;
|
|
break;
|
|
case EX_RSHIFT:
|
|
res.e_v.e_con = leaf1.e_v.e_con >> leaf2.e_v.e_con;
|
|
break;
|
|
case EX_UMINUS:
|
|
res.e_v.e_con = -leaf1.e_v.e_con;
|
|
break;
|
|
case EX_NOT:
|
|
res.e_v.e_con = !leaf1.e_v.e_con;
|
|
break;
|
|
case EX_COMP:
|
|
res.e_v.e_con = ~leaf1.e_v.e_con;
|
|
break;
|
|
case EX_PLUS:
|
|
if (leaf1.e_typ >= EV_FRAG) {
|
|
if (leaf2.e_typ >= EV_FRAG)
|
|
return(undefres);
|
|
res.e_typ = leaf1.e_typ;
|
|
} else
|
|
res.e_typ = leaf2.e_typ;
|
|
res.e_v.e_con = leaf1.e_v.e_con + leaf2.e_v.e_con;
|
|
break;
|
|
case EX_MINUS:
|
|
if (leaf1.e_typ >= EV_FRAG) {
|
|
if (leaf2.e_typ == EV_CONST)
|
|
res.e_typ = leaf1.e_typ;
|
|
else if (leaf2.e_typ != leaf1.e_typ)
|
|
return(undefres);
|
|
} else if (leaf2.e_typ >= EV_FRAG)
|
|
return(undefres);
|
|
res.e_v.e_con = leaf1.e_v.e_con - leaf2.e_v.e_con;
|
|
break;
|
|
case EX_POINTERSIZE:
|
|
res.e_v.e_con = pointersize;
|
|
break;
|
|
case EX_WORDSIZE:
|
|
res.e_v.e_con = wordsize;
|
|
break;
|
|
case EX_NOTREG:
|
|
res.e_v.e_con = !inreg(leaf1.e_v.e_con);
|
|
break;
|
|
case EX_DEFINED:
|
|
leaf1 = compute(&enodes[pexp->ex_lnode]);
|
|
res.e_v.e_con = leaf1.e_typ != EV_UNDEF;
|
|
break;
|
|
case EX_SAMESIGN:
|
|
res.e_v.e_con = (leaf1.e_v.e_con ^ leaf2.e_v.e_con) >= 0;
|
|
break;
|
|
case EX_ROM:
|
|
if ((sp = iargs[pexp->ex_lnode - 1].ia_sp) != (sym_p) 0 &&
|
|
sp->s_rom != (offset *) 0) {
|
|
leaf2 = compute(&enodes[pexp->ex_rnode]);
|
|
if (leaf2.e_typ != EV_CONST ||
|
|
leaf2.e_v.e_con < 0 ||
|
|
leaf2.e_v.e_con >= MAXROM)
|
|
return(undefres);
|
|
res.e_v.e_con = sp->s_rom[(int)(leaf2.e_v.e_con)];
|
|
break;
|
|
} else
|
|
return(undefres);
|
|
case EX_SFIT:
|
|
mask = 0;
|
|
for (i=leaf2.e_v.e_con - 1;i < 8*sizeof(offset); i++)
|
|
mask |= ((offset)1)<<i;
|
|
res.e_v.e_con = (leaf1.e_v.e_con&mask) == 0 ||
|
|
(leaf1.e_v.e_con&mask) == mask;
|
|
break;
|
|
case EX_UFIT:
|
|
mask = 0;
|
|
for (i=leaf2.e_v.e_con;i < 8*sizeof(offset); i++)
|
|
mask |= ((offset)1)<<i;
|
|
res.e_v.e_con = (leaf1.e_v.e_con&mask) == 0;
|
|
break;
|
|
case EX_ROTATE:
|
|
res.e_v.e_con = rotate(leaf1.e_v.e_con,leaf2.e_v.e_con);
|
|
break;
|
|
}
|
|
return(res);
|
|
}
|
|
|
|
#ifdef ALLOWSPECIAL
|
|
extern bool special();
|
|
#endif
|
|
|
|
bool tryrepl(lpp,bp,patlen)
|
|
line_p *lpp;
|
|
register byte *bp;
|
|
int patlen;
|
|
{
|
|
int rpllen,instr,rplval;
|
|
register line_p lp;
|
|
line_p replacement,*rlpp,tp;
|
|
|
|
rpllen = *bp++&BMASK;
|
|
if (rpllen == BMASK) {
|
|
rpllen = *bp++&BMASK;
|
|
rpllen |= (*bp++&BMASK)<<8;
|
|
}
|
|
#ifdef ALLOWSPECIAL
|
|
if (rpllen == 1 && *bp == 0)
|
|
return(special(lpp,bp+1,patlen));
|
|
#endif
|
|
replacement = (line_p) 0;
|
|
rlpp = &replacement;
|
|
while (rpllen--) {
|
|
instr = *bp++&BMASK;
|
|
rplval = *bp++&BMASK;
|
|
if (rplval == BMASK) {
|
|
rplval = (*bp++&BMASK);
|
|
rplval |= (*bp++&BMASK)<<8;
|
|
}
|
|
if (rplval)
|
|
lp = repline(compute(&enodes[rplval]),patlen);
|
|
else
|
|
lp = newline(OPNO);
|
|
|
|
/*
|
|
* One replacement instruction is generated,
|
|
* link in list and proceed with the next one.
|
|
*/
|
|
|
|
if (instr == op_lab)
|
|
lp->l_a.la_np->n_line = lp;
|
|
*rlpp = lp;
|
|
rlpp = &lp->l_next;
|
|
lp->l_instr = instr;
|
|
}
|
|
|
|
/*
|
|
* Replace instructions matched by the created replacement
|
|
*/
|
|
|
|
|
|
OPTIM((bp[0]&BMASK)|(bp[1]&BMASK)<<8);
|
|
for (lp= *lpp;patlen>0;patlen--,tp=lp,lp=lp->l_next)
|
|
;
|
|
tp->l_next = (line_p) 0;
|
|
*rlpp = lp;
|
|
lp = *lpp;
|
|
*lpp = replacement;
|
|
while ( lp != (line_p) 0 ) {
|
|
tp = lp->l_next;
|
|
oldline(lp);
|
|
lp = tp;
|
|
}
|
|
return(TRUE);
|
|
}
|
|
|
|
bool trypat(lpp,bp,len)
|
|
line_p *lpp;
|
|
register byte *bp;
|
|
int len;
|
|
{
|
|
register iarg_p iap;
|
|
int i,patlen;
|
|
register line_p lp;
|
|
eval_t result;
|
|
|
|
patlen = *bp++&BMASK;
|
|
if (patlen == BMASK) {
|
|
patlen = *bp++&BMASK;
|
|
patlen |= (*bp++&BMASK)<<8;
|
|
}
|
|
if (len == 3) {
|
|
if (patlen<3)
|
|
return(FALSE);
|
|
} else {
|
|
if (patlen != len)
|
|
return(FALSE);
|
|
}
|
|
|
|
/*
|
|
* Length is ok, now check opcodes
|
|
*/
|
|
|
|
for (i=0,lp= *lpp;i<patlen && lp != (line_p) 0;i++,lp=lp->l_next)
|
|
if (lp->l_instr != *bp++)
|
|
return(FALSE);
|
|
if (i != patlen)
|
|
return(FALSE);
|
|
|
|
/*
|
|
* opcodes are also correct, now comes the hard part
|
|
*/
|
|
|
|
for(i=0,lp= *lpp,iap= iargs; i<patlen;i++,iap++,lp=lp->l_next) {
|
|
switch(lp->l_optyp) {
|
|
case OPNO:
|
|
iap->ia_ev.e_typ = EV_UNDEF;
|
|
break;
|
|
default:
|
|
iap->ia_ev.e_typ = EV_CONST;
|
|
iap->ia_ev.e_v.e_con = (lp->l_optyp&BMASK)-Z_OPMINI;
|
|
break;
|
|
case OPSHORT:
|
|
iap->ia_ev.e_typ = EV_CONST;
|
|
iap->ia_ev.e_v.e_con = lp->l_a.la_short;
|
|
break;
|
|
#ifdef LONGOFF
|
|
case OPOFFSET:
|
|
iap->ia_ev.e_typ = EV_CONST;
|
|
iap->ia_ev.e_v.e_con = lp->l_a.la_offset;
|
|
break;
|
|
#endif
|
|
case OPNUMLAB:
|
|
iap->ia_ev.e_typ = EV_NUMLAB;
|
|
iap->ia_ev.e_v.e_np = lp->l_a.la_np;
|
|
break;
|
|
case OPSYMBOL:
|
|
iap->ia_ev.e_typ = lp->l_a.la_sp->s_frag;
|
|
iap->ia_sp = lp->l_a.la_sp;
|
|
iap->ia_ev.e_v.e_con = lp->l_a.la_sp->s_value;
|
|
break;
|
|
case OPSVAL:
|
|
iap->ia_ev.e_typ = lp->l_a.la_sval.lasv_sp->s_frag;
|
|
iap->ia_sp = lp->l_a.la_sval.lasv_sp;
|
|
iap->ia_ev.e_v.e_con = lp->l_a.la_sval.lasv_sp->s_value + lp->l_a.la_sval.lasv_short;
|
|
break;
|
|
#ifdef LONGOFF
|
|
case OPLVAL:
|
|
iap->ia_ev.e_typ = lp->l_a.la_lval.lalv_sp->s_frag;
|
|
iap->ia_sp = lp->l_a.la_lval.lalv_sp;
|
|
iap->ia_ev.e_v.e_con = lp->l_a.la_lval.lalv_sp->s_value + lp->l_a.la_lval.lalv_offset;
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
i = *bp++&BMASK;
|
|
if ( i==BMASK ) {
|
|
i = *bp++&BMASK;
|
|
i |= (*bp++&BMASK)<<8;
|
|
}
|
|
if ( i != 0) {
|
|
/* there is a condition */
|
|
result = compute(&enodes[i]);
|
|
if (result.e_typ != EV_CONST || result.e_v.e_con == 0)
|
|
return(FALSE);
|
|
}
|
|
return(tryrepl(lpp,bp,patlen));
|
|
}
|
|
|
|
int
|
|
basicblock(alpp) line_p *alpp; {
|
|
register line_p *lpp,lp;
|
|
unsigned short hash[3];
|
|
line_p *next;
|
|
register byte *bp;
|
|
int i;
|
|
short index;
|
|
bool madeopt;
|
|
int count = 0;
|
|
|
|
lpp = alpp; madeopt = FALSE;
|
|
while ((*lpp) != (line_p) 0 && ((*lpp)->l_instr&BMASK) != op_lab) {
|
|
lp = *lpp;
|
|
next = &lp->l_next;
|
|
hash[0] = lp->l_instr&BMASK;
|
|
lp=lp->l_next;
|
|
if (lp != (line_p) 0) {
|
|
hash[1] = (hash[0]<<4)^(lp->l_instr&BMASK);
|
|
lp=lp->l_next;
|
|
if (lp != (line_p) 0)
|
|
hash[2] = (hash[1]<<4)^(lp->l_instr&BMASK);
|
|
else
|
|
hash[2] = ILLHASH;
|
|
} else {
|
|
hash[1] = ILLHASH;
|
|
hash[2] = ILLHASH;
|
|
}
|
|
|
|
/*
|
|
* hashvalues computed. Try for longest pattern first
|
|
*/
|
|
|
|
for (i=2;i>=0;i--) {
|
|
index = pathash[hash[i]&BMASK];
|
|
while (index != 0) {
|
|
bp = &pattern[index];
|
|
if((bp[PO_HASH]&BMASK) == (hash[i]>>8))
|
|
if(trypat(lpp,&bp[PO_MATCH],i+1)) {
|
|
madeopt = TRUE;
|
|
next = lpp;
|
|
i = 0; /* dirty way of double break */
|
|
break;
|
|
}
|
|
index=(bp[PO_NEXT]&BMASK)|(bp[PO_NEXT+1]<<8);
|
|
}
|
|
}
|
|
if (lpp == next) {
|
|
count++;
|
|
if (count > 1000) {
|
|
/* probably loop in table */
|
|
fprintf(stderr, "Warning: possible loop in patterns; call an expert\n");
|
|
next = &((*lpp)->l_next);
|
|
count = 0;
|
|
}
|
|
}
|
|
else count = 0;
|
|
lpp = next;
|
|
}
|
|
lpp = alpp;
|
|
if (repl_muls) {
|
|
while ((lp = *lpp) != (line_p) 0 && (lp->l_instr&BMASK) != op_lab) {
|
|
line_p b_repl, e_repl;
|
|
int cnt;
|
|
|
|
if ((cnt = (lp->l_instr & BMASK)) != op_loc && cnt != op_ldc) {
|
|
lpp = &lp->l_next;
|
|
continue;
|
|
}
|
|
|
|
cnt = repl_mul(lp, &b_repl, &e_repl);
|
|
|
|
lp = *lpp;
|
|
if (cnt > 0 && cnt <= repl_muls) {
|
|
*lpp = b_repl;
|
|
e_repl->l_next = lp->l_next->l_next;
|
|
oldline(lp->l_next);
|
|
oldline(lp);
|
|
lpp = &e_repl->l_next;
|
|
madeopt = TRUE;
|
|
}
|
|
else {
|
|
while (b_repl != (line_p) 0) {
|
|
line_p n = b_repl->l_next;
|
|
|
|
oldline(b_repl);
|
|
b_repl = n;
|
|
}
|
|
lpp = &lp->l_next;
|
|
}
|
|
}
|
|
}
|
|
return madeopt;
|
|
}
|
|
|
|
repl_mul(lp, b, e)
|
|
register line_p lp;
|
|
line_p *b, *e;
|
|
{
|
|
register line_p next = lp->l_next;
|
|
int ins;
|
|
int sz;
|
|
unsigned long n;
|
|
int n0, n1;
|
|
int virgin = 1;
|
|
int retval = 0;
|
|
|
|
*b = 0;
|
|
if (! next) return 0;
|
|
if ((ins = (next->l_instr & BMASK)) != op_mli && ins != op_mlu) {
|
|
return 0;
|
|
}
|
|
switch(next->l_optyp) {
|
|
case OPNO:
|
|
return 0;
|
|
case OPSHORT:
|
|
sz = next->l_a.la_short;
|
|
break;
|
|
#ifdef LONGOFF
|
|
case OPOFFSET:
|
|
sz = next->l_a.la_offset;
|
|
break;
|
|
#endif
|
|
default:
|
|
sz = (next->l_optyp & BMASK) - Z_OPMINI;
|
|
break;
|
|
}
|
|
if (ins == op_loc && sz != wordsize) return 0;
|
|
if (ins == op_ldc && sz != 2*wordsize) return 0;
|
|
if (! repl_longmuls && sz != wordsize) return 0;
|
|
switch(lp->l_optyp) {
|
|
case OPSHORT:
|
|
n = (long) lp->l_a.la_short;
|
|
break;
|
|
#ifdef LONGOFF
|
|
case OPOFFSET:
|
|
n = lp->l_a.la_offset;
|
|
break;
|
|
#endif
|
|
default:
|
|
n = (long)((lp->l_optyp & BMASK) - Z_OPMINI);
|
|
break;
|
|
}
|
|
|
|
#define newinstr(res, opcode, val) (*(res) = newline((short)(val)+Z_OPMINI), (*(res))->l_instr = (opcode))
|
|
|
|
while (n) {
|
|
/* first find "0*1*$" in n */
|
|
for (n1 = 0; n & 1; n>>=1) ++n1; /* count "1" bits */
|
|
if (n)
|
|
for (n0 = 0; !(n & 1); n>>=1) /* count "0" bits */
|
|
++n0;
|
|
else
|
|
n0 = 0;
|
|
|
|
if (n1 == 0) {
|
|
if (n0) {
|
|
newinstr(b, op_loc, n0); b = &((*b)->l_next);
|
|
newinstr(b, op_slu, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
}
|
|
} else if (n1 == 1) {
|
|
if (virgin) {
|
|
newinstr(b, op_dup, sz); b = &((*b)->l_next);
|
|
virgin = 0;
|
|
}
|
|
else {
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_dup, 2*sz); b = &((*b)->l_next);
|
|
newinstr(b, op_asp, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_adu, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
}
|
|
if (n) {
|
|
newinstr(b, op_loc, n0+n1); b = &((*b)->l_next);
|
|
newinstr(b, op_slu, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
}
|
|
} else {
|
|
if (virgin) {
|
|
newinstr(b, op_dup, sz); b = &((*b)->l_next);
|
|
if (sz == wordsize) {
|
|
newinstr(b, op_loc, 0); b = &((*b)->l_next);
|
|
}
|
|
else {
|
|
newinstr(b, op_ldc, 0); b = &((*b)->l_next);
|
|
}
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
virgin = 0;
|
|
}
|
|
else {
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_dup, 2*sz); b = &((*b)->l_next);
|
|
newinstr(b, op_asp, sz); b = &((*b)->l_next);
|
|
}
|
|
newinstr(b, op_sbu, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
if (n1 != 8*sz) {
|
|
newinstr(b, op_loc, n1); b = &((*b)->l_next);
|
|
newinstr(b, op_slu, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_dup, 2*sz); b = &((*b)->l_next);
|
|
newinstr(b, op_asp, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_adu, sz); b = &((*b)->l_next);
|
|
newinstr(b, op_exg, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
}
|
|
if (n0) {
|
|
newinstr(b, op_loc, n0); b = &((*b)->l_next);
|
|
newinstr(b, op_slu, sz); b = &((*b)->l_next);
|
|
retval++;
|
|
}
|
|
}
|
|
}
|
|
newinstr(b, op_asp, sz);
|
|
if (virgin) {
|
|
b = &((*b)->l_next);
|
|
newinstr(b, sz == wordsize ? op_loc : op_ldc, 0);
|
|
}
|
|
*e = *b;
|
|
return retval == 0 ? 1 : retval;
|
|
#undef newinstr
|
|
}
|