:Title[Lisp0];
*
* Edit History
* February 2, 1985 10:34 PM, Masinter, Def2BR was set wrong!
* January 22, 1985 12:24 AM, Masinter, remove Altomode instructions no
* longer used
* January 5, 1985 11:39 PM, JonL, Flush ZeroBR; setup Val2BR and Def2BR
* for 64K Litatoms scheme
* February 18, 1984 4:54 PM, JonL, SAVEUCODESTATE uses SubrArgArea
*
instead of statsBuffer
* January 31, 1984 7:31 PM, JonL, embellish SAVEUCODESTATE
* January 31, 1984 5:05 PM, Masinter, add SAVEUCODESTATE as subroutine
* January 23, 1984 6:51 PM, JonL, debugging previous change
* January 20, 1984 6:26 AM, JonL, added check for BLT in pageFault
* January 4, 1983 2:48 PM, Masinter

:insert[DisplayDefs.mc];
mc[max.pvar.for.fault, 3000];
* Code for Interface with BCPL

KnowRBase[LTEMP0];
TOP LEVEL;
InsSet[LispInsSet, 1];

mc[UCODE.CHECK, 0];
mc[STKOV.PUNT, sub[0,SubovFXP!]]; * says context switch to Subov
mc[NWW.INTERRUPT, 2];
mc[PAGE.FAULT, sub[0,FAULTFXP!]];
mc[STATS.PUNT, 4];

*--------------------------------------------------------------------
opSUBR:
*--------------------------------------------------------------------
LTEMP0← Id;
T← LTEMP2← Id;* Beta byte is # args
T← T + T;
T← TSP← (TSP) - (Cnt← T);* Move args from stack
LTEMP1← SubrArgArea, Branch[.subr1];

