ack/mach/proto/ncg/nextem.c

147 lines
2.8 KiB
C
Raw Normal View History

1985-01-08 15:34:54 +00:00
#ifndef NORCSID
1994-06-24 14:02:31 +00:00
static char rcsid[] = "$Id$";
1985-01-08 15:34:54 +00:00
#endif
#include <assert.h>
#include <stdlib.h>
#include <stdio.h>
1985-01-08 15:34:54 +00:00
#include <em_spec.h>
#include <em_flag.h>
#include "param.h"
#include "tables.h"
#include "types.h"
#include <cgg_cg.h>
#include "data.h"
#include "result.h"
#include "extern.h"
/*
1987-03-10 01:26:51 +00:00
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
1985-01-08 15:34:54 +00:00
*
* Author: Hans van Staveren
*/
#ifndef NDEBUG
#include <stdio.h>
extern char em_mnem[][4];
#endif
static int argtyp(int);
static byte *trypat(byte *bp, int len) {
int patlen,i;
1985-01-08 15:34:54 +00:00
result_t result;
getint(patlen,bp);
if (len == 3) {
if (patlen < 3)
return(0);
} else {
if (patlen != len)
return(0);
}
for (i=0;i<patlen;i++)
1985-01-08 15:34:54 +00:00
if (emp[i].em_instr != (*bp++&BMASK))
return(0);
for (i=0;i<patlen;i++)
if (emp[i].em_optyp==OPNO)
dollar[i].e_typ=EV_UNDEF;
else if ((dollar[i].e_typ=argtyp(emp[i].em_instr))==EV_INT)
dollar[i].e_v.e_con=emp[i].em_u.em_ioper;
else {
dollar[i].e_v.e_addr.ea_str=emp[i].em_soper;
dollar[i].e_v.e_addr.ea_off=0;
}
getint(i,bp);
if (i!=0) {
struct emline *svp = saveemp;
saveemp = emp;
compute(&enodes[i], &result);
if (result.e_typ != EV_INT || result.e_v.e_con == 0) {
saveemp = svp;
1985-01-08 15:34:54 +00:00
return(0);
}
1985-01-08 15:34:54 +00:00
}
#ifndef NDEBUG
if (Debug) {
fprintf(stderr,"Matched:");
1986-09-22 13:14:54 +00:00
for (i=0;i<patlen;i++) {
#ifdef USE_TES
1990-07-18 14:53:19 +00:00
if (emp[i].em_instr == op_lab)
fprintf(stderr," lab");
else
#endif
fprintf(stderr," %3.3s",em_mnem[emp[i].em_instr-sp_fmnem]);
1986-09-22 13:14:54 +00:00
if (emp[i].em_soper)
fprintf(stderr," %s",emp[i].em_soper);
1986-09-22 13:14:54 +00:00
}
1985-01-08 15:34:54 +00:00
fprintf(stderr,"\n");
}
#endif
saveemp = emp;
emp += patlen;
return(bp);
}
extern char em_flag[];
static int argtyp(int mn) {
1985-01-08 15:34:54 +00:00
/* op_lab is a special opcode which represents a label definition. It's
* not actually a real EM instruction. Therefore if we try to look it
* up in em_flag, we'll get a buffer overrun... */
if (mn == op_lab)
return EV_UNDEF;
switch (em_flag[mn-sp_fmnem]&EM_PAR) {
1985-01-08 15:34:54 +00:00
case PAR_W:
case PAR_S:
case PAR_Z:
case PAR_O:
case PAR_N:
case PAR_L:
case PAR_F:
case PAR_R:
case PAR_C:
return(EV_INT);
default:
return(EV_ADDR);
}
}
byte *nextem(int toplevel) {
int i;
1985-01-08 15:34:54 +00:00
short hash[3];
byte *bp;
1985-01-08 15:34:54 +00:00
byte *cp;
int index;
struct emline *ep;
1985-01-08 15:34:54 +00:00
if (toplevel) {
if (nemlines && emp>emlines) {
nemlines -= emp-emlines;
for (i=0,ep=emlines;i<nemlines;i++)
*ep++ = *emp++;
emp=emlines;
}
fillemlines();
}
hash[0] = emp[0].em_instr;
hash[1] = (hash[0]<<4) ^ emp[1].em_instr;
hash[2] = (hash[1]<<4) ^ emp[2].em_instr;
for (i=2;i>=0;i--) {
index = pathash[hash[i]&BMASK];
while (index != 0) {
bp = &pattern[index];
if ( bp[PO_HASH] == (hash[i]>>8))
if ((cp=trypat(&bp[PO_MATCH],i+1)) != 0)
return(cp);
index = (bp[PO_NEXT]&BMASK) | (bp[PO_NEXT+1]<<8);
}
}
return(0);
}