{File name: Process.mc
bj 18-Jan-88 23:44:16 fix comments at PEnd:, comment out MesaIntBr's
bj 7-Jan-88 16:20:50 temporarily put back in stackerr checking code.
Trow 9-Nov-87 21:15:03 Add interrupt check at Enqueue, Reschedule.
Trow 9-Nov-87 20:22:51 Eliminate uTickCount at Scan.
Trow 9-Nov-87 17:22:10 Split memory reference at FreeState.
BJackson 9-Nov-87 14:37:28 quick fix for interrupts.
Trow 5-Nov-87 0:15:49 Fix interrupt at IdleLoop.
Trow 22-Oct-87 21:56:08 Change XwdDisp to XdwDisp.
BJackson 22-Oct-87 3:29:43 use "map.rd" instead of "30".
Trow 13-Oct-87 15:27:07 Reverse targets 1 and 2 of XwdDisp.
Fiala 15-May-86 19:15:45 Fixes for 4MB storage in map. Noted following problems for future:
1) QRead tested XDirtyDisp and proceeded only if flags were ok for a write but did
a read and set map flags only for a read.
2) QWrite formerly or'ed 0A0 into the map flags instead of 0B0 like all other writes;
I changed it to be 30 for the new format, like all other writes.
***Note Kiku also has these bugs (if bugs).
Frandeen September 29, 1981 9:26 AM: Fix CancelBR at MREAbort+1.
Frandeen August 20, 1981 2:20 PM: Fix for new assembler.
Sandman March 26, 1981 10:43 AM to fix timing restrictions
Sandman March 26, 1981 3:53 PM Change Wait to check condition.abortable
Created: February 13, 1981 1:12 PM,
Description: Dandelion Process Microcode,
Author: Sandman,
}
{*****************************************************************************
Definitions
*****************************************************************************}
{Xfer defs}
Set[xtPSwitch, 0];
Set[L2.RSStashr,09];
Set[L2.Pop0Incr1or2, 0 {00xx}]; {also in Jump.mc}
{Offsets and Magic locations}
Set[PSB.link, 0];
Set[PSB.link.vector, 1];
Set[PSB.flags, 1];
Set[PSB.flags.waiting, 2];
Set[PSB.context, 2];
Set[PSB.timeout, 3];
Set[Condition.wakeup, 1];
Set[Condition.abortable, 2];
Set[Monitor.lock, 1];
Set[FaultQueue.condition, 1];
Set[PDAHi, 1];
Set[PDA.ready, 0];
Set[PDA.count, 1];
Set[PDA.timeout, 2];
Set[PDA.state, 10'b];
Set[PDA.interrupt, 20'b];
Set[PDA.lastCV, 36'b];
Set[PDA.fault, 60'b];
Set[PSBSize, 4];
Set[StartPsbM4, 74'b];
{
uPFlags holds opcode dispatch values and general flags. Its encoding:
Bit Meaning
8 0: no Requeue, 1: Requeue occured
9 Interrupt
10 Process Trap
11 PCValid
12-15 Opcode dispatch
}
{Flag values}
Set[pME, 0];
Set[pMR, 1];
Set[pMX, 2];
Set[pMW, 9];
Set[pNC, 4];
Set[pBC, 5];
Set[pREQ, 8];
Set[pInt, 47]; Set[pIntLoc, 7]; {Also in Dandelion.df}
Set[pIdle, 77]; {pIdle MOD 16 = pInt MOD 16 = pIntLoc}
Set[pFault, 3];
Set[pFaultPC, 0D3];
Set[pFaultNoPC, 0E3];
Set[pTrap, 20];
Set[cPCInvalid, 10];
Set[cRequeueFlag, 80];
Set[L3.Normal, 0];
Set[L3.Interrupt, 1];
Set[L3.Fault, 3];
Set[L0.SrcOK, 0];
Set[L0.SrcNull, 1];
Set[L2.Fault, 1];
Set[L2.Trap, 0]; { Same as L2.MRE4}
{PRead Returns}
Set[L2.LP0, 0];
Set[L2.MRE3, 1];
Set[L2.DQ2, 4];
Set[L2.NQ1, 6];
Set[L2.WH2, 7];
Set[L2.NQ3, 8];
Set[L2.RS3, 9];
Set[L2.LP1, 0A];
Set[L2.AS0, 0B];
Set[L2.NQ0, 0C]; {also QRead}
Set[L2.LP2, 0D];
Set[L2.TS0, 0E];
Set[L2.TS2, 0F];
{PMRead Returns}
Set[L2.WH0, 0]; {0 MOD 2}
Set[L2.RS0, 2]; {also PFetch}
Set[L2.DQ1, 3]; {3 MOD 4}
Set[L2.EM2, 4]; {0 MOD 2; also NQRets}
Set[L2.CQ3, 5];
Set[L2.CQ2, 6]; {0 MOD 2}
Set[L2.CQ1, 7]; {3 MOD 4}
{PFetch Returns}
Set[L2.RS0, 2]; {also PMRead}
Set[L2.RS2, 0]; {0 MOD 2}
Set[L2.SP0, 1];
Set[L2.EF0, 3];
Set[L2.LP5, 4];
Set[L2.DQ5, 5];
Set[L2.MW2, 6]; {also PWrite, QRead}
{PWrite Returns}
Set[L2.WH1, 0];
Set[L2.SP1, 1]; {1 MOD 4; paired with L2.ASS0}
Set[L2.PSS0, 2];
Set[L2.AAS0, 3]; {3 MOD 4; contains SP1}
Set[L2.NQ4, 4];
Set[L2.FS0, 5];
Set[L2.MW2, 6]; {also PFetch, QRead}
Set[L2.DQ0, 7]; {3 MOD 4; also QRead}
Set[L2.AAS1, 9];
Set[L2.PSS1, 8];
Set[L2.SP2, 0A];
Set[L2.LP3, 0B];
Set[L2.FS1, 0C];
{QRead Returns}
Set[L2.MRE4, 0]; {Same as L2.Trap}
Set[L2.CQ0, 1];
Set[L2.MRE0, 2];
Set[L2.ME0, 4]; {also QWrite}
Set[L2.EM0, 5];
Set[L2.MW2, 6]; {also PFetch, PWrite}
Set[L2.DQ0, 7]; {3 MOD 4; also PWrite}
Set[L2.BC0, 8];
Set[L2.RQ0, 9];
Set[L2.F0, 0A];
Set[L2.NQ0, 0C]; {also PRead}
{QWrite Returns}
Set[L2.ME0, 4]; {also QRead}
Set[L2.MRE1, 1]; {also PRead}
Set[L2.DQ4, 2]; {0 MOD 2; DQm; paired with DQl}
{DQl has 3}
Set[L2.NQ2, 0];
Set[L2.MW3, 5]; {1 MOD 2}
Set[L2.EM1, 7]; {3 MOD 4}
Set[L2.INT2, 8];
Set[L2.CQ4, 9]; {1 MOD 2}
Set[L2.NQ8, 0A];
{Requeue Returns}
Set[rhT.PT, 0]; {Must be zero}
Set[rhT.BCast, 1];
Set[rhT.IntNN, 2];
Set[rhT.TS, 3];
{4 contains EMa}
Set[rhT.EM, 5]; {paired with EMa}
Set[rhT.Fault, 6];
Set[rhT.FaultNN, 7];
{*****************************************************************************
Opcodes
*****************************************************************************}
@ME: Rx ← pME, push, GOTO[Check1], c1, opcode[1];
@MR: Rx ← pMR, push, GOTO[Check2], c1, opcode[2];
@MW: Rx ← pMW, c1, opcode[3];
MWx: TT ← ErrnIBnStkp, push, XDisp, c2;
{bj} STK ← TOS, pop, DISP4[Test2, 8], c3;
{ STK ← TOS, pop, BRANCH[Short2of2, Long2of2, 0D], c3; }
@MD: Rx ← pMX, push, GOTO[Check1], c1, opcode[4];
@NC: Rx ← pNC, push, GOTO[Check1], c1, opcode[5];
@BC: Rx ← pBC, push, GOTO[Check1], c1, opcode[6];
@REQ: Rx ← pREQ, GOTO[MWx], c1, opcode[7];
Check1: TT ← ErrnIBnStkp, XDisp, c2;
{ rhT ← 0, STK ← TOS, BRANCH[Long1of1, Short1of1, 0E], c3;
Long1of1: T ← STK, pop, GOTO[Got1], c1;
Short1of1: T ← UvMDS, GOTO[Got1], c1;
}
{bj} rhT ← 0, STK ← TOS, DISP4[Test1, 8], c3;
Test1: T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[ 8,10,Test1];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[ 9,10,Test1];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0A,10,Test1];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0B,10,Test1];
T ← STK, pop, GOTO[Got1], c1, at[0C,10,Test1];
T ← UvMDS, GOTO[Got1], c1, at[0D,10,Test1];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0E,10,Test1];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0F,10,Test1];
{bj}
Got1: UvQ1Hi ← T, c2;
T ← STK, pop, c3;
UvQ1 ← T, pop, GOTO[SaveRegs], c1;
{bj}
BadStack: TT ← UvPCpage, L1←0, GOTO[StashPC0], c2;
Check2: TT ← ErrnIBnStkp, XDisp, c2;
rhT ← 0, STK ← TOS, DISP4[Test2, 8], c3;
Test2: T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[ 8,10,Test2];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[ 9,10,Test2];
T ← STK, pop, GOTO[Got2], c1, at[0A,10,Test2];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0B,10,Test2];
T ← UvMDS, GOTO[Got2], c1, at[0C,10,Test2];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0D,10,Test2];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0E,10,Test2];
T ← sStackError, L2←L2.TRAPStashr, GOTO[BadStack], c1, at[0F,10,Test2];
{bj}
{
Check2: TT ← ErrnIBnStkp, XDisp, c2;
rhT ← 0, STK ← TOS, BRANCH[Short2of2, Long2of2, 0D], c3;
Long2of2: T ← STK, pop, GOTO[Got2], c1;
Short2of2: T ← UvMDS, GOTO[Got2], c1;
}
Got2: UvQ2Hi ← T, c2;
T ← STK, pop, c3;
UvQ2 ← T, c1;
uStkDepth ← TT, YDisp, c2;
rhT ← 0, BRANCH[Short1of2, Long1of2, 0D], c3;
Long1of2: T ← STK, pop, GOTO[Got1], c1;
Short1of2: T ← UvMDS, GOTO[Got1], c1;
SaveRegs: UrL ← L, c2;
UrG ← G, G ← rhG, c3;
UrGHi ← G, c1;
UrPC ← PC, PC ← rhPC, c2;
UrPCHi ← PC, c3;
L0←L0.SrcOK, G ← PDA.lastCV{for interrupts}, c1;
uPFlags ← Rx, YDisp, L3←L3.Normal, c2;
rhPC ← PDAHi, DISP3[Enter], c3;
PRestart: uFaultParm0 ← G, c3;
TOS ← ~uStkDepth, c1;
stackP ← TOS, TOS ← pFault, c2;
TT ← uPFlags, XDisp, c3;
G ← rhG, L2Disp, BRANCH[PFa, PFb, 7], c1;
PFa: uFaultParm1 ← G, pop, BRANCH[PTrap, PFault], c2;
PFb: uFaultParm1 ← G, BRANCH[PTrap, PFault], c2;
PTrap: uPFlags ← Rx, GOTO[PTail1], c3;
PFault: uPFlags ← TOS, GOTO[Fault], c3;
{*****************************************************************************
Monitor Entry opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of monitor lock to be entered
Exit:
*****************************************************************************}
Enter: uStkDepth ← TT, L2←L2.ME0{QRRets, QWRets}, c1, at[pME,10,Enter];
G ← UvQ1, CALL[Q1Read3], c2;
Map ← UvQ2 ← [rhG, G], push, BRANCH[MEUnLocked, MELocked, 2], c1, at[L2.ME0,10,QRRets];
MEUnLocked:
T ← T or Monitor.lock, push, CALL[QWrite3], c2;
TOS ← 1, c3, at[L2.ME0,10,QWRets];
STK ← TOS, pop, GOTO[PTail2]{L not used}, c1;
MELocked: PC ← uPSB, pop, c2;
EnterFailed:
L ← UvQ1Hi, L2←L2.EF0{PFRets}, c3;
Map ← Q ← [rhPC, PC+PSB.link], c1;
rhG ← PDAHi, G ← PDA.ready, CALL[PFetch3], c2;
Rx ← MD or Rx, c3, at[L2.EF0,10,PFRets];
MAR ← [rhTT, PC+PSB.link], c1;
MDR ← Rx or u8000, L2←L2.DQ0{QRRets}, c2;
CallDQ: UvQ2Hi ← L, CALL[QRead1], c3;
{*****************************************************************************
Monitor ReEnter opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of monitor lock to be entered
UvQ2Hi,,UvQ2 hold virtual address of condition variable just left
Exit:
*****************************************************************************}
ReEnter: uStkDepth ← TT, L2←L2.MRE0{QRRets}, c1, at[pMR,10,Enter];
G ← UvQ1, CALL[Q1Read3], c2;
PC ← uPSB, BRANCH[MREUnLocked, MRELocked, 2], c1, at[L2.MRE0,10,QRRets];
MRELocked: UvQ2 ← G, GOTO[EnterFailed], c2;
MREUnLocked:
G ← UvQ2, c2;
rhG ← UvQ2Hi, c3;
Map ← [rhG, G], L2←L2.CQ0{QRRets}, c1;
TOS ← T, CALL[QRead3], c2;
PC ← uPSB, c3, at[pMR,10,CQRets];
Map ← Q ← [rhPC, PC or PSB.flags], L2←L2.MRE3{PRRets}, c1;
Rx ← uPMask, CALL[PRead3], c2;
MAR ← [rhTT, PC+PSB.flags], L2←L2.MRE4{QRRets}, c1, at[L2.MRE3,10,PRRets];
MDR ← Ybus ← T and ~Rx, YDisp, CANCELBR[$,0], c2;
AbortNotOK: rhG ← UvQ1Hi, BRANCH[MRENoAbort, MREAbort, 0E], c3;
MRENoAbort: Map ← [rhG, G], L2←L2.MRE1{QWRets}, c1; {write locked monitor}
T ← TOS or Monitor.lock, push, CALL[QWrite3], c2;
TOS ← 1, push, c3, at[L2.MRE1,10,QWRets];
STK ← TOS, pop, GOTO[PTail2]{L not used}, c1;
MREAbort: Rx ← pTrap, CALL[Q2Read2], c1;
[] ← T and Condition.abortable, NZeroBr, CANCELBR[$,3], c1, at[L2.MRE4,10,QRRets];
G ← UvQ1, BRANCH[AbortNotOK, PRestart], c2;
{*****************************************************************************
Monitor Exit and Depart opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of monitor lock to be exited
Exit:
*****************************************************************************}
Depart: uStkDepth ← TT, L2←L2.EM0{QRRets}, c1, at[pMX,10,Enter];
G ← UvQ1, CALL[Q1Read3], c2;
MXDExit: L ← UrL, GOTO[PTail2], c1;
{*****************************************************************************
Exit Monitor Subroutine
Entry:
rhG,,G hold virtual address of monitor lock to be exited
T has monitor lock
Exit:
*****************************************************************************}
ExitMonitor:
Map ← [rhG, G], L2←L2.EM1{QWRets}, CANCELBR[$,3], c1, at[L2.EM0,10,QRRets];
T ← T and ~Monitor.lock, CALL[QWrite3], c2;
T ← T and Q, ZeroBr, rhT ← rhT.EM{NQRets}, c3, at[L2.EM1,10,QWRets];
Map ← Q ← [rhPC, T+PSB.link], L2←L2.EM2{PMRets}, BRANCH[EMa, EMb], c1;
EMa: UvQ2 ← PDA.ready, CALL[PMRead3], c2;
L ← PDAHi, c1, at[L2.EM2,10,PMRRets];
PC ← T, L2←L2.DQ0{QRRets}, CALL[CallDQ], c2;
EMb: Xbus ← uPFlags, XLDisp, c2, at[rhT.EM,10,NQRets];
PC ← uPSB, BRANCH[MXDExit, MWExit, 2], c3;
{*****************************************************************************
Monitor Exit and Wait opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of monitor lock to be exited
UvQ2Hi,,UvQ2 hold virtual address of condition on which to wait
TOS holds timeout value
Exit:
*****************************************************************************}
Wait: uTicks ← TOS, L2←L2.CQ0{QRRets}, CALL[CleanSecond], c1, at[pMW,10,Enter];
rhG ← UvQ1Hi, c3, at[pMW,10,CQRets];
Rx ← UvQ2, c1;
UvQTemp ← Rx, c2;
Rx ← UvQ2Hi, c3;
Map ← [rhG, G], L2←L2.EM0{QRRets}, c1;
UvQTempHi ← Rx, CALL[QRead3], c2;
MWExit: rhG ← G ← UvQTempHi, c1;
UvQ2Hi ← G, c2;
G ← UvQTemp, L2←L2.MW2{QRRets, PFRets, PWRets}, c3;
Map ← UvQ2 ← [rhG, G], CALL[QRead2], c1;
Map ← Q ← [rhPC, PC or PSB.flags], CANCELBR[PFetch2,3], c1, at[L2.MW2,10,QRRets];
TOS ← MD, XLDisp, c3, at[L2.MW2,10,PFRets];
[] ← T and Condition.abortable, ZeroBr, BRANCH[MWa, MWb, 2], c1;
MWa: Rx ← uPTC, CANCELBR[MWd, 1], c2;
MWb: Rx ← uPTC, BRANCH[MWc, MWd], c2;
MWc: GOTO[PTail1], c3;
MWd: Xbus ← T LRot0, XLDisp, c3;
TT ← uTicks, ZeroBr, BRANCH[MWNoWW, MWWW, 2], c1;
MWNoWW: rhG ← PDAHi, G ← PDA.ready, BRANCH[MWMakeTime, MWHaveTime], c2;
MWHaveTime:
Xbus ← 0, XDisp, GOTO[MWWriteTime], c3;
MWMakeTime:
TT ← TT + Rx, ZeroBr, c3;
MWWriteTime:
Map ← Q ← [rhPC, PC or PSB.timeout], BRANCH[MWTimeOK, MWIncTime], c1;
MWTimeOK: T ← TT, rhT ← rhT.PT{NQRets}, CALL[PWrite3], c2;
MWIncTime: T ← 0 + 0 + 1, rhT ← rhT.PT{NQRets}, CALL[PWrite3], c2;
T ← TOS or PSB.flags.waiting, L2←L2.DQ0{QRRets, PWRets}, c3, at[L2.MW2,10,PWRets];
Map ← Q ← [rhPC, PC or PSB.flags], CALL[PWrite2];
CALL[QRead1], c3, at[L2.DQ0,10,PWRets];
MWWW: L2←L2.MW3{QWRets}, CANCELBR[$,1], c2;
T ← T and ~Condition.wakeup, CALL[QWrite1], c3;
GOTO[PTail1], c3, at[L2.MW3,10,QWRets];
{*****************************************************************************
Notify opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of condition to be notified
Exit:
*****************************************************************************}
Notify: uStkDepth ← TT, L2←L2.CQ0{QRRets}, CALL[CleanFirst], c1, at[pNC,10,Enter];
T ← T and Q, ZeroBr, L2←L2.WH0{PMRRets}, c3, at[pNC,10,CQRets];
Map ← Q ← [rhPC, T+PSB.link], BRANCH[DoNotify, NoNotify], c1;
DoNotify: L ← PDAHi, CALL[PMRead3], c2;
NoNotify: TT ← uPFlags, GOTO[PTail3], {L not used} c2;
{*****************************************************************************
Broadcast opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of condition to be broadcast
Exit:
*****************************************************************************}
BCast: uStkDepth ← TT, L2←L2.CQ0{QRRets}, CALL[CleanFirst], c1, at[pBC,10,Enter];
BCe: rhT ← rhT.BCast{NQRets}, T ← T and Q, ZeroBr, c3, at[pBC,10,CQRets];
BCd: Map ← Q ← [rhPC, T+PSB.link], L2←L2.WH0{PMRRets}, BRANCH[BCa, BCb], c1;
BCa: L ← PDAHi, CALL[PMRead3], c2;
BCb: rhT ← 0, GOTO[PTail0], c2;
BCc: G ← UvQ1, L2←L2.BC0{QRRets}, c2, at[rhT.BCast,10,NQRets];
rhG ← UvQ1Hi, CALL[QRead1], c3;
CANCELBR[$,3], c1, at[L2.BC0,10,QRRets];
GOTO[BCe], c2;
{*****************************************************************************
WakeHead subroutine
Entry: rhG, G contains virtual address of condition to wake
T contains queue.tail.link
L has PDAHi
Exit:
*****************************************************************************}
WakeHead: Map ← Q ← [rhPC, T or PSB.timeout], L2←L2.WH1{PWRets}, c1, at[L2.WH0,10,PMRRets];
T ← 0, UvQ2 ← PDA.ready, CALL[PWrite3], c2;
PC ← Q - PSB.timeout, L2←L2.WH2{PRRets}, c3, at[L2.WH1,10,PWRets];
Map ← Q ← [rhPC, PC or PSB.flags], CALL[PRead2], c1;
MAR ← [rhTT, PC+PSB.flags], c1, at[L2.WH2,10,PRRets];
MDR ← T and ~PSB.flags.waiting, L2←L2.DQ0{QRRets}, CANCELBR[CallDQ,0], c2;
{*****************************************************************************
Requeue opcode
Entry:
UvQ1Hi,,UvQ1 hold virtual address of source queue
UvQ2Hi,,UvQ2 hold virtual address of destination queue
TOS has process
Exit:
*****************************************************************************}
Requeue: PC ← TOS, L2←L2.RQ0{QRRets}, CALL[Q2Read2], c1, at[pREQ,10,Enter];
G ← UvQ1, L2←L2.DQ0{QRRets}, CANCELBR[$,3], c1, at[L2.RQ0,10,QRRets];
rhG ← UvQ1Hi, [] ← G or UvQ1Hi, ZeroBr, c2;
L0←L0.SrcOK, BRANCH[QRead1, RQa], c3;
RQa: Map ← Q ← [rhPC, PC+PSB.link], L2←L2.DQ1{PMRRets}, GOTO[DQx], c1;
{*****************************************************************************
Dequeue subroutine:
Entry:
rhT # 0 => src = NIL
UvQ1Hi,,UvQ1 has src
PC has process
Exit:
Returns to NQ
Register Usage
PC = process
T = temp
TOS = prev
Rx has queue
UvLPhigh has psb.link
*****************************************************************************}
Dequeue: Map ← Q ← [rhPC, PC+PSB.link], L2←L2.DQ1{PMRRets}, CANCELBR[$, 3], c1, at[L2.DQ0,10,QRRets];
DQx: Rx ← T, CALL[PMRead3], c2; {Fetch[@psb.link]}
TOS ← uPFlags, c1, at[L2.DQ1,10,PMRRets];
TOS ← TOS or cRequeueFlag, c2;
uPFlags ← TOS, L0Disp, c3;
[] ← T - PC, ZeroBr, BRANCH[DQa, DQb] c1;
DQa: TOS ← 0, uPsbLink ← T, BRANCH[DQc, DQd], c2;{src#NIL}
DQb: TOS ← 0, uPsbLink ← T, BRANCH[DQe, DQf], c2;{src=NIL}
DQc: TOS ← Rx and uPMask2, GOTO[DQg], c3;{link.next#psb}
DQd: T ← Rx and uPMask2, GOTO[DQj], c3;{link.next=psb}
DQe: TOS ← PC, GOTO[DQg], c3;{link.next#psb}
DQf: GOTO[DQk], c3;{link.next=psb}
DQg: Map ← Q ← [rhPC, TOS+PSB.link], L2←L2.DQ2{PRRets}, c1;
DQgh: TOS ← Q, CALL[PRead3], c2;
Q ← T and uPMask, c1, at[L2.DQ2,10,PRRets];
[] ← Q - PC, ZeroBr, c2;
T ← T and ~uPMask, BRANCH[DQh, DQi], c3;
DQh: Map ← Q ← [rhPC, Q+PSB.link], L2←L2.DQ2{PRRets}, GOTO[DQgh], c1;
DQi: MAR ← [rhTT, TOS+PSB.link], c1;
MDR ← T or uPsbLink, L0Disp, c2;
T ← Rx and uPMask2, BRANCH[DQj, DQk], c3;
DQj: [] ← T - PC, ZeroBr, c1;
T ← Rx and ~uPMask2, BRANCH[DQm, DQl], c2;
DQl: T ← TOS or T, L2←L2.DQ4{QWRets}, c3;
Map ← [rhG, G], CALL[QWrite2], c1;
DQk: Map ← Q ← [rhPC, PC or PSB.flags], L2←L2.DQ5{PFRets}, c1;
T ← ~uPMask, CALL[PFetch3], c2;
T ← MD and T, c3, at[L2.DQ5,10,PFRets];
MAR ← [rhTT, PC+PSB.flags], c1;
MDR ← T or uPsbLink, CANCELBR[$,0], c2;
DQm:
{db} G ← UvQ2, {MesaIntBr,}{bj} GOTO[Enqueue], c3, at[L2.DQ4,10,QWRets];
{*****************************************************************************
Enqueue subroutine:
Entry:
UvQ2Hi,,UvQ2 has dst
PC has process
rhT has return
Exit:
Register Usage
L has process.priority
T = temp
TOS = prev
Rx = queue, prev
*****************************************************************************}
Enqueue:
Map ← Q ← [rhPC, PC+PSB.link], L2←L2.NQ0{QRRets, PRRets},
BRANCH[$, EnqueueInt], c1;
Rx ← uPPMask, CALL[PRead3], c2;
UreturnPC ← T, c1, at[L2.NQ0,10,PRRets];
L ← T and uPPMask, CALL[Q2Read3], c2;
Q ← T and uPMask, ZeroBr, CANCELBR[$,3], c1, at[L2.NQ0,10,QRRets];
T ← T and ~uPMask, BRANCH[NQb, NQc], c2;
NQc: T ← T or PC, L2←L2.NQ8{QWRets}, c3;
Map ← [rhG, G], c1;
Q ← ~uPMask, CALL[QWrite3], c2;
T ← Q and UreturnPC, L2←L2.NQ4{PWRets}, c3, at[L2.NQ8,10,QWRets];
Map ← Q ← [rhPC, PC+PSB.link], c1;
T ← T or PC, CALL[PWrite3], c2;
NQb: TOS ← Q, uTemp ← T, L2←L2.NQ1{PRRets}, c3;
Map ← Q ← [rhPC, TOS+PSB.link], c1;
uPsbLink ← T or PC, CALL[PRead3], c2;
Rx ← T and Rx, Ufsi ← T, c1, at[L2.NQ1,10,PRRets];
[] ← Rx - L, CarryBr, L2←L2.NQ3{PRRets}, c2;
NQi: Rx ← T and uPMask, BRANCH[NQd, NQe], c3;
NQe: Map ← [rhG, G], L2←L2.NQ2{QWRets}, c1;
T ← uPsbLink, CALL[QWrite3], c2;
Q ← ~uPMask, GOTO[NQg], c3, at[L2.NQ2,10,QWRets];
NQd: Map ← Q ← [rhPC, Rx+PSB.link], CALL[PRead2], c1;
TT ← T and uPPMask, c1, at[L2.NQ3,10,PRRets];
[] ← L - TT - 1, CarryBr, c2;
Q ← ~uPMask, BRANCH[NQf, NQg], c3;
NQf: TOS ← Rx, c1;
Ufsi ← T, GOTO[NQi], c2;
NQg: Map ← [rhPC, PC+PSB.link], c1;
T ← Q and UreturnPC, c2;
TT ← rhTT ← MD, c3;
MAR ← [rhTT, PC+PSB.link], c1;
MDR ← T or Rx, c2;
T ← Q and Ufsi, L2←L2.NQ4{PWRets}, c3;
Map ← Q ← [rhPC, TOS+PSB.link], c1;
T ← T or PC, CALL[PWrite3], c2;
Xbus ← rhT, XDisp, c3, at[L2.NQ4,10,PWRets];
NQRet: L ← UrL, DISP2[NQRets], c1;
{*****************************************************************************
Reschedule
Entry:
uPFlags.0: reSchedulePending BOOLEAN
uPFlags.1..3: 0: Opcode, 2: Trap, 4: Interrupt, 7: Idle,
5: Fault, 6: FaultNoPC
Exit:
Register Usage
*****************************************************************************}
PTail0: Noop, c3;
PTail1: L ← UrL, c1;
PTail2: TT ← uPFlags, c2, at[rhT.PT,10,NQRets];
PTail3: G ← UrG, L0←L0.JRemap, c3;
rhG ← UrGHi, c1;
PC ← UrPC, L2←L2.RSStashr{StashRets}, c2;
rhRx ← TT LRot12, XDisp, c3;
PTNormal:
DISP4[PEnd], c1;
PEnd:
PC ← PC + PC16, IBDisp, GOTO[SimpleEnd], c2, at[0,10];
{***not possible, c2, at[1,10,PEnd];}
T ← sProcessTrap, GOTO[Trapc3], c2, at[2,10,PEnd];
{***not possible, c2, at[3,10,PEnd];}
[] ← PC - 1, PgCrOvDisp, push, L1←L1r.Refill, GOTO[IntRet], c2, at[4,10,PEnd];
{***not possible to have fault but no requeue, c2, at[5,10,PEnd];}
{***not possible to have fault but no requeue, c2, at[6,10,PEnd];}
GOTO[SPIdle], {was in idle loop} c2, at[7,10,PEnd];
TT ← UvPCpage, L1 ← 1, GOTO[StashPC1], c2, at[8,10,PEnd];
{***not possible, c2, at[9,10,PEnd];}
T ← sProcessTrap, GOTO[Trapc3], c2, at[0A,10,PEnd];
{***not possible, c2, at[0B,10,PEnd];}
[] ← PC - 1, PgCrOvDisp, push, L1←L1r.Refill, GOTO[IntRet], c2, at[0C,10,PEnd];
TT ← UvPCpage, L1 ← 0, GOTO[StashPC0], c2, at[0D,10,PEnd];
{db} {MesaIntBr,}{bj} GOTO[Reschedule], c2, at[0E,10,PEnd];
GOTO[SPIdle], {was in idle loop} c2, at[0F,10,PEnd];
SimpleEnd: rhPC ← UrPCHi, DISPNI[OpTable], c3;
Reschedule:
{db} PC ← uPSB, BRANCH[SaveProcess, RescheduleInt], c3, at[L2.RSStashr,10,StashRets];
IntRet: TOS ← STK, pop, BRANCH[PCPageOK, IncPage, 1], c3;
PCPageOK: Xbus ← rhRx, XDisp, Q ← 0, L2←L2.Pop0Incr1or2, c1;
PageOKx: TT ← UvPCpage, BRANCH[FastEasy, SlowEasy, 7], c2;
FastEasy: rhTT ← UvChigh, GOTO[RReMap], c3;
SlowEasy: Q ← UvC, L1←0, c3;
L2←L2.RSStashr{StashRets}, GOTO[StashPCa1], c1;
IncPage: Rx ← LRot1 ErrnIBnStkp, c1;
Xbus ← Rx LRot0, XwdDisp, c2;
TT ← UvPCpage, DISP2[IBState], c3;
IBState:
Xbus ← rhRx, XDisp, Q ← 0, L2←L2.Pop0Incr1or2, GOTO[PageOKx], c1, at[0,4,IBState];
Xbus ← rhRx, XDisp, Q ← 0, L2←L2.Pop0Incr1or2, GOTO[PageOKx], c1, at[1,4,IBState];
Xbus ← rhRx, XDisp, Q ← 0, L2←L2.Pop0Incr1or2, GOTO[PageOKx], c1, at[2,4,IBState];
Xbus ← rhRx, XDisp, Q ← 0, L2←L2.Pop0Incr1or2, c1, at[3,4,IBState];
TT ← TT + 0FF + 1, BRANCH[FastEasy, SlowEasy, 7], c2;
SPIdle: G ← PDA.lastCV{for interrupts}, L2←L2.RS0{PMRRets, PFRets}, c3, at[L2.SP2,10,PWRets];
SPRet: Map ← Q ← [rhPC, PDA.ready], CALL[PMRead2], c1;
Map ← Q ← [rhPC, T+PSB.link], ZeroBr, c1, at[L2.RS0,10,PMRRets];
UreturnPC ← T, BRANCH[PFetch3, BusyWaita], c2;
PC ← MD and Q, L2←L2.RS2{PFRets}, c3, at[L2.RS0,10,PFRets];
RSe:
{bj}
RegDef[uHelp, U, 5F];
z1: [] ← uHelp xor PC, ZeroBr, ,c1;
z2: BRANCH[z3a, z3b], ,c2;
z3a: GOTO[z4], ,c3;
z3b: GOTO[z4], ,c3;
z4:
Map ← Q ← [rhPC, PC{+PSB.link}], uPSB ← PC, CALL[PFetch2], c1;
T ← MD, XLDisp, c3, at[L2.RS2,10,PFRets];
uPsbLink ← T, BRANCH[RSa, RSb, 2], c1;
RSb: GOTO[RSc], c2;
RSa: T ← T and uPPMask, c2;
T ← T LRot4, L2←L2.RS3{PRRets}, c3;
Map ← Q ← [rhPC, PDA.state or T], CALL[PRead2], c1;
[] ← T, ZeroBr, L2←L2.RS2{PFRets}, c1, at[L2.RS3,10,PRRets];
[] ← UreturnPC xor PC, ZeroBr, BRANCH[RSc, RSd], c2;
RSd: PC ← uPsbLink and Q, BRANCH[RSe, BusyWaitb], c3;
RSc: uPCValid ← 0, L ← 0, CANCELBR[LoadProcess,1], c3;
BusyWaita: Noop, c3;
BusyWaitb: T ← pIdle, c1;
[] ← uWDC, NZeroBr, c2;
IdleLoop:
uPFlags ← T, MesaIntBr, BRANCH[$, WakeError], c3;
{db} BRANCH[NoInt, IdleInt {InterruptsDaybreak}], c1;
NoInt:
GOTO[IdleLoop], c2;
WakeError:
T ← sWakeupError, CANCELBR[$, 1], c1;
GOTO[TrapGo], c2;
{*****************************************************************************
SaveProcess subroutine
Entry:
L3 has preemption booleon: 0 => FALSE, 1 => TRUE
Exit:
Register Usage
*****************************************************************************}
SaveProcess: Map ← Q ← [rhPC, PC+PSB.link], L2←L2.SP0{PFRets}, c1;
TOS ← PSB.link.vector, CALL[PFetch3], c2;
T ← MD or TOS, L3Disp, c3, at[L2.SP0,10,PFRets];
MAR ← [rhTT, PC+PSB.link], BRANCH[SPa, SPb, 2], c1;
SPa: MDR ← T and ~PSB.link.vector, GOTO[SPc], c2;
SPb: MDR ← T, T ← T and uPPMask, c2;
T ← T LRot4, GOTO[AllocAndSave], c3;
AASRet: T ← TOS, L2←L2.SP2{PWRets}, c3, at[L2.SP1,10,PWRets];
SPd: Map ← Q ← [rhPC, PC or PSB.context], CALL[PWrite2], c1;
SPc: T ← UvL, L2←L2.SP2{PWRets}, GOTO[SPd], c3;
{*****************************************************************************
LoadProcess subroutine
Entry: Ufsi has state vector of saved process if fault
L has 0
Exit: Returns to EFCHaveLink
Register Usage
*****************************************************************************}
LoadProcess:
Map ← Q ← [rhPC, PC or PSB.context], L2←L2.LP0{PRRets}, c1;
rhT ← xtPSwitch, stackP ← 0, CALL[PRead3], c2;
Q ← T + State.word, c1, at[L2.LP0,10,PRRets];
Xbus ← uPsbLink, TOS ← T, XLDisp, L2←L2.LP2{PRRets}, c2;
Rx ← uPsbLink, NegBr, BRANCH[LPa, LPb, 2], c3;
LPa: MAR ← [rhTT, PC+PSB.link], BRANCH[LPc, LPd], c1;
LPd: MDR ← Rx and ~u8000, push, c2;
push, c3;
STK ← 0, pop, c1;
LPc: TT ← T, c2;
Ybus ← TT, YDisp, c3;
LPe: Map ← Q ← [rhMDS, TT], UvL ← TT, L0←L0.XFStartRead, DISP4[XferFrame, 0C], c1;
LPb: Map ← Q ← [rhPC, Q], CANCELBR[PRead2, 1], c1;
FSRet: TOS ← TOS + State.frame, L2←L2.LP5{PFRets}, c3, at[L2.FS0,10,PWRets];
Map ← Q ← [rhPC, TOS], CALL[PFetch2], c1;
TT ← MD, XDisp, GOTO[LPe], c3, at[L2.LP5,10,PFRets];
{*****************************************************************************
FreeState subroutine
Entry: T has priority
TOS has state
Exit:
Register Usage
T, Q, rhTT, TT, TOS
*****************************************************************************}
{db}
FreeState:
Map ← Q ← [rhPC, PDA.state or T], c1;
Noop, c2;
TT ← rhTT ← MD, c3;
MAR ← [rhTT, Q + 0], c1;
Noop, c2;
T ← MD, c3;
MAR ← [rhTT, Q + 0], c1;
MDR ← TOS, c2;
L2 ← L2.FS0 {PWRets}, c3;
Map ← Q ← [rhPC, TOS], c1;
stackP ← uStkDepth, CALL[PWrite3], c2;
{db}
{*****************************************************************************
AllocAndSave subroutine
Entry: T has priority
Exit: TOS has state
Register Usage
T, Q, rhRx, Rx, TOS, uTemp
*****************************************************************************}
AllocAndSave:
Map ← Q ← [rhPC, PDA.state or T], L2←L2.AS0{PRRets}, c1;
uTemp ← Q, CALL[PRead3], c2;
Map ← Q ← [rhPC, T], c1, at[L2.AS0,10,PRRets];
TOS ← Q, c2;
rhRx ← Rx ← MD, c3;
MAR ← [rhRx, T+0], c1;
T ← uTemp, c2;
Rx ← MD, c3;
MAR ← [rhTT, T+0], c1;
MDR ← Rx, T ← Q + State.word, c2;
G ← ~ErrnIBnStkp, c3;
Map ← Q ← [rhPC, T], L2←L2.PSS0{PWRets}, c1;
T ← 0F and G, CALL[PWrite3], c2;
[] ← T - cSSm2, CarryBr, c3, at[L2.PSS0,10,PWRets];
stackP ← Rx ← T + 2, BRANCH[PSSstore, PSScarry], c1;
PSScarry: stackP ← Rx ← cSS, c2;
Noop, c3;
Noop, c1;
PSSstore: Q ← Q - State.word, c2;
Q ← Q + Rx, push, L2←L2.PSS1{PWRets}, c3;
PSSloop: Map ← Q ← [rhPC, Q-1], BRANCH[$,PSSRet], c1;
T ← STK, pop, CALL[PWrite3], c2;
Rx ← Rx - 1, ZeroBr, GOTO[PSSloop], c3, at[L2.PSS1,10,PWRets];
PSSRet: T ← UvL, L3Disp, c2;
Q ← TOS + State.frame, L2←L2.SP1{PWRets}, BRANCH[AASa, AASb, 1], c3;
AASa: Map ← Q ← [rhPC, Q], CALL[PWrite2], c1;
AASb: Map ← Q ← [rhPC, Q], CALL[PWrite2], c1;
T ← uFaultParm0, L2←L2.AAS1{PWRets}, c3, at[L2.AAS0,10,PWRets];
Map ← Q ← [rhPC, Q+1], CALL[PWrite2], c1;
T ← uFaultParm1, L2←L2.SP1{PWRets}, c3, at[L2.AAS1,10,PWRets];
Map ← Q ← [rhPC, Q+1], CALL[PWrite2], c1;
{*****************************************************************************
PLoadStack subroutine
Entry: T has state.word
TOS has stack pointer
Exit: TOS has stack pointer
T has priority
Returns to FreeState
Register Usage
T, rhTT, TT, Q, Rx, TOS
*****************************************************************************}
PLoadStack: uStkDepth ← T, c1, at[L2.LP2,10,PRRets];
rhTT ← T LRot8, c2;
TT ← rhTT, c3;
T ← T and 0F, c1;
Q ← T - cSSm1, CarryBr, c2;
Q ← Q + cSSp1, BRANCH[LPload, LPcarry], c3;
LPcarry: Q ← cSS, c1;
Noop, c2;
Noop, c3;
LPload: UBrkByte ← TT, Rx ← Q, c1;
Q ← TOS - 1, GOTO[PLPa], c2;
LPloop: Map ← Q ← [rhPC, Q+1], c1;
G ← uPsbLink, c2;
rhTT ← TT ← MD, c3;
MAR ← [rhTT, Q+0], c1;
Rx ← Rx - 1, ZeroBr, c2;
PLPa: T ← MD, STK ← T, push, BRANCH[LPloop, LPdone], LOOPHOLE[mdok], c3;
LPdone: STK ← T, c1;
T ← G LRot4, c2;
T ← T and 7, GOTO[FreeState], c3;
{*****************************************************************************
Interrupt Handler
Entry:
CALLing sequence has done Rx ← pInt, GOTO[SaveRegs];
Exit:
Register Usage
*****************************************************************************}
Interrupt:
Rx ← uWP, c1, at[pIntLoc,10,Enter];
{db} {ClrIntErr}, uWP ← 0, L3←L3.Interrupt, c2;
CVLoop:
Rx ← RShift1 Rx, Xbus ← Rx LRot0, XLDisp, c3;
uWW ← Rx, NZeroBr, BRANCH[CheckNext, IntThisCV, 2], c1;
IntThisCV:
Q ← PDA.interrupt + G, CANCELBR[$,1], c2;
[] ← G, rhG ← PDAHi, ZeroBr, c3;
{db} Map ← UvQ1 ← [rhG, Q], BRANCH[Int, Scan], c1;
Int:
uIntLevel ← G, G ← Q, L2←L2.CQ0{QRRets}, CALL[QRead3], c2;
rhT ← rhT.IntNN{NQRets}, Q ← T and Q, ZeroBr, c3, at[pIntLoc,10,CQRets];
Map ← Q ← [rhPC, Q+PSB.link], L2←L2.WH0{PRRets}, BRANCH[INTa, INTb], c1;
INTa:
L ← PDAHi, L0←L0.SrcOK, CALL[PMRead3], {Do Naked Notify} c2;
INTb:
L2←L2.INT2{QWRets}, c2;
T ← T or 1, CALL[QWrite1], c3;
INTc:
G ← uIntLevel, c3, at[L2.INT2,10,QWRets];
Rx ← uWW, NZeroBr, c1;
CheckNext:
G ← G - 2, BRANCH[IntDone, CVLoop], c2;
IntDone:
T ← UrPC, GOTO[PTail1], c3;
IntDidNN:
GOTO[INTc], c2, at[rhT.IntNN,10,NQRets];
{*****************************************************************************
Fault Handler
Entry: T has FaultIndex*2
parameters stored in uFaultParm0 and uFaultParm1
CALLing sequence has done Rx ← pFault, GOTO[SaveRegs];
Exit:
Register Usage
*****************************************************************************}
Fault: UvQ2 ← T, L3←L3.Fault, c1, at[pFault,10,Enter];
rhG ← G ← PDAHi, c2;
UvQ2Hi ← G, G ← PDA.ready, c3;
PC ← uPSB, L2←L2.DQ0{QRRets}, c1;
rhT ← rhT.Fault{NQRets}, c2;
CALL[QRead1], c3;
rhT ← rhT.FaultNN{NQRets}, c2, at[rhT.Fault,10,NQRets];
G ← UvQ2, L2←L2.F0{QRRets}, c3;
Map ← G ← [rhG, G+FaultQueue.condition], c1;
UvQ2 ← PDA.ready, CALL[QRead3], c2;
Map ← Q ← [rhPC, T and Q], ZeroBr, CANCELBR[$,3], c1, at[L2.F0,10,QRRets];
L ← PDAHi, L2←L2.WH0{PMRRets}, BRANCH[PMRead3, Fb], c2;
Fb: PC ← Q or Condition.wakeup, c3;
MAR ← [rhTT, G+0], c1;
MDR ← PC, T ← uPCValid, NZeroBr, c2;
Fc: BRANCH[Fd, Fe], c3;
Fd: TT ← pFaultNoPC, GOTO[Ff], c1;
Fe: TT ← pFaultPC, GOTO[Ff], c1;
Ff: L ← UrL, GOTO[PTail3], c2;
FaultDidNN: [] ← uPCValid, NZeroBr, GOTO[Fc], c2, at[rhT.FaultNN,10,NQRets];
{*****************************************************************************
Check For Timeouts
Entry:
Exit:
Register Usage
*****************************************************************************}
{db *****
CheckTime:
TT ← LShift1 uTickCount, XDisp, c2;
uTickCount ← TT, BRANCH[Scan, NoScan, 0B], c3;
NoScan: T ← UrPC, c1;
GOTO[PTail0], c2;
***** db}
Scan:
Noop, c2;
Noop, c3;
T ← uPTC, L0←L0.SrcNull, c1;
T ← T + 1, c2;
uPTC ← T, L2←L2.TS0{PRRets}, c3;
Map ← Q ← [rhPC, PDA.count], c1;
G ← PDAHi, CALL[PRead3], c2;
TT ← T, c1, at[L2.TS0,10,PRRets];
PC ← StartPsbM4, c2;
UvQ2Hi ← G, c3;
UvQ2 ← PDA.ready, GOTO[CheckEndx], c1;
TSLoop: Map ← Q ← [rhPC, PC or PSB.timeout], L2←L2.TS2{PRRets}, c1;
uBLTTemp ← TT, CALL[PRead3], c2;
Q ← T, ZeroBr, c1, at[L2.TS2,10,PRRets];
[] ← Q xor uPTC, NZeroBr, BRANCH[CanTimeout, NoTimeout], c2;
CanTimeout: T ← uBLTTemp, BRANCH[Timeout, CheckEnd], c3;
Timeout: MAR ← [rhTT, PC+PSB.flags], c1;
CANCELBR[$,0], c2;
T ← MD, c3;
MAR ← [rhTT, PC+PSB.flags], c1;
MDR ← T and ~PSB.flags.waiting, CANCELBR[$,0], c2;
rhT ← rhT.TS{NQRets}, c3;
MAR ← [rhTT, PC+PSB.timeout], c1;
MDR ← 0, CANCELBR[$,0], c2;
GOTO[Dequeue], c3;
GOTO[NoTimeout], c2, at[rhT.TS,10,NQRets];
NoTimeout: T ← uBLTTemp, CANCELBR[CheckEnd,1], c3;
CheckEnd: TT ← T - 1, ZeroBr, c1;
CheckEndx: T ← UrPC, BRANCH[TSNotDone, TSDone], c2;
TSNotDone: PC ← PC + PSBSize, GOTO[TSLoop], c3;
TSDone:
{db
uTickCount ← L xor ~L, GOTO[PTail1], c3;
db}
{db} GOTO[PTail1], c3;
{*****************************************************************************
CleanupQueue subroutine
Entry:
calling instruction in c1
L2 holds L2.CQ0{QRRets}
returns to uPFlags
Exit:
T holds condition queue
G holds UvQ1
Register Usage
PC = process
T = cleanup or link
Rx = head of queue
UreturnPC = condition and ~uPMask
MUST NOT USE L or TOS
*****************************************************************************}
CleanFirst: G ← UvQ1, CALL[Q1Read3], c2;
CleanSecond:
G ← UvQ2, CALL[Q2Read3], c2;
PC ← T and Q, ZeroBr, CANCELBR[$,3], c1, at[L2.CQ0,10,QRRets];
T ← T and ~uPMask, BRANCH[CQNE, CQEmpty], c2;
CQNE: T ← T and ~Condition.wakeup, L2←L2.CQ1{PMRRets}, c3;
Map ← Q ← [rhPC, PC or PSB.flags], c1;
UreturnPC ← T, CALL[PMRead3], c2;
[] ← T, ZeroBr, L2←L2.CQ4{QWRets}, c1, at[L2.CQ1,10,PMRRets];
CQLoop1: [] ← T - PC, ZeroBr, BRANCH[$, CQClean], c2;
[] ← T, ZeroBr, BRANCH[$, CQMakeEmpty], c3;
Map ← Q ← [rhPC, T or PSB.flags], L2←L2.CQ2{PMRRets}, BRANCH[$, CQHead], c1;
PC ← T, CALL[PMRead3], c2;
L2←L2.CQ4{QWRets}, GOTO[CQLoop1], c1, at[L2.CQ2,10,PMRRets];
CQHead: Rx ← PC, c2;
L2←L2.CQ3{PMRRets}, c3;
CQLoop2: Map ← Q ← [rhPC, PC], CALL[PMRead2], c1;
[] ← T - Rx, ZeroBr, L2←L2.CQ4{QWRets}, c1, at[L2.CQ3,10,PMRRets];
BRANCH[$, CQFoundTail], c2;
PC ← T, L2←L2.CQ3{PMRRets}, GOTO[CQLoop2], c3;
CQFoundTail:
T ← UreturnPC or PC, CALL[QWrite1], c3;
CQMakeEmpty:
Map ← [rhG, G], CANCELBR[$,1], c1;
T ← UreturnPC, CALL[QWrite3], c2;
CQClean: T ← PC, CANCELBR[CQRet,1], GOTO[CQRet], c3;
CQEmpty: T ← 0, GOTO[CQRet], c3;
GOTO[CQRet], c3, at[L2.CQ4,10,QWRets];
CQRet: Xbus ← uPFlags, XDisp, c1;
G ← UvQ1, DISP4[CQRets], c2;
{*****************************************************************************
PRead subroutine:
Entry:
c2, c3 => Q has virtual address; Map reference started by caller
L2 holds return
Exit State:
T has memory data
rhTT, TT, has real address
Q has uPMask
returnee executed on c1
*****************************************************************************}
PRead2: Noop, c2;
PRead3: TT ← rhTT ← MD, c3;
PR: MAR ← [rhTT, Q + 0], c1;
Q ← uPMask, L2Disp, c2;
T ← MD, DISP4[PRRets], c3;
{*****************************************************************************
PMRead subroutine:
Entry:
c2, c3 => Q has virtual address; Map reference started by caller
L2 holds return
Exit State:
T has memory data and uPMask
rhTT, TT, has real address
Q has uPMask
returnee executed on c1
*****************************************************************************}
PMRead2: Noop, c2;
PMRead3: TT ← rhTT ← MD, c3;
PMR: MAR ← [rhTT, Q + 0], c1;
Q ← uPMask, L2Disp, c2;
T ← MD and Q, DISP4[PMRRets], c3;
{*****************************************************************************
PFetch subroutine:
Entry:
c2, c3 => Q has virtual address; Map reference started by caller
L2 holds return
Exit State:
rhTT, TT, has real address
Q has uPMask
memory reference started
returnee executed on c3
*****************************************************************************}
PFetch2: Noop, c2;
PFetch3: TT ← rhTT ← MD, c3;
MAR ← [rhTT, Q + 0], L2Disp, c1;
Q ← uPMask, DISP4[PFRets], c2;
{*****************************************************************************
PWrite subroutine:
Entry:
Q has virtual address
calling instruction executed on c1 => Map reference started by caller
T has data to be written
L2 holds return
Exit State:
T has memory data
rhTT, TT, Q has real address
returnee executed on c3
*****************************************************************************}
PWrite2: Noop, c2;
PWrite3: TT ← rhTT ← MD, c3;
PW: MAR ← [rhTT, Q + 0], L2Disp, c1;
MDR ← T, RET[PWRets], c2;
{*****************************************************************************
QRead subroutine:
Entry:
rhG,,G has virtual address
L2 holds return
Exit State:
rhG, G has virtual address
T has MD and 77777B
XLDisp pending on MD to save ww bit
returnee executed on c1
*****************************************************************************}
QRead1: Map ← [rhG, G], c1;
QRead2: Noop, c2;
QRead3:
{db} TT ← rhTT ← MD, XdwDisp, c3;
QR: MAR ← [rhTT, G + 0], DISP2[QRMpFx], c1;
QRMpFx: GOTO[QRMpFx1], c2, at[0, 4, QRMpFx];
{db} GOTO[QRMpFx1], c2, at[1, 4, QRMpFx];
GOTO[QRMpFx1], c2, at[3, 4, QRMpFx];
{db} Q ← uPMask, L2Disp, c2, at[2, 4, QRMpFx];
T ← MD, XLDisp, RET[QRRets], c3;
QRMpFx1:
{db} Xbus ← TT LRot0, XdwDisp, c3;
Map ← [rhG, G], DISP2[QRFlgFx], c1;
QRFlgFx:
{db} MDR ← TT or map.referenced, GOTO[QRx], c2, at[0,4,QRFlgFx];
{db} MDR ← TT or map.referenced, GOTO[QRx], c2, at[2,4,QRFlgFx];
{db} T ← qWriteProtect, L2←L2.Fault, GOTO[PRestart], c2, at[1,4,QRFlgFx];
T ← qPageFault, L2←L2.Fault, GOTO[PRestart], c2, at[3,4,QRFlgFx];
QRx: Xbus ← rdw.x10, XDisp, GOTO[QR], c3;
{*****************************************************************************
Q1Read subroutine:
Entry:
UvQ1Hi, UvQ1 has virtual address
calling instruction executed on c1
L2 holds return
Exit State:
same as QRead
*****************************************************************************}
Q1Read3: rhG ← UvQ1Hi, GOTO[QRead1], c3;
{*****************************************************************************
Q2Read subroutine:
Entry:
UvQ2Hi, UvQ2 has virtual address
calling instruction executed on c1
L2 holds return
Exit State:
same as QRead
*****************************************************************************}
Q2Read2: G ← UvQ2, c2;
Q2Read3: rhG ← UvQ2Hi, GOTO[QRead1], c3;
{*****************************************************************************
QWrite subroutine:
Entry:
rhG, G has virtual address
L2 holds return
T has data to be written
Exit State:
rhG, G has virtual address
returnee executed on c3
*****************************************************************************}
QWrite1:
Map ← [rhG, G], c1;
QWrite2:
Noop, c2;
QWrite3:
{db} TT ← MD, rhTT ← MD, XdwDisp, c3;
QW: MAR ← [rhTT, G + 0], DISP2[QWMpFx], L2Disp, c1;
QWMpFx: CANCELBR[QWMpFx1,0F], c2, at[0, 4, QWMpFx];
{db} CANCELBR[QWMpFx1,0F], c2, at[1, 4, QWMpFx];
CANCELBR[QWMpFx1,0F], c2, at[3, 4, QWMpFx];
QWMpOk:
{db} MDR ← T, RET[QWRets], c2, at[2, 4, QWMpFx];
QWMpFx1:
{db} Xbus ← TT LRot0, XdwDisp, c3;
Map ← [rhG, G], DISP2[QWFlgFx], c1;
QWFlgFx:
{bj} MDR ← TT or map.rd, GOTO[QWx], c2, at[0,4,QWFlgFx];
{db} MDR ← TT or map.rd, GOTO[QWx], c2, at[2,4,QWFlgFx];
{db} T ← qWriteProtect, L2←L2.Fault, GOTO[PRestart], c2, at[1,4,QWFlgFx];
T ← qPageFault, L2←L2.Fault, GOTO[PRestart], c2, at[3,4,QWFlgFx];
QWx: Xbus ← rdw.x10, XDisp, GOTO[QW], c3;