.subr0:
T← (fetch← T) + 1;* Tight loop to move args
memBase← MDS;* from stackframe to the
LTEMP1← (store← LTEMP1)+ 1, dbuf← Md;* BCPL subr arg area
.subr1:
memBase← StackBR, branch[.subr0, Cnt#0&-1];

LTEMP1← (Id) - (PCX’) - 1, branch[.storepuntpc];

IFUpause[175, 3, StackBR, 0, opSUBR, noNData, 0, 0];
*SUBRCALL

*--------------------------------------------------------------------
opCNTXTSWITCH:
*--------------------------------------------------------------------
T← (TSP) - 1;
TSP← (fetch← T) - 1;
* fetch arg
rbase← rbase[NWW];
T← Md, NWW← (NWW) and not (100000c);
* turn on interrupts
pd← NWW, rbase← rbase[LTEMP0];
branch[.+2, alu=0], LTEMP0← (0s) - T; * LTEMP0← - (context#)
pd← LTEMP0, RescheduleNow;
* reschedule if int pending...
LTEMP1← (Id) - (PCX’) - 1, branch[.storepuntpc];

IFUpause[176, 1, StackBR, 0, opCNTXTSWITCH, noNData, 0, 0];
*CNTXTSWITCH

*--------------------------------------------------------------------
UCODECHECKPUNT: GLOBAL,
*--------------------------------------------------------------------

* call: SaveLink← Link, Branch[uCodeCheckPunt];

LTEMP3← T, T← Link, Call[SAVEUCODESTATE];* First, save state;
LTEMP4← SubrArgArea;* Then, set up ucode
LTEMP4← (store← LTEMP4) + 1, dbuf← SmallHi;* address of punter
LTEMP4← (store← LTEMP4) + 1, dbuf← T;* as arg to BCPL subr
LTEMP2← 1c;
LTEMP0← UCODE.CHECK, branch[BCPLEXIT];

*--------------------------------------------------------------------
SUBROUTINE;
SAVEUCODESTATE: GLOBAL,
*--------------------------------------------------------------------
* Clobbers LTEMP4 and BR
* Called by
*
rbase← rbase[LTEMP0];
*
KnowRBase[LTEMP0];
*
LTEMP3← T, T← Link, Call[SAVEUCODESTATE];
* <<< CAN BE BUMMED >>>
memBase← MDS;* Save some volatile
Q← LTEMP4;* ucode regs in stats
LTEMP4← (220c);
LTEMP4← (store← LTEMP4) + 1, dbuf← (125377c);* 0 PassWord
LTEMP4← (store← LTEMP4) + 1, dbuf← T;* 1 Link at call here
T← Link;
T← (store← LTEMP4) + 1, dbuf← T;* 2 ucode.addr + 1 of
RBase←rbase[FLTEMUPC];* of call to here
T←(store← T) + 1, dbuf← FLTEMUPC;* 3 PC at fault
RBase←rbase[LTEMP0];
T← (store← T) + 1, dbuf← SaveLink;* 4 maybe saved link
T← (store← T) + 1, dbuf← LTEMP3;* 5 T register
T← (store← T) + 1, dbuf← LTEMP0;* 6 LTEMP0
T← (store← T) + 1, dbuf← LTEMP1;* 7 LTEMP1
T← (store← T) + 1, dbuf← PVAR;* 10 PVAR
T← (store← T) + 1, dbuf← TSP;* 11 TSP
T← (store← T) + 1, dbuf← PSTATE;* 12 PSTATE
T← (store← T) + 1, dbuf← DEFLO;* 13 DEFLO
T← (store← T) + 1, dbuf← NARGS;* 14 NARGS
T← (store← T) + 1, dbuf← Q;* 15 LTEMP4
LTEMP0← TIOA&StkP;
T← (Store← T) + 1, dbuf← LTEMP0;* 16 stackp
LTEMP0← 1c;
StkP← LTEMP0;
T← (Store← T) + 1, dbuf← Stack&+1;
T← (Store← T) + 1, dbuf← Stack&+1;
T← (Store← T) + 1, dbuf← Stack&+1;
T← (Store← T) + 1, dbuf← Stack&+1;
T← (Store← T) + 1, dbuf← Stack&+1;
T← LTEMP3, Return;
TOP LEVEL;

*--------------------------------------------------------------------
RAIDPUNT:
*--------------------------------------------------------------------

* like UCODEPUNT, but registers are OK;

RBase← rbase[LTEMP0];
memBase← MDS;
LTEMP0← SubrArgArea;
LTEMP0← (store← LTEMP0) + 1, dbuf← SmallHi;
store← LTEMP0, dbuf← 0c;
LTEMP2← 1c;
LTEMP0← UCODE.CHECK, Branch[PUNT];

*--------------------------------------------------------------------
STKOVPUNT:
*--------------------------------------------------------------------
LTEMP0← STKOV.PUNT, branch[.puntz];

*--------------------------------------------------------------------
STATSPUNT:
*--------------------------------------------------------------------
LTEMP0← STATS.PUNT, branch[.puntz];

*--------------------------------------------------------------------
NWWPUNT:* old NWW: exit to BCPL w/reschedule still set
*--------------------------------------------------------------------
rbase← rbase[LTEMP0];
LTEMP0← NWW.INTERRUPT, branch[.puntz];

*--------------------------------------------------------------------
KEYPUNT: KnowRBase[NWW]; * new NWW: context switch to KBD context
*--------------------------------------------------------------------
NWW← (100000c); * turn off interrupts
rbase← rbase[LTEMP0];
LTEMP0← sub[0, KbdFXP!]c, branch[.puntz];

.puntz:
LTEMP2← A0, branch[PUNT];

*--------------------------------------------------------------------
PAGEFAULTPUNT:
*--------------------------------------------------------------------
rbase← rbase[FltPipe0];
memBase← InterfaceBR;
T← IFPFAULTHI;
T← (store← T) + 1, dbuf← FltPipe0;
store← T, dbuf← FltPipe1;
rbase← rbase[LTEMP0];
* Check first for pagefault while "in function call" (which is OK).
:if[Debugging];
Branch[.+2, R>=0], pd← (PSTATE) and (PS.PFOK);
Branch[.pfp1];* OK to fault in fn call. No constraint on .pfp1
Branch[.+2, alu#0], pd← (PSTATE) and (add[PS.INBITBLT!, PS.INBLT!]c);
uCodeCheck[PageFaultWhenNotOK];
:else;
pd← (PSTATE) and (add[PS.INBITBLT!, PS.INBLT!]c), * PSTATE is -1
Branch[.+2, R>=0];* in fn call
Branch[.pfp1];* OK to fault in fn call. No constraint on .pfp1
:endif;
* Check also for fault in BLT or BITBLT (which need stack patchup).
Branch[.pfp1, alu=0], PSTATE, pd← (PSTATE) and (PS.INBLT);
Branch[.+2, alu=0], memBase← StackBR, T← (TSP) - 1;
stack← (stack) + 1;
PSTATE← A0, store← T, dbuf← Stack, Branch[.pfp1];
.pfp1:
** check for page fault in page fault context
pd← (PVAR) - (max.pvar.for.fault);
Branch[.+2, carry], LTEMP0← PAGE.FAULT;
uCodeCheck[PageFaultRecursion];
:if[Debugging];
rbase← rbase[NWW];
Branch[.+2, R>=0], NWW, rbase← rbase[LTEMP0];
uCodeCheck[NWW?];
:endif;
.pfp2:
LTEMP2← A0, branch[PUNT];



*--------------------------------------------------------------------
* common punt code
*--------------------------------------------------------------------
PUNT:
T← (PVAR) - (FXBACK[FLAGS]);
memBase← StackBR, PSTATE, branch[.normalpunt, R>=0];

* punt in call

fetch← T, LTEMP1← FXInCall;
LTEMP1← (LTEMP1) or Md;
store← T, dbuf← LTEMP1;
T← (store← TSP) + 1, dbuf← SmallHi;
T← (store← T) + 1, dbuf← NARGS;
T← (store← T) + 1, dbuf← 0c;* can’t fault if DEFHI nonzero
TSP← (store← T) + 1, dbuf← DEFLO, branch[.puntfixstack];

.normalpunt:
fetch← T, LTEMP1← FXNoPushReturn;
LTEMP1← (LTEMP1) or Md;
store← T, dbuf← LTEMP1;
LTEMP1← not (PCX’);

.storepuntpc:
* from SUBR and context switch, too
T← (PVAR) - (FXBACK[PC]);
store← T, dbuf← LTEMP1;


.puntfixstack:
T← (PVAR) - (FXBACK[NEXT]);
store← T, T← dbuf← TSP;* store NEXT
T← (ESP) - T;
branch[.+2, carry], TSP← (store← TSP) + 1, dbuf← FreeStackBlock;
uCodeCheck[NoStackAtPunt];
store← TSP, dbuf← T;

* LTEMP0 = punt or subr#, or else (- context#)
* LTEMP2 = number of args
* PVAR ok

BCPLEXIT:
memBase← interfaceBR;
PVAR← (PVAR) - (FX.PVAR);
branch[.ctxswitch, R<0], Q← LTEMP0;
:if[Debugging];
PSTATE← (PS.INBCPL);
:endif;
store← add[CurrentFXP!]s, dbuf← PVAR;
:if[FNStats];
branch[.+2, R>=0], FnStatsPtr;
branch[.bcplxend];
nop;* Following Call constrains addresses
DEFLO← Q, Call[.subrstat];
memBase← MDS;
T← StatsBufferPtr;
store← T, dbuf← FnStatsPtr, Branch[.bcplxend];
:endif; * FNStats

.bcplxend:
T← LTEMP2, rbase← rbase[spAC0];
StkP← spAC2;
Stack&+1← Q;* value for AC2 Punt or subr #
Stack&-1← T;* # of args
T← AemuRestartLoc, branch[start];

KnowRBase[LTEMP0];

.ctxswitch:
T← (0s) - (LTEMP0);* context#
fetch← T;
:if[Debugging];
PSTATE← (PS.PCXBAD);
:else;
PSTATE← A0;
:endif;
PVAR← Md, store← T, dbuf← PVAR;
PVAR← (PVAR) + (FX.PVAR), branch[RTN2];


:if[FNStats];

SUBROUTINE;

*--------------------------------------------------------------------
* Stats writing
*--------------------------------------------------------------------

.subrstat:
DEFHI← A0;
T← LSH[LTEMP2, 10];
T← T or (CALL.EVENT), branch[.storestat];

FNSTAT: * fn in LTEMP0, 1, NARGS set.
T← LTEMP0, memBase← MDS;
T← LCY[T, NARGS, 10];
T← T or (CALL.EVENT);
T← (store← FnStatsPtr) + 1, dbuf← T;
T← (store← T) + 1, dbuf← LTEMP1, branch[.stattail];

.storeretstat:
T← (RETURN.EVENT), branch[.storestat];

.storestat:
T← T or (DEFHI), memBase← MDS;
T← (store← FnStatsPtr) + 1, dbuf← T;
T← (store← T) + 1, dbuf← DEFLO;
.stattail:
FnStatsPtr← T;
T← 30c;
T← T + (400c);
TaskingOFF;
fetch← T;
T← (store← FnStatsPtr) + 1, dbuf← Md;
RBase← RBase[RTClock];
T← (store← T) + 1, dbuf← RTClock;
TaskingON;
RBase← RBase[FnStatsPtr];
FnStatsPtr← T;
pd← T - (StatsBufferBoundary);
branch[.+2, alu<=0];
ReSchedule;
Return;


TOP LEVEL;

:endif; * FNStats


*--------------------------------------------------------------------

KnowRBase[AEmRegs];
m[MBXI, KnowRBase[AEmRegs]
Top level];

LTrap:
ETEMP2← Id, call[GetPC];
branch[.+2, alu<0], ETEMP4← T + 1;
BigBdispatch← ETEMP2;
branch[LTrapDispatch], StkP← spAC0;

EmuNext:
rbase← rbase[AEmRegs], global;
T← ETEMP4, branch[start];

EmuSkip:
rbase← rbase[AEmRegs];
T← (ETEMP4) + 1, branch[start];

*--------------------------------------------------------------------
* arrive at the Lisp dispatch locations with StkP← spAC0
*--------------------------------------------------------------------
* vanMelle claims 0,1,2,4,5,6,10,11 are used

LTrapDispatch:
branch[MBIX], dispTable[20];
* 00
branch[ReadFlags];
* 01
branch[SetFlags];
* 02
branch[NPTrap];
* 03was XferPage
branch[BGetBase];
* 04
branch[BPutBase];
* 05
branch[BGetBase32];
* 06
branch[NPTrap];
* 07 was BGetBasePtr
branch[BPutBase32];
* 10
branch[InitLispRegs];
* 11

%
branch[EmuNext];* 12 was GetRamVersion
branch[NPTrap];
* 13 was GetFXP
branch[NPTrap];
* 14 was SetFXP
branch[uPCTrace];
* 15
branch[XBitBlt];
* 16
branch[NPTrap];
* 17 was CallFN
%

MBXI;
*--------------------------------------------------------------
MBIX: rbase← rbase[LTEMP1];
LTEMP0← Stack&+1;
* AC0: hi part of return value
LTEMP1← Stack;
* AC1: lo part of return value

:if[Debugging];
PSTATE← (PS.PCXBAD);
:else;
PSTATE← A0;
:endif;

:if[FNStats];
rbase← rbase[PVAR];
T← StatsBufferPtr;
memBase← MDS;
fetch← T;
FnStatsPtr← Md;
pd← FnStatsPtr;
branch[.+2, alu#0];
FnStatsPtr← -1c;

:else;
FnStatsPtr← T-T-1;
memBase← MDS;
:endif;

* memBase=MDS
rbase← rbase[NWW];
T← (R400) + (52C);
* WW (= 452B)
fetch← T, T← (100000C);
T← (Md) and not (T);
NWW ← (NWW) or T;
branch[.+2, alu=0], rbase← rbase[LTEMP0];
Reschedule;

T← add[100000, LShift[LispInsSet, 10]]c;
* set InsSet
InsSetOrEvent← T;

MemBX← 0s;
* SET MemBX

T← StackEmpty;
* set StkP
StkP← T;

memBase← interfaceBR;
fetch← add[CurrentFXP!]s;
PVAR← Md;
PVAR← (PVAR) + (FX.PVAR), branch[RTN2];


MBXI;
*--------------------------------------------------------------
ReadFlags:
call[flushVp], T← Stack;
RMap← ETEMP3, call[waitforMapBuf];
* uses T only
Stack&+1← not (Map’);
T← not (Pipe4’);
Stack&-1← T and (m1pipe4.wpdref),
* wp, d, & ref from pipe4
branch[EmuNext];


KnowRBase[LTEMP0];

*--------------------------------------------------------------
opREADFLAGS:
T← (TSP) - 1;
fetch← T;
call[flushVp], T← Md, rbase←rbase[ETEMP3];
RMap← ETEMP3, call[waitforMapBuf];

pd← Id, rbase← rbase[TSP];
branch[.+2, alu=0], LTEMP0← not (Pipe4’);
LTEMP0← not (Map’), branch[.readtail];
LTEMP0← (LTEMP0) and (m1pipe4.wpdref), branch[.readtail];

.readtail:

T← (Id) - (PCX’) - 1;
PCF← T;
* restart IFU
T← (TSP) - 1, memBase← StackBR;
store← T, dbuf← LTEMP0, NextOpCode;

regOP1[161, StackBR, opREADFLAGS, 0];
* readflags
regOP1[162, StackBR, opREADFLAGS, 1];
* readrp

MBXI;
*--------------------------------------------------------------
SetFlags:
StkP← spAC2;
T← (Stack&-2) + (3c);
fetch← T;
* fetch flags
ETEMP2← Md, T← Stack&+1, call[flushVP];
*flush cache
T← lsh [ETEMP2, 2];
* position the wp&dirty bits
T← T and (TIOAvacantMapEntry);
* mask out any extra bits
T← Stack&-1, TIOA← T;
B← T, TASKINGOFF;
* get bmux stable for Map←
Map← ETEMP3, B← T;
* write the map
TASKINGON;
call[waitforMapBuf], TIOA← ETEMP3;
* clear TIOA

* flushVp did one IFUReset, must do a second

branch[.+2, R>=0], ETEMP2, IFUReset;
* check for ref bit
fetch← ETEMP3;* reference it
branch[EmuNext];

*--------------------------------------------------------------------
SUBROUTINE;

FlushVP:
* vp is in T, uses ETEMP3, sets memBase
memBase← LScratchBR, B← Md;
* finish any stores
IFUreset;
* stop IFU from making refs
ETEMP3← lsh[T, 10];
T← rsh[T, 10];
BrHi← T;
BrLo← ETEMP3;
T← 360C;

FlushVPLoop: flush← T;
T← T - (20C);
branch[FlushVPLoop, alu>=0];
B← Md, ETEMP3← A0, return;

TOP LEVEL;

% * no longer used
MBXI;
*--------------------------------------------------------------
XferPage:
call[setXferBR], memBase← BBDSTBR, T← Stack&+1;
call[setXferBR], memBase← BBSRCBR, T← Stack&-1;
T← rhmask;
Cnt← T;
fetch← T, flipMemBase;
T← (store← T) - 1, dbuf← Md, flipMemBase, branch[.-1, Cnt#0&-1];
branch[EmuSkip];

SUBROUTINE;

setXferBR:
ETEMP3← RSH[T, 10];
BrHi← ETEMP3;
ETEMP3← LSH [T, 10];
BrLo← ETEMP3, return;
TOP LEVEL;

%


MBXI;
*--------------------------------------------------------------
BGetBase:
call[BFetch];
Stack← Md, branch[EmuSkip];

MBXI;
*--------------------------------------------------------------
BGetBase32:
call[BFetch];
Stack&+1← Md;
fetch← 1s;
Stack← Md, branch[EmuSkip];

% no longer used
MBXI;
*--------------------------------------------------------------
BGetBasePtr: T← rhmask;
call[BFetch];
Stack&+1← T and (Md);
fetch← 1s;
Stack← Md, branch[EmuSkip];
%

SUBROUTINE;
BFetch:
memBase← LScratchBR;
BrHi← Stack&+1;
BrLo← Stack&-1;
fetch← 0s, return;
TOP LEVEL;


MBXI;
*--------------------------------------------------------------
BPutBase:
StkP← spAC2;
T← (Stack&-2) + (3c);
fetch← T;
T← Md, call[BStore], memBase← LScratchBR;
B← Md, branch[EmuSkip];

MBXI;
*--------------------------------------------------------------
BPutBase32:
StkP← spAC2;
T← (Stack&-2) + (3c);
fetch← T;
T← Md;
T← (fetch← T) + 1;
T← Md, fetch← T;
call[BStore], ETEMP2← Md, memBase← LScratchBR;
T← ETEMP2;
store← 1s, dbuf← T, branch[EmuSkip];

*--------------------------------------------------------------------
SUBROUTINE;
BStore:
BrHi← Stack&+1;
BrLo← Stack&-1;
* restore to AC0
store← 0s, dbuf← T, return;
TOP LEVEL;

MBXI;
*--------------------------------------------------------------
InitLispRegs:

rbase← rbase[RMForIFU];

MemBX← 0s;
* SET MemBX

AllOnes← T-T-1;

:if[Debugging];
PSTATE← (PS.INBCPL);
:endif;

T← stackHI;
LTEMP0 ← A0, memBase← StackBR, call[setBR];

LTEMP0← (LTEMP0) - (2c);
T← T - 1, memBase← StackM2BR, call[setBR];

T← VALspace;
LTEMP0← A0, memBase← ValSpaceBR, call[setBR];
T← T + 1, memBase← Val2BR, call[setBR];

T← LTEMP0← A0, memBase← ScratchLZBR, call[setBR];

T← DEFspace;
LTEMP0← A0, memBase← DefBR, call[setBR];
T← T + 1, memBase← Def2Br, call[setBR];

T← HTMAINspace;
LTEMP0← HTMAINbase;
memBase← htMainBR, call[setBR];

LTEMP0← HTOVERFLOWbase;
memBase← HTOfloBR, call[setBR];

* Note that it is required that
* DTDspace = MDSTYPEspace = UFNspace = INTERFACEspace

T← INTERFACEspace;
LTEMP0← INTERFACEbase;
memBase← interfaceBR, call[setBR];

LTEMP0← DTDbase;
memBase← dtdBR, call[setBR];

LTEMP0← (LTEMP0) + (lshift[ListType!, 4]c);
memBase← ListpDTDBR, call[setBR];

LTEMP0 ← MDSTYPEbase;
memBase← tybaseBR, call[setBR];

LTEMP0 ← UFNTablebase;
memBase← ufnBR, call[setBR];


memBase← MDS;
T← and[RamVersion, 177400]c;
T← T + (and[RamVersion, 377]c);
Stack← (store← Stack) + 1, dbuf← T;

T← and[RamMinBcplVersion, 177400]c;
T← T + (and[RamMinBcplVersion, 377]c);
Stack← (store← Stack) + 1, dbuf← T;

T← and[RamMinLispVersion, 177400]c;
T← T + (and[RamMinLispVersion, 377]c);
Stack← (store← Stack) + 1, dbuf← T;

* now initialize/find out display width

rbase← rbase[DisplayConfig];
T← OR[177400, MaxWidthWordsAlto!]C, DisplayConfig,
Branch[ILRDisp, R even];* Branch if Alto monitor

* Switch to using the entire width of the LF monitor:

DisplayConfig← (DisplayConfig) OR (2C);
T← Or[177400, MaxWidthWordsLF!]C;

* T has the new words/scanline
* 377 in the LH to signal DHT to reinitialize.

MaxWidthWords← T;

ILRDisp: * <<<can bum>>>
T← T AND (377C);
Stack← (store← Stack) + 1, dbuf← T;

rbase← rbase[RealPages];
T← RealPages;
Stack← (store← Stack) + 1, dbuf← T;
* # pages
T← 2000c;
* "dummy" #pages/module
Stack← (store← Stack) + 1, dbuf← T,
* (doesn’t matter)
branch[EmuNext];

SUBROUTINE;

KnowRBase[LTEMP0];
setBR:
BrHi← T;
BrLo← LTEMP0, return;
TOP LEVEL;


*--------------------------------------------------------------
TOP LEVEL;
knowrbase[LTEMP0];
InsSet[LispInsSet, 1];

opUPCTRACE:
T← (fetch← TSP) + 1;
T← Md, fetch← T; rbase← rbase[Events];
Q← Md, call[SetPCHistAddr];
NextOpCode;

regOP1[377, StackM2BR, opUPCTRACE, noNData];


*--------------------------------------------------------------
* Memory system initialization stubs for lisp


InitMapWarm:
Branch[ResumeEmulator]; * i.e., don’t do anything

*-----------------------------------------------------------
WaitForMapBuf:
* Wait for map operation to complete
* Clobbers nothing
*-----------------------------------------------------------
Subroutine;
PD← T-T-1;
PD← PRef, Branch[., ALU<0];* MapBufBusy is sign bit
Return;