*----------------------------------------------------------- Title[DMesaDefs.mc...July 30, 1985 5:20 PM...Rumph]; *----------------------------------------------------------- *----------------------------------------------------------- * Assembly options *----------------------------------------------------------- * Compile the Alto-Mesa emulator unless AltoMode is previously set to 0. IfDef[AltoMode, , Set[AltoMode, 177777]]; * nEntryPoints may be either 1 or 3 in Alto mode, only 1 in PrincOps mode. * Note that 1 entry point => restart opcode after page fault. IfDef[nEntryPoints, , Set[nEntryPoints, 1]]; *----------------------------------------------------------- * RM registers *----------------------------------------------------------- SetRMRegion[RMforIFU]; *RM region 0 :If[AltoMode]; ********** Alto version ********** RVN[DLink]; * Xfer destination link RVN[SLink]; * Xfer source link RVN[ATPReg]; * Allocation trap parameter RVN[OTPReg]; * Other trap parameter RVN[Sticky]; * Sticky flags for floating point :Else; ******** PrincOps version ******** Reserve[1]; * NWW: defined in RegisterDefs.mc RVN[CurrentPSB]; * PDA-relative address of current PSB RVN[CurrentTime]; * Current process time, in ~50 ms ticks RVN[XferFlags]; * Xfer control and state flags RVN[TrapParam]; * Xfer trap parameter :EndIf; ********************************** RVN[XTSReg]; * Xfer trap state RVN[WDC]; * Wakeup disable counter RVN[MDSHi]; * Main data space -- high part * Reserve[1]; * An unused register -- wow! RVN[RTempX]; * Not any more! RVN[RTemp0]; * General temporaries RVN[RTemp1]; RVN[RTemp2]; RVN[RTemp3]; RVN[RTemp4]; RVN[RTemp5]; * RVRel 16 and 17 must be temps, for RVN[RTemp6]; * multiply and divide subroutines :If[Not[AltoMode]]; ******** PrincOps version ******** SetRMRegion[AEmRegs]; * RM region 1 (overlay Alto Emulator registers) Reserve[1]; * Skip over R400 RVN[TickCount]; * Divides down 17-ms interrupts to make ticks RVN[Sticky]; * Sticky flags for floating point RVN[MaintPanel]; * Maintenance panel code * From CedarOps.mc -- do some consolidating of RM! * Set[!RCRegs, !AEmRegs]; SetRMRegion[RCRegs]; RVN[RCFlags]; * B0=1 => microcode enabled, 0 disabled * B15 = copy of zct.markingDecrements RVN[WPOffset]; * Write pointer, relative to WPBR RVN[RCWord]; * Contents of RC word at nhp↑ Reserve[1]; * ZCTBlockLink, defined in CedarOps.mc RVN[RCresidue]; RVN[RCResMask]; * holds the Residue Mask from zct % RME[RCTemp0, ETemp0]; RME[RCTemp1, ETemp1]; RME[RCTemp2, ETemp2]; RME[RCTemp3, ETemp3]; % RVN[RCTemp0]; RVN[RCTemp1]; RVN[RCTemp2]; RVN[RCTemp3]; SetRMRegion[AEmRegs]; :EndIf; ********************************** * RM registers (cont'd) * Special uses of the Mesa RM registers: :If[AltoMode]; ********** Alto version ********** RME[PDAHi, MDSHi]; * PDA is the same as MDS at present RME[XferFlags, RTemp6]; * Xfer control flags * XferFlags bit assignments. MC[xf.trap, 0]; * Not used in Alto mode MC[xf.push, 0]; * Not used in Alto mode MC[xf.free, 1]; * Free local frame :Else; ******** PrincOps version ******** * Temporary register assignments, used only inside Xfer and when calling Xfer. RME[DLink, RTemp5]; * Xfer destination link RME[SLink, RTemp6]; * Xfer source link * Temporary register assignments, for passing parameters thru fault handler. * FaultParam0 and FaultParam1 must be preserved across Requeue. RME[FaultParam0, TrapParam]; * First or low-order fault parameter RME[FaultParam1, RTemp3]; * Second or high-order fault parameter * XferFlags bit assignments. * Note: xf.invalidContext must have the appropriate value (usually 0) any * time a fault or trap may occur. Other flags are relevant only inside Xfer. * The low 3 bits are dispatched upon inside Xfer, so be circumspect about * reassigning them. MC[xf.invalidContext, 100000]; * Current context is invalid MC[xf.trap, 4]; * Store TrapParam in destination frame MC[xf.push, 2]; * Push SLink and DLink onto stack MC[xf.free, 1]; * Free local frame :EndIf; ********************************** * 16 and 17 must be temporaries (for MulSub and DivSub) RM[MulDivArg, Or[!RMforIFU, 17]]; *----------------------------------------------------------- * Base registers *----------------------------------------------------------- * BRs 0-3 are addressable as MemBX-relative registers. * Of these, 0 and 1 are used as MemBX-relative registers by Mesa, * thereby consuming registers 4, 5, 10, 11, 14, 15 also. * The other registers in [0..17] are used as ordinary registers * and should not be addressed from the IFU. BRX[L, 0]; * {mds, L} Local frame, = G xor 1 BRX[G, 1]; * {mds, G} Global frame, = L xor 1 BR[SD, 2]; * {mds, SD} System Dispatch table * BRs 34-37 are addressable from the IFU. * 36 and 37 are permanently assigned in ADefs.mc (MDS and CODE). * 34 and 35 are also used in DMesaProcess.mc. BR[LPtr, 34]; * Temp for Long Pointer instructions :If[AltoMode]; ********** Alto version ********** BR[PDA, IP[MDS]]; * PDA is the same as MDS at present :Else; ******** PrincOps version ******** BR[PDA, 3]; * Base of Process Data Area :EndIf; ********************************** BR[MapBitsBR, 25]; *----------------------------------------------------------- * System constants *----------------------------------------------------------- MC[AV, 1000]; Set[SDLoc, 1100]; MC[GFT, 1400]; * SD indices for Mesa traps MC[sBreak, 0]; MC[sStackError, 2]; MC[sWakeupError, 3]; MC[sXferTrap, 4]; MC[sUnimplemented, 5]; MC[sControlFault, 7]; MC[sSwapTrap, 10]; MC[sUnbound, 13]; MC[sZeroDivisor, 14]; MC[sDivideCheck, 15]; MC[sHardwareError, 16]; MC[sProcessTrap, 17]; MC[sBoundsFault, 20]; MC[sPointerFault, 21]; MC[sBoot, 276]; :If[AltoMode]; ********** Alto version ********** * SD indices for Mesa faults (handled as traps in Alto Mesa and * pre-Rubicon Pilot Mesa. MC[sAllocListEmpty, 6]; MC[sPageFault, 11]; MC[sWriteProtect, 12]; :Else; ******** PrincOps version ******** * Fault queue offsets (2 * corresponding PrincOps FaultQueueIndex) MC[qFrameFault, 0]; MC[qPageFault, 2]; MC[qWriteProtectFault, 4]; :EndIf; ********************************** * Miscellaneous definitions :If[AltoMode]; ********** Alto version ********** MC[MesaStopLoc, 26]; * Communication with the Nova World MC[CurrentState, 23]; :Else; ******** PrincOps version ******** MC[PDAhi, 1]; * PDA starts at VM 200000 MC[pilotMDSHi, 76]; * MDS initially starts at VM 17400000 :EndIf; ********************************** * Mesa instruction set number Set[MesaInsSet, IfE[AltoMode, 0, 1, 2]]; InsSet[MesaInsSet, nEntryPoints]; * Stack size (size of StateVector.stk array) Set[sizeStack, IfE[AltoMode, 0, 16, 10]]; *----------------------------------------------------------- * Mesa opcode definitions *----------------------------------------------------------- Set[MOpcodeVal, 0]; M[MOp, (IfSE[#1, , UndefOp[MOpcodeVal], Set[MOp#1, MOpcodeVal]] Set[MOpcodeVal, Add[MOpcodeVal, 1]])]; M[MesaOps, (MOp[#1] MOp[#2] MOp[#3] MOp[#4] MOp[#5] MOp[#6] MOp[#7] MOp[#8])]; M[UndefOp, IFUPause[#1, 1, L, 0, UnimplOpcodeTrap, 17, 0, 0]]; * xx0 xx1 xx2 xx3 xx4 xx5 xx6 xx7 MesaOps[NOOP, ME, MRE, MXW, MXD, NOTIFY,BCAST, REQUEUE]; * 00x MesaOps[LL0, LL1, LL2, LL3, LL4, LL5, LL6, LL7]; * 01x MesaOps[LLB, LLDB, SL0, SL1, SL2, SL3, SL4, SL5]; * 02x MesaOps[SL6, SL7, SLB, PL0, PL1, PL2, PL3, LG0]; * 03x MesaOps[LG1, LG2, LG3, LG4, LG5, LG6, LG7, LGB]; * 04x MesaOps[LGDB, SG0, SG1, SG2, SG3, SGB, LI0, LI1]; * 05x MesaOps[LI2, LI3, LI4, LI5, LI6, LIN1, LINI, LIB]; * 06x MesaOps[LIW, LINB, LADRB, GADRB, LCO, , ASSIGNREF,ASSIGNREFNEW]; * 07x MesaOps[R0, R1, R2, R3, R4, RB, W0, W1]; * 10x MesaOps[W2, WB, RF, WF, RDB, RD0, WDB, WD0]; * 11x MesaOps[RSTR, WSTR, RXLP, WXLP, RILP, RIGP, WILP, RIL0]; * 12x MesaOps[WS0, WSB, WSF, WSDB, RFC, RFS, WFS, RBL]; * 13x MesaOps[WBL, RDBL, WDBL, RXLPL, WXLPL, RXGPL, WXGPL, RILPL]; * 14x MesaOps[WILPL, RIGPL, WIGPL, RSTRL, WSTRL, RFL, WFL, RFSL]; * 15x MesaOps[WFSL, LP, SLDB, SGDB, PUSH, POP, EXCH, LINKB]; * 16x MesaOps[DUP, NILCK, NILCKL,BNDCK, , , , ]; * 17x MesaOps[J2, J3, J4, J5, J6, J7, J8, J9]; * 20x MesaOps[JB, JW, JEQ2, JEQ3, JEQ4, JEQ5, JEQ6, JEQ7]; * 21x MesaOps[JEQ8, JEQ9, JEQB, JNE2, JNE3, JNE4, JNE5, JNE6]; * 22x MesaOps[JNE7, JNE8, JNE9, JNEB, JLB, JGEB, JGB, JLEB]; * 23x MesaOps[JULB, JUGEB, JUGB, JULEB, JZEQB, JZNEB, JIB, JIW]; * 24x MesaOps[ADD, SUB, MUL, DBL, DIV, LDIV, NEG, INC]; * 25x MesaOps[AND, OR, XOR, SHIFT, DADD, DSUB, DCOMP, DUCOMP]; * 26x MesaOps[ADD01, , , , , , , ]; * 27x MesaOps[EFC0, EFC1, EFC2, EFC3, EFC4, EFC5, EFC6, EFC7]; * 30x MesaOps[EFC8, EFC9, EFC10, EFC11, EFC12, EFC13, EFC14, EFC15]; * 31x MesaOps[EFCB, LFC1, LFC2, LFC3, LFC4, LFC5, LFC6, LFC7]; * 32x MesaOps[LFC8, LFC9, LFC10, LFC11, LFC12, LFC13, LFC14, LFC15]; * 33x MesaOps[LFC16, LFCB, SFC, RET, LLKB, PORTO, PORTI, KFCB]; * 34x MesaOps[DESCB, DESCBS,BLT, BLTL, BLTC, , ALLOC, FREE]; * 35x MesaOps[IWDC, DWDC, STOP, CATCH, MISC, BITBLT,STARTIO, ]; * 36x MesaOps[DST, LST, LSTF, , WR, RR, BRK, ]; * 37x *----------------------------------------------------------- * IFU entry point macros *----------------------------------------------------------- * Call(s) should appear IMMEDIATELY BEFORE the entry vector. * Each call defines a label "@opcode" at the current IM location, and * forces the next nEntryPoints instructions to comprise an IFU entry vector. * Note: due to MicroD symbol table overflow problems, a label "@opcode" * is defined only for the first opcode in a group of consecutive declarations. * IFUR[opcode, length, memBase, options]; Regular opcode * IFUP[opcode, length, memBase, options]; Pause opcode * IFUJ[opcode, length, memBase, options]; Jump opcode * opcode = one of the opcode names defined above * length = 1, 2, or 3 * memBase = defined MemBase name * Options (zero or more, in any order): * N[n] n = 0 to 16 or "noN" * SignExtend * PackedAlpha * RBase[rBase] rBase = MesaRegs or AEmRegs (Alto mode only -- not PrincOps) * Disp[n] n = -40 to 37 (displacement for 1-byte jumps) M[IFUR, IOpLabel[#1] (IOp#4, IOp#5, IOp#6, IOp#7, IFUDeflt) IFUReg[MOp#1, #2, #3, RShift[CRB@, 4], ILC, IValN, IValSign, IValPA]]; M[IFUP, IOpLabel[#1] (IOp#4, IOp#5, IOp#6, IOp#7, IFUDeflt) IFUPause[MOp#1, #2, #3, RShift[CRB@, 4], ILC, IValN, IValSign, IValPA]]; M[IFUJ, IOpLabel[#1] (IOp#4, IOp#5, IOp#6, IOp#7, IFUDeflt) IFUJmp[MOp#1, #2, #3, RShift[CRB@, 4], ILC, IValSign]]; M[IOp, ]; M[IOpN, Set[IValN, #1]]; M[IOpSignExtend, Set[IValSign, 1]]; M[IOpPackedAlpha, Set[IValPA, 1]]; M[IOpDisp, Set[IValSign, #1]]; :If[AltoMode]; ********** Alto version ********** Equate[IOpRBase, KnowRBase]; :EndIf; ********************************** M[IFUDeflt, TopLevel[] KnowRBase[RMforIFU] Set[IValN, 17] Set[IValSign, 0] Set[IValPA, 0]]; Set[LastILC, 177777]; M[IOpLabel, IfE[IP[ILC], LastILC, , @#1: Set[LastILC, IP[ILC]]]]; *----------------------------------------------------------- * IFU exit macros and top-of-stack conventions *----------------------------------------------------------- * [0]: TOS = Stack[StkP], TOS-1 = Stack[StkP-1] * [1]: TOS = MD, TOS-1 = T & Stack[StkP-1] * [2]: TOS = T & Stack[StkP+1], TOS-1 = Stack[StkP] * State 0 is the canonical form, with all values stored in the hardware stack. * It is the only form if a single entry point is being used. * IFUNextN = IFUJump[N] if 3 entry points are being used. * If only 1 entry point, IFUNext0 or IFUNext0CF is the normal exit (IFUNext0CF * checks for faults first). IFUNext1 branches to code that puts MD on the * stack and exits. IFUNext2 is ordinarily illegal, but can be used with the * special "StackT" source and destination: * StackT← (StackT)+T, IFUNext2; * is equivalent to: * Stack&-1← T← (Stack&-1)+T, IFUJump[2]; * if 3 entry points, but * Stack← (Stack)+T, IFUJump[0]; * if 1 entry point. Set[multEntryPoints, IfE[nEntryPoints, 1, 0, 177777]]; M[IfMEP, If[multEntryPoints]]; M[If1EP, If[Not[multEntryPoints]]]; :IfMEP; M[IFUNext0, IFUJump[0]]; M[IFUNext1, IFUJump[1]]; M[IFUNext2, IFUJump[2]]; M[IFUNext0CF, IFUJump[0]]; M[StackT←, Stack&-1←T←]; Equate[StackT, Stack&-1]; :Else; M[IFUNext0, IFUJump[0]]; M[IFUNext1, Branch[PushMD]]; M[IFUNext2, IFUJump[0]]; M[IFUNext0CF, Branch[CheckFault0]]; Equate[StackT←, Stack←]; Equate[StackT, Stack]; :EndIf; *----------------------------------------------------------- * MISC opcode entry definition *----------------------------------------------------------- * MiscTable[n] places the current instruction at the entry point for * MISC operation n. Set[MT0Loc, 4000]; * Base for 000-077 dispatch Set[MT1Loc, 4400]; * Base for 100-177 dispatch Set[MT2Loc, 6000]; * Base for 200-277 dispatch Set[MT3Loc, 6400]; * Base for 300-377 dispatch M[MiscTable, At[Select[RShift[#1, 6], MT0Loc, MT1Loc, MT2Loc, MT3Loc], Add[LShift[And[#1, 77], 2], 1]]]; :If[AltoMode]; ********** Alto version ********** M[MiscOpcodeUnimplemented, (MiscTable[#1], T← sUnimplemented, StkP-1, Branch[SavePCAndTrap])]; :Else; ******** PrincOps version ******** M[MiscOpcodeUnimplemented, (MiscTable[#1], Branch[UnimplOpcodeTrap])]; :EndIf; ********************************** % Dorado Alto-Mesa microcode Note: most of these conventions currently apply to Pilot Mesa as well, though the PrincOps says otherwise! This code is strictly for Mesa 6 AV=1000C SD=1100C GFT=1400C State Block Configuration (Alto Mesa) alpha Stack[1] ... alpha+7 Stack[8] alpha+10 brkbyte,,skp alpha+11 mx (dest) alpha+12 my (source) Local frame L -> global frame L+1 PC (Alto: if PC odd then -(PC/2) else PC/2) L+2 -> caller's frame L+3 Alto: unused; PrincOps: place to store trap parameter L+4 local 0 L+5 ... Global frame configuration G-3 ... G-2 external link 1 G-1 external link 0 G GFI,,code links(1 bit) [code links=0 => links are in the frame, =1 => in the code segment] G+1 code base low G+2 code base high (Alto XMesa or PrincOps) G+3 global 0 G+4 ... Code segment C-3 ... C-2 external link 1 C-1 external link 0 CP reserved CP+1 reserved ... CP+2n+2 word pc offset (n = procedure number, as in LFCn) CP+2n+3 frame size index (in lower byte) ... % % *----------------------------------------------------------- Alto-Mesa notes: *----------------------------------------------------------- Alto-Mesa differs from PrincOps in the following systematic ways: 1. The byte order of code is backwards (right=even, left=odd). This is true ONLY of code, not of arbitrary byte arrays (e.g., jump tables). 2. All 3-byte instructions are aligned. This means that if the opcode is an even byte, the odd byte is ignored and alpha,,beta is taken from the next full word. (Note the byte order of alpha,,beta!) A few 1-byte instructions are also aligned. This means that if the opcode is an even byte, the odd byte is ignored and the next opcode is taken from the even byte of the next word. 3. Jump displacements are measured from the last byte of the instruction, not from the opcode. 4. Frame displacements are handled inconsistently. Local 0 is word 4 of the local frame. In "Byte" instructions (e.g., LLB), the operand byte already includes the offset, but in "N" and "Pair" instructions it does not. That is, the following instructions refer to the same item (local 0): LL0 LLB 4 RILP <0,,x> (n.b. at present, Pilot follows this convention also.) 5. Alto-Mesa imposes "minimal stack" requirements in certain instructions. However, this Mesa implementation requires minimal stack only in the process/monitor opcodes. %