ack/lang/occam/comp/em.c
1994-06-24 14:02:31 +00:00

466 lines
8.3 KiB
C

/* $Id$ */
/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
#include "sizes.h"
#include "Lpars.h"
#include <em.h>
#include "em.h"
/* This file is used to shield code.c as much as possible from em dependant
* details. It introduces some call overhead but not enough for a coffee
* break. (Sorry)
* Note that functions with a leading upper case letter normally decide between
* word or double word arith.
*/
int wz = 4, pz = 4, vz = 4;
int Lflag;
static Lab=0;
char *Malloc();
void init()
{
C_init((arith) wz, (arith) pz);
}
void openfile(file) char *file;
{
if (C_open(file) < 0) {
fatal("Could not open output file");
}
}
void meswp()
{
C_ms_emx((arith) wz, (arith) pz);
}
void maxdes()
{
long max = (1L << (8*wz-1)) - 1;
C_df_dnam("maxcdes");
rom(wz, 0L); rom(wz, max); rom(wz, (long) (wz+vz));
C_df_dnam("maxwdes");
rom(wz, 0L); rom(wz, max); rom(wz, (long) vz);
C_df_dnam("maxbdes");
rom(wz, 0L); rom(wz, max); rom(wz, 1L);
}
int new_label(L) register *L;
{
if (*L==0) *L= ++Lab;
return *L;
}
void Label(L) register L;
{
if (L!=0) C_df_ilb((label) L);
}
static Dot_label=0;
int new_dot_label(L) int *L;
{
return *L= ++Dot_label;
}
void dot_label(L) int L;
{
C_df_dlb((label) L);
}
void branch(L) int *L;
{
C_bra((label) new_label(L));
}
char *proc_label(L, name) register L; register char *name;
{
static char *lab=nil;
register char *n;
if (lab!=nil) free(lab);
lab=Malloc(strlen(name)+(1+sizeof(int)*3+1));
/* That is: P<L><name>\0 */
sprint(lab, "P%d", L);
n=lab+strlen(lab);
while (*name!=0) {
*n++ = *name=='.' ? '_' : *name;
name++;
}
*n=0;
return lab;
}
void magic() /* magic? should be called invisible */
{
C_magic();
}
void cwv()
{
if (vz>wz) {
C_loc((arith) wz);
C_loc((arith) vz);
C_cii();
}
}
void cvw()
{
if (vz>wz) {
C_loc((arith) vz);
C_loc((arith) wz);
C_cii();
}
}
void Loc(cst) long cst;
{
if (vz>wz) C_ldc((arith) cst); else C_loc((arith) cst);
}
void Lol(offset) int offset;
{
if (vz>wz) C_ldl((arith) offset); else C_lol((arith) offset);
}
void Lolp(offset) int offset;
{
if (pz>wz) C_ldl((arith) offset); else C_lol((arith) offset);
}
void Lil(offset) register offset;
{
if (vz>wz) {
Lolp(offset);
C_loi((arith) vz);
} else
C_lil((arith) offset);
}
void Lof(offset) int offset;
{
if (vz>wz) C_ldf((arith) offset); else C_lof((arith) offset);
}
void Lofp(offset) int offset;
{
if (pz>wz) C_ldf((arith) offset); else C_lof((arith) offset);
}
void Lif(offset) register offset;
{
Lofp(offset);
C_loi((arith) vz);
}
void Stl(offset) int offset;
{
if (vz>wz) C_sdl((arith) offset); else C_stl((arith) offset);
}
void Inl(offset) register offset;
{
if (vz>wz) {
C_ldl((arith) offset);
C_ldc((arith) 1);
C_adi((arith) vz);
C_sdl((arith) offset);
} else
C_inl((arith) offset);
}
void Del(offset) register offset;
{
if (vz>wz) {
C_ldl((arith) offset);
C_ldc((arith) 1);
C_sbi((arith) vz);
C_sdl((arith) offset);
} else
C_del((arith) offset);
}
void Loe(name, offset) char *name; int offset;
{
if (vz>wz)
C_lde_dnam(name, (arith) offset);
else
C_loe_dnam(name, (arith) offset);
}
static int operators[]= { '<', '>', '=', GE, LE, NE };
void bxx(pos, op, L) register pos, op, L;
{
register i;
if (op==AFTER) {
C_sbi((arith) vz);
if (vz>wz) {
C_ldc((arith) 0);
C_cmi((arith) vz);
}
if (pos) C_zle((label) L); else C_zgt((label) L);
} else {
for (i=0; operators[i]!=op; i++) ;
if (pos && (i+=3)>=6) i-=6;
if (vz>wz) {
C_cmi((arith) vz);
switch(i) {
case 0:
C_zlt((label) L);
break;
case 1:
C_zgt((label) L);
break;
case 2:
C_zeq((label) L);
break;
case 3:
C_zge((label) L);
break;
case 4:
C_zle((label) L);
break;
case 5:
C_zne((label) L);
break;
}
} else {
switch(i) {
case 0:
C_blt((label) L);
break;
case 1:
C_bgt((label) L);
break;
case 2:
C_beq((label) L);
break;
case 3:
C_bge((label) L);
break;
case 4:
C_ble((label) L);
break;
case 5:
C_bne((label) L);
break;
}
}
}
}
void Txx(op) register int op;
{
switch(op) {
case '<':
C_tlt();
break;
case '>':
C_tgt();
break;
case '=':
C_teq();
break;
case GE:
C_tge();
break;
case LE:
C_tle();
break;
case NE:
C_tne();
break;
}
cwv();
C_ngi((arith) vz);
}
void xxi(op) register op;
{
switch(op) {
case '+':
C_adi((arith) vz);
break;
case '-':
C_sbi((arith) vz);
break;
case '*':
C_mli((arith) vz);
break;
case '/':
C_dvi((arith) vz);
break;
case BS:
C_rmi((arith) vz);
break;
}
}
void aar() { C_aar((arith) wz); }
void adp(offset) int offset; { C_adp((arith) offset); }
void and() { C_and((arith) vz); }
void asp(size) int size; { C_asp((arith) size); }
void blm(size) int size; { C_blm((arith) size); }
void blt(lab) int lab; { C_blt((label) lab); }
void cal(lab) char *lab; { C_cal(lab); }
void cmi() { C_cmi((arith) vz); }
void com() { C_com((arith) vz); }
void del(offset) int offset; { C_del((arith) offset); }
void x_end(size) int size; { C_end((arith) size); }
void exp(lab) char *lab; { C_exp(lab); }
void ior() { C_ior((arith) vz); }
void lae(lab, offset) char *lab; int offset;
{ C_lae_dnam(lab, (arith) offset); }
void laedot(lab) int lab; { C_lae_dlb((label) lab, (arith) 0); }
void lal(offset) int offset; { C_lal((arith) offset); }
void lar() { C_lar((arith) wz); }
void ldc0() { C_ldc((arith) 0); }
void ldl(offset) int offset; { C_ldl((arith) offset); }
void lfr(size) int size; { C_lfr((arith) size); }
void loc(cst) int cst; { C_loc((arith) cst); }
void loi(size) int size; { C_loi((arith) size); }
void lol(offset) int offset; { C_lol((arith) offset); }
void lor0() { C_lor((arith) 0); }
void lxa(offset) int offset; { C_lxa((arith) offset); }
void lxl(offset) int offset; { C_lxl((arith) offset); }
void meserr() { C_ms_err(); }
void ngi() { C_ngi((arith) vz); }
void pro(lab) char *lab; { C_pro_narg(lab); }
void ret(size) int size; { C_ret((arith) size); }
void init_rt() { C_cal("init"); }
void sli() { C_sli((arith) vz); }
void sri() { C_sri((arith) vz); }
void ste(lab, offset) char *lab; int offset;
{ C_ste_dnam(lab, (arith) offset); }
void sti(size) int size; { C_sti((arith) size); }
void stl(offset) int offset; { C_stl((arith) offset); }
void trp() { C_trp(); }
void tst() { /* No flags in EM */ }
void xor() { C_xor((arith) vz); }
void zeq(lab) int lab; { C_zeq((label) lab); }
void zgt(lab) int lab; { C_zgt((label) lab); }
void zlt(lab) int lab; { C_zlt((label) lab); }
void zne(lab) int lab; { C_zne((label) lab); }
char *itoa(i) long i;
{
static char a[sizeof(long)*3];
sprint(a, "%ld", i);
return a;
}
void rom(size, c) int size; long c;
{
C_rom_icon(itoa(c), (arith) size);
}
void lin()
{
static oldline=0;
extern lineno;
if (Lflag) return;
if (lineno!=oldline)
C_lin((arith) (oldline=lineno));
}
static struct ftree {
char *file;
int lab;
struct ftree *left, *right;
} std_f = { "stdin", 0, nil, nil }, *curr_f= &std_f, *main_f=nil;
char *curr_file="stdin";
static void do_fil(f) struct ftree *f;
{
if (Lflag) return;
if (f->lab==0) {
dot_label(new_dot_label(&f->lab));
C_rom_scon(f->file, (arith) (strlen(f->file)+1));
}
C_fil_dlb((label) f->lab, (arith) 0);
}
void fil()
{
do_fil(curr_f);
}
void main_fil()
{
do_fil(main_f==nil ? &std_f : main_f);
}
int set_file(f) char *f;
{
char *strcpy();
static struct ftree *ftop=nil;
register struct ftree *pf, **apf= &ftop;
register cmp;
while ((pf= *apf)!=nil && (cmp=strcmp(f, pf->file))!=0)
apf= cmp<0 ? &pf->left : &pf->right;
if (pf==nil) {
*apf= pf= (struct ftree *) Malloc(sizeof *pf);
pf->file=strcpy(Malloc(strlen(f)+1), f);
pf->lab=0;
pf->left=pf->right=nil;
}
curr_f=pf;
curr_file=pf->file;
if (main_f==nil) {
main_f=curr_f;
return 0;
} else
return curr_f!=main_f;
}
void par_begin()
{
C_zer((arith) pz);
C_lal((arith) curr_offset);
C_cal("pc_begin");
C_asp((arith) (2*pz));
}
void par_fork(NONZERO) int *NONZERO;
{
C_zer((arith) pz);
C_cal("pc_fork");
C_asp((arith) pz);
C_lfr((arith) wz);
C_zne((label) new_label(NONZERO));
}
void resumenext()
{
C_cal("resumenext");
}
void no_deadlock()
{
C_zre_dnam("deadlock", (arith) 0);
}
void par_end()
{
C_cal("parend");
}
void closefile()
{
C_close();
}