ack/util/int/do_misc.c

504 lines
8.9 KiB
C

/*
* Sources of the "MISCELLANEOUS" group instructions
*/
/* $Header$ */
#include <em_abs.h>
#include "logging.h"
#include "global.h"
#include "log.h"
#include "trap.h"
#include "warn.h"
#include "mem.h"
#include "memdirect.h"
#include "shadow.h"
#include "text.h"
#include "read.h"
#include "fra.h"
#include "rsb.h"
#include "linfil.h"
extern int running; /* from main.c */
/* Two useful but unofficial registers */
long LIN;
ptr FIL;
PRIVATE index_jump(), range_check(), search_jump();
PRIVATE gto();
#define asp(l) newSP(SP + arg_f(l))
DoASP(l)
register long l;
{
/* ASP f: Adjust the stack pointer by f */
LOG(("@M6 DoASP(%ld)", l));
asp(l);
}
DoASS(l)
register size l;
{
/* ASS w: Adjust the stack pointer by w-byte integer */
LOG(("@M6 DoASS(%ld)", l));
spoilFRA();
l = spop(arg_wi(l));
asp(l);
}
#define block_move(a1,a2,n) \
if (in_stack(a1)) { \
if (in_stack(a2)) st_mvs(a1, a2, n); \
else st_mvd(a1, a2, n); } \
else { if (in_stack(a2)) dt_mvs(a1, a2, n); \
else dt_mvd(a1, a2, n); }
DoBLM(l)
register size l;
{
/* BLM z: Block move z bytes; first pop destination addr, then source addr */
register ptr dp1, dp2; /* Destination Pointers */
LOG(("@M6 DoBLM(%ld)", l));
spoilFRA();
dp1 = dppop();
dp2 = dppop();
block_move(dp1, dp2, arg_z(l));
}
DoBLS(l)
register size l;
{
/* BLS w: Block move, size is in w-byte integer on top of stack */
register ptr dp1, dp2;
LOG(("@M6 DoBLS(%ld)", l));
spoilFRA();
l = upop(arg_wi(l));
dp1 = dppop();
dp2 = dppop();
block_move(dp1, dp2, arg_z(l));
}
DoCSA(l)
register size l;
{
/* CSA w: Case jump; address of jump table at top of stack */
LOG(("@M6 DoCSA(%ld)", l));
spoilFRA();
index_jump(arg_wi(l));
}
DoCSB(l)
register size l;
{
/* CSB w: Table lookup jump; address of jump table at top of stack */
LOG(("@M6 DoCSB(%ld)", l));
spoilFRA();
search_jump(arg_wi(l));
}
DoDCH()
{
/* DCH -: Follow dynamic chain, convert LB to LB of caller */
register ptr lb;
LOG(("@M6 DoDCH()"));
spoilFRA();
lb = dppop();
if (!is_LB(lb)) {
wtrap(WDCHBADLB, ESTACK);
}
dppush(st_lddp(lb + rsb_LB));
}
DoDUP(arg)
size arg;
{
/* DUP s: Duplicate top s bytes */
register ptr oldSP = SP;
LOG(("@M6 DoDUP(%ld)", arg));
spoilFRA();
st_inc(arg_s(arg));
st_mvs(SP, oldSP, arg);
}
DoDUS(l)
register size l;
{
/* DUS w: Duplicate top w bytes */
register ptr oldSP;
LOG(("@M6 DoDUS(%ld)", l));
spoilFRA();
l = upop(arg_wi(l));
oldSP = SP;
st_inc(arg_s(l));
st_mvs(SP, oldSP, l);
}
DoEXG(l)
register size l;
{
/* EXG w: Exchange top w bytes */
register ptr oldSP = SP;
LOG(("@M6 DoEXG(%ld)", l));
spoilFRA();
st_inc(arg_w(l));
st_mvs(SP, oldSP, l);
st_mvs(oldSP, oldSP + l, l);
st_mvs(oldSP + l, SP, l);
st_dec(l);
}
DoFIL(arg)
register unsigned long arg;
{
/* FIL g: File name (external 4 := g) */
register ptr p = i2p(arg);
LOG(("@M6 DoFIL(%lu)", p));
spoilFRA();
if (p > HB) {
wtrap(WILLFIL, EILLINS);
}
putFIL(arg_g(p));
}
DoGTO(arg)
register unsigned long arg;
{
/* GTO g: Non-local goto, descriptor at g */
register ptr p = i2p(arg);
LOG(("@M6 DoGTO(%lu)", p));
gto(arg_gto(p));
}
DoLIM()
{
/* LIM -: Load 16 bit ignore mask */
LOG(("@M6 DoLIM()"));
spoilFRA();
wpush(IgnMask);
}
DoLIN(l)
register unsigned long l;
{
/* LIN n: Line number (external 0 := n) */
LOG(("@M6 DoLIN(%lu)", l));
spoilFRA();
putLIN((long) arg_lin(l));
}
DoLNI()
{
/* LNI -: Line number increment */
LOG(("@M6 DoLNI()"));
spoilFRA();
putLIN((long)getLIN() + 1);
}
DoLOR(l)
register long l;
{
/* LOR r: Load register (0=LB, 1=SP, 2=HP) */
LOG(("@M6 DoLOR(%ld)", l));
spoilFRA();
switch ((int) arg_r(l)) {
case 0:
dppush(LB);
break;
case 1:
dppush(SP);
break;
case 2:
dppush(HP);
break;
}
}
DoLPB()
{
/* LPB -: Convert local base to argument base */
register ptr lb;
LOG(("@M6 DoLPB()"));
spoilFRA();
lb = dppop();
if (!is_LB(lb)) {
wtrap(WLPBBADLB, ESTACK);
}
dppush(lb + rsbsize);
}
DoMON()
{
/* MON -: Monitor call */
LOG(("@M6 DoMON()"));
spoilFRA();
moncall();
}
DoNOP()
{
/* NOP -: No operation */
LOG(("@M6 DoNOP()"));
spoilFRA();
message("NOP instruction");
}
DoRCK(l)
register size l;
{
/* RCK w: Range check; trap on error */
LOG(("@M6 DoRCK(%ld)", l));
spoilFRA();
range_check(arg_wi(l));
}
DoRTT()
{
/* RTT -: Return from trap */
LOG(("@M6 DoRTT()"));
switch (poprsb(1)) {
case RSB_STP:
warning(WRTTEMPTY);
running = 0; /* stop the machine */
return;
case RSB_CAL:
warning(WRTTCALL);
return;
case RSB_RTT:
/* OK */
break;
case RSB_NRT:
warning(WRTTNRTT);
running = 0; /* stop the machine */
return;
default:
warning(WRTTBAD);
return;
}
/* pop the trap number */
uwpop();
/* restore the Function Return Area */
FRA_def = uwpop();
FRASize = uwpop();
popFRA(FRASize);
}
DoSIG()
{
/* SIG -: Trap errors to proc identifier on top of stack, \-2 resets default */
register long tpi = spop(psize);
LOG(("@M6 DoSIG()"));
spoilFRA();
if (OnTrap == TR_HALT) {
npush(-2L, psize);
}
else npush(TrapPI, psize);
if (tpi == -2) {
OnTrap = TR_HALT;
TrapPI = 0;
}
else {
tpi = arg_p(tpi); /* do not test earlier! */
OnTrap = TR_TRAP;
TrapPI = tpi;
}
}
DoSIM()
{
/* SIM -: Store 16 bit ignore mask */
LOG(("@M6 DoSIM()"));
spoilFRA();
IgnMask = (uwpop() | PreIgnMask) & MASK2;
}
DoSTR(l)
register long l;
{
/* STR r: Store register (0=LB, 1=SP, 2=HP) */
LOG(("@M6 DoSTR(%ld)", l));
spoilFRA();
switch ((int) arg_r(l)) {
case 0:
newLB(dppop());
pop_frames();
break;
case 1:
newSP(dppop());
break;
case 2:
newHP(dppop());
break;
}
}
DoTRP()
{
/* TRP -: Cause trap to occur (Error number on stack) */
register unsigned int tr = (unsigned int)uwpop();
LOG(("@M6 DoTRP()"));
spoilFRA();
if (tr > 15 || !(IgnMask&BIT(tr))) {
wtrap(WTRP, (int)tr);
}
}
/* Service routines */
PRIVATE gto(p)
ptr p;
{
register ptr old_LB = LB;
register ptr new_PC = dt_ldip(p);
register ptr new_SP = dt_lddp(p + psize);
register ptr new_LB = dt_lddp(p + (2 * psize));
while (old_LB < new_LB) {
PI = st_lds(old_LB + rsb_PI, psize);
old_LB = st_lddp(old_LB + rsb_LB);
}
if (old_LB != new_LB) {
wtrap(WGTORSB, EBADGTO);
}
newLB(new_LB);
pop_frames();
newSP(new_SP);
newPC(new_PC);
}
/*
The LIN and FIL routines.
The values of LIN and FIL are kept in EM machine registers
(variables LIN and FIL) and in the data space.
*/
putLIN(lin)
long lin;
{
dt_unprot(i2p(LINO_AD), (long)LINSIZE);
dt_stn(i2p(LINO_AD), lin, (long)LINSIZE);
LIN = lin;
dt_prot(i2p(LINO_AD), (long)LINSIZE);
}
putFIL(fil)
ptr fil;
{
dt_unprot(i2p(FILN_AD), psize);
dt_stdp(i2p(FILN_AD), fil);
FIL = fil;
dt_prot(i2p(FILN_AD), psize);
}
/********************************************************
* Case jump by indexing *
* *
* 1. pop case descriptor pointer. *
* 2. pop table index. *
* 3. Calculate (table index) - (lower bound). *
* 4. Check if in range. *
* 5. If in range: load Program Counter value. *
* 6. Else: load default value. *
********************************************************/
PRIVATE index_jump(nbytes)
size nbytes;
{
register ptr cdp = dppop(); /* Case Descriptor Pointer */
register long t_index = /* Table INDEX */
spop(nbytes) - mem_lds(cdp + psize, nbytes);
register ptr nPC; /* New Program Counter */
if (t_index >= 0 && t_index <= mem_lds(cdp + nbytes + psize, nbytes)) {
nPC = mem_ldip(cdp + (2 * nbytes) + ((t_index + 1) * psize));
}
else if ((nPC = mem_ldip(cdp)) == 0) {
trap(ECASE);
}
newPC(nPC);
}
/********************************************************
* Case jump by table search *
* *
* 1. pop case descriptor pointer. *
* 2. pop search value. *
* 3. Load number of table entries. *
* 4. Check if search value in table. *
* 5. If found: load Program Counter value. *
* 6. Else: load default value. *
********************************************************/
PRIVATE search_jump(nbytes)
size nbytes;
{
register ptr cdp = dppop(); /* Case Descriptor Pointer */
register long sv = spop(nbytes);/* Search Value */
register long nt = /* Number of Table-entries */
mem_lds(cdp + psize, nbytes);
register ptr nPC; /* New Program Counter */
while (--nt >= 0) {
if (sv == mem_lds(cdp + (nt+1) * (nbytes+psize), nbytes)) {
nPC = mem_ldip(cdp + nbytes + (nt+1)*(nbytes+psize));
if (nPC == 0)
trap(ECASE);
newPC(nPC);
return;
}
}
nPC = mem_ldip(cdp);
if (nPC == 0)
trap(ECASE);
newPC(nPC);
}
/********************************************************
* Range check *
* *
* 1. Load range descriptor. *
* 2. Check against lower and upper bound. *
* 3. Generate trap if necessary. *
* 4. DON'T remove integer. *
********************************************************/
PRIVATE range_check(nbytes)
size nbytes;
{
register ptr rdp = dppop(); /* Range check Descriptor Pointer */
register long cv = /* Check Value */
st_lds(SP, nbytes);
if (must_test && !(IgnMask&BIT(ERANGE))) {
if ( cv < mem_lds(rdp, nbytes)
|| cv > mem_lds(rdp + nbytes, nbytes)
) {
trap(ERANGE);
}
}
}