Dandelion Microcode Reference

14 Feb 80
R. Garner

This document describes Rev. D of the CP.
(previous revisions: 31 Oct 79, 8 Jan 80, 22 Jan 80)

XEROX

OPD SYSTEMS DEVELOPMENT
3408 Hillview Ave. / Palo Alto / California 94304
Contents

A. Branches and Dispatches 6
B. Shifting 9
C. Link Registers & Subroutines 11
D. SU Registers 14
E. Mesa Stack 15
F. Mesa Instruction Buffer 18
G. Memory 21
H. The Map 23
I. Bus Sinks 27
J. Bus Sources 27
K. Miscellaneous Functions 28
L. Microcode Conventions 29
M. MASS 31
N. Burdock-Kernel 36
O. Timing Constraints 39

Appendix: Antithetical List of Microinstructions

Where the files are saved:

[Iris]<Workstation>Mass>RevC>Mass.image "bcd" -- or
[Iris]<Workstation>Jarvis>Burdock.image
[Iris]<Workstation>mc>KernelDlion.fb

[Iris]<Workstation>LH>DMR.press
[Iris]<Workstation>LH>dmr1.bravo ,dmr2.bravo
[Iris]<Workstation>LH>DMRSil.dm
### Field Description

<table>
<thead>
<tr>
<th>Field</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>rA</td>
<td>2001 A reg addr, U addr [0-3]</td>
</tr>
<tr>
<td>rB</td>
<td>2001 B reg addr, RH reg addr</td>
</tr>
<tr>
<td>aS</td>
<td>2001 alu Source operand pair</td>
</tr>
<tr>
<td>aF</td>
<td>2001 alu Function</td>
</tr>
<tr>
<td>aD</td>
<td>2001 alu Destination/shift control</td>
</tr>
<tr>
<td>ep</td>
<td>Even Parity</td>
</tr>
<tr>
<td>Cn</td>
<td>2001 Carry In, Shift Ends, writeSU if enSU = 1</td>
</tr>
<tr>
<td>enSU</td>
<td>enable SU reg file</td>
</tr>
<tr>
<td>mem</td>
<td>MAR (if c1), MDR (if c2), +MD (if c3)</td>
</tr>
<tr>
<td>fs</td>
<td>Function field Selector</td>
</tr>
<tr>
<td>fX</td>
<td>X Function</td>
</tr>
<tr>
<td>fY</td>
<td>Y Function</td>
</tr>
<tr>
<td>fZ</td>
<td>Z Function</td>
</tr>
<tr>
<td>INIA</td>
<td>Next Instruction Address</td>
</tr>
</tbody>
</table>

### Instruction Format

<table>
<thead>
<tr>
<th>rA</th>
<th>rB</th>
<th>aS</th>
<th>aF</th>
<th>aD</th>
<th>fX</th>
<th>fY</th>
<th>fZ</th>
<th>INIA</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Field</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>aS</td>
<td>7</td>
</tr>
<tr>
<td>aF</td>
<td>5</td>
</tr>
<tr>
<td>aD</td>
<td>3</td>
</tr>
<tr>
<td>fX</td>
<td>2</td>
</tr>
<tr>
<td>fY</td>
<td>1</td>
</tr>
<tr>
<td>fZ</td>
<td>0</td>
</tr>
</tbody>
</table>

### Notes

2. When writing SU (Cin=1, enSU=1) and doing Cin=pc16, iX Cin=pc16 must be used.
3. MAR+ causes $a=0$ & al=“R0” in biss=1 of ALU; tests for PageCrossBr; cancels MDR+ or IBDDisp if PageCross true.
4. XBus[0-7] is zeroed for $fZ=[7..1]$ and $fS$ = Nibble/Byte.

---

**Figure 1. Dandelion Microinstruction Format**

7 Feb 80
Filed: WorkstationXLHDMRFormat-D.sil
WorkstationXLHDMRpress
Figure 2. Dandelion Processor Data Paths
<table>
<thead>
<tr>
<th>Mem</th>
<th>Size</th>
<th>Address</th>
<th>Read</th>
<th>Written</th>
</tr>
</thead>
<tbody>
<tr>
<td>Link</td>
<td>8x4</td>
<td>fX</td>
<td>any c</td>
<td>any c</td>
</tr>
<tr>
<td>CS</td>
<td>4Kx48</td>
<td>NIA</td>
<td>all c</td>
<td>all c</td>
</tr>
<tr>
<td>TPC</td>
<td>8x12</td>
<td>Nt</td>
<td>c2</td>
<td>c3</td>
</tr>
<tr>
<td>TC</td>
<td>8x4</td>
<td>Nt</td>
<td>c2</td>
<td>c1</td>
</tr>
</tbody>
</table>

> denotes a register which is loaded at end of each cycle
+ denotes logical OR
(IB[4-7] is ored with INIA[8-11] and IB[0-3] replaces INIA[4-7])

Figure 3. Dandelion Processor Control Paths

24 Oct 79
<Workstation>LHHCPControlPaths.sil
<Workstation>LHHDMR.press
### A. Branches and Dispatches

<table>
<thead>
<tr>
<th>NegBr</th>
<th>source</th>
<th>INIA dest</th>
<th>( \text{sign of alu result (Not Y.0)} )</th>
</tr>
</thead>
<tbody>
<tr>
<td>ZeroBr</td>
<td>F.0</td>
<td>11</td>
<td>alu output equal to zero</td>
</tr>
<tr>
<td>YOddBr</td>
<td>F = 0</td>
<td>11</td>
<td>least significant Y bus bit</td>
</tr>
<tr>
<td>MesalntBr</td>
<td>Y.15</td>
<td>11</td>
<td>Mesa interrupt bit</td>
</tr>
<tr>
<td>NibCarryBr</td>
<td>Cout.12</td>
<td>11</td>
<td>alu carry out of low nibble</td>
</tr>
<tr>
<td>PgCarryBr</td>
<td>Cout.8</td>
<td>11</td>
<td>alu carry out of low Byte</td>
</tr>
<tr>
<td>CarryBr</td>
<td>Cout.0</td>
<td>11</td>
<td>alu carry out</td>
</tr>
<tr>
<td>XRefBr</td>
<td>X.11</td>
<td>11</td>
<td>present &amp; referenced map bit</td>
</tr>
<tr>
<td>IODisp</td>
<td>bp.39.,bp.139,</td>
<td>10,11</td>
<td>IO branches (bp = Backplane pin)</td>
</tr>
<tr>
<td>XHDisp</td>
<td>X.4.,X.0</td>
<td>10,11</td>
<td>X High bus</td>
</tr>
<tr>
<td>XLDisp</td>
<td>X.8.,X.15</td>
<td>10,11</td>
<td>X Low bus</td>
</tr>
<tr>
<td>PgCrOvDisp</td>
<td>pgCross.,OVR</td>
<td>10,11</td>
<td>Page Cross &amp; Overflow bits</td>
</tr>
<tr>
<td>XDisp</td>
<td>X[12-15]</td>
<td>8,9,10,11</td>
<td>X bus</td>
</tr>
<tr>
<td>YDisp</td>
<td>Y[12-15]</td>
<td>8,9,10,11</td>
<td>Y bus</td>
</tr>
<tr>
<td>Xc2ncDisp</td>
<td>Y[12-13],.,pc16</td>
<td>8,9,10,11</td>
<td>X bus, cycle2.,-,pc16</td>
</tr>
<tr>
<td>YIDisp</td>
<td>Y[12-13],.,bp.39.,bp.139</td>
<td>8,9,10,11</td>
<td>IO branches (bp = Backplane pin)</td>
</tr>
<tr>
<td>IBDisp</td>
<td>ib</td>
<td>[4-11]</td>
<td>Instruction Buffer</td>
</tr>
<tr>
<td>L0Disp, pRet0</td>
<td>Link0</td>
<td>8,9,10,11</td>
<td>Link0 dispatch (INIA.7 = 1)</td>
</tr>
<tr>
<td>L7Disp, pRet7</td>
<td>Link7</td>
<td>8,9,10,11</td>
<td>Link7 dispatch (INIA.7 = 1)</td>
</tr>
</tbody>
</table>

1. PageCross branch (called PgCrossBr) is defined to be "PageCarry xor af.2." This has the effect of toggling PageCarry when doing subtraction (\( af = S - R \)). PageCross equals PageCarry when doing addition (\( af = R + S \)). Thus, assuming one uses positive displacements, such as \( R + 1 \) or \( R - 1 \), PgCrossBr will consistently indicate when a page boundary has been crossed.

The \( af = S - R \) form of subtraction, unlike \( af = S + R \), does not cause PageCarry to be toggled on subtraction (since \( af.2 = 0 \)). However, the \( af = S - R \) form covers most of the common subtraction cases: \( B - L, A - L, B - A, A - constant, \) and \( Q - constant \). It does not include \( D - L \). In \( A + B \), if either \( A \) or \( B \) is negative, PgCross branch will always be true. Moral: Always add or subtract positive displacements and PgCross branch will be true to you.

2. Even in the absence of ALU arithmetic, the NibCarryBr, PageCarryBr, and PgCrOvDisp branches can produce non-zero results (i.e., branch). When \( af = \text{RandS} \), NibCarry and PageCarry are the logical inner product of \( R \) with \( S \). If \( af = \text{notRandS} \) and \( as = \text{"0.B"}, \text{"0.A"}, \text{or } \text{"0.Q"} \), then NibCarryBr tests for the low nibble equaling zero and PgCarryBr tests for the low byte equaling zero. If \( af = \text{"RandS"}, \text{NibCarry is the logical inner sum of R with S. If as = "0.B", "0.A","0.Q","0.D", then NibCarryBr tests for the low nibble equaling 0F. For the final coup de grace, if af = "RandS" and as = "0.Q", "0.A", or "0.B", then NibCarryBr is true if the low nibble is 8, 0C, 0E, or 0F.

### Notes:

1. Branches take two microinstructions to specify. In the first microinstruction the branch or dispatch condition (abbreviated DispBr) is declared by an FY. The second instruction should contain a "BRANCH([Label0, Label1])" phrase.

   ```
   Reg + Reg xor RegA, ZeroBr,
   BRANCH([NotZero, Zero]),
   ```

   **c1; c2;**

   ```
   NotZero:  Nop  {here if result nonzero},
   Zero:     Nop  {here if result zero},
   ```

   **c3; c3;**

2. The "at[x, y, Label]" macro is used to constrain the location of instructions. It tells MASS to place the instruction at a control store location which is "x MOD y" and in the same "MOD group" as the instruction labeled "Label." Thus, the above example could be rewritten as (The "at's are NOT required.)

   ```
   Reg + Reg xor RegA, ZeroBr,
   BRANCH([NotZero, Zero]),
   ```

   **c1; c2;**

   ```
   NotZero:  Nop,  {here if result nonzero},
   Zero:     Nop,  {here if result zero},
   ```

   **c3, at[0,2,Zero]; c3, at[1,2,NotZero];**
3. A dispatch specifies more than a single bit which is OR'd into INIA. Instead of a "BRANCH" macro, dispatches are specified by "DISP2[Label]", "DISP3[Label]" or "DISP4[Label]" (abbreviated DISPn), where n specifies the number of bits used for the dispatch. "at" clauses ARE required.

    [] = MD, XLDisp,
    DISP2(Table),
    c3;
    c1;

Table:
Noop, {here if MD.8,MD.15 = 0}  
Noop, {here if MD.8,MD.15 = 1}  
Noop, {here if MD.8,MD.15 = 2}  
Noop, {here if MD.8,MD.15 = 3}  

4. A two-way branch on a dispatch field is notationally accomplished by specifying a mask which has 1's in those bit positions of the dispatch which should be ignored in the branch. The mask should be the same width as the one implied by the dispatch. The mask is a third argument to the "BRANCH" macro. The only legitimate values for the third argument have exactly one zero in their binary (and a leading zero is used if needed); they are [1,2,3,5,6,7,0B,0D,0E]. "0F" is illegal since it has no zero in its binary. "at" clauses are NOT required.

    [] = Reg LRot8, XLDisp,
    BRANCH[NotSet, Set, 0B], {branch on bit 13 of X bus}
    c1;
    c2;

NotSet:
Noop, {here if bit 5 of Reg = 0}  
Noop, {here if bit 5 of Reg = 1}  

Set:
Noop, {here if bit 5 of Reg = 0}  
Noop, {here if bit 5 of Reg = 1}  

5. A dispatch on a sub-field of a dispatch is again specified with a mask which says which bits of the larger dispatch should be ignored. The mask is a second argument to the "DISPn" macro. "at" clauses are NOT required.

    [] = RHReg, XLDisp,
    DISP4(Table, 9),
    c3;
    c1;

Table:
Noop, {here if RHReg.13,RHReg.14 = 0}  
Noop, {here if RHReg.13,RHReg.14 = 1}  
Noop, {here if RHReg.13,RHReg.14 = 2}  
Noop, {here if RHReg.13,RHReg.14 = 3}  

6. The "CANCELBR" macro is used to cancel pending branch/dispatch conditions by forcing the argument address to have ones where pending bits would normally be OR'd in. CANCELBR may be necessary after a path of two instructions which specify branching or after a MAR+ (see Memory section). MASS will give a warning message when it thinks there should be a CANCELBR. (It uses the principle that all DispBr's or pRet's should be followed by either a BRANCH, DISPn, RET, or CANCELBR)

ZeroBr,
NegBr, BRANCH[NZ, Z],  

NZ:  
BRANCH[Pos, Neg],

Z:  
CANCELBR[Zero],

Zero:  
Noop, {placed "at[1,2]" by MASS}

7. Pending bits of a dispatch or "pRet" are canceled by a mask which says which bits should be ignored. The mask is the second argument to the CANCELBR macro. Thus mask = 0F causes the argument address to be placed "at[0F,10]."

ZeroBr,
pRet0, BRANCH[NZ, Z],  

NZ:  
RET[NZReturn],

Z:  
CANCELBR[NotYet, OF],

Zero:  
Noop, {placed "at[0F,10]" by MASS}
A general rule for the branch masks described above: The "mask" always indicates bits which should be ignored.

8. The "GOTOABS" macro sends control to an absolute control store location.

\[ \text{GOTOABS}[0], \quad e3; \]
B. Shifting

Single-bit shifts and rotates occur at the output of the ALU and the results can only go to an R register (or Q on double length shifts). Four-bit rotates occur between the Y bus (ALU F output or A bypass) and the X bus. If the result of the 4-bit rotate is destined for an R register, it must have been placed onto the Y bus via the A bypass (which implies that aD = 2). Single-bit shifting uses the fX field, single-bit rotating fX or fY, and 4-bit rotates fZ.

<table>
<thead>
<tr>
<th>Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>LShift1, RShift1</td>
<td>Left, Right Shift R by 1 (fX = shift)</td>
</tr>
<tr>
<td>LRot1, RRot1</td>
<td>Left, Right Rotate R by 1 (fX = cycle or fY = cycle)</td>
</tr>
<tr>
<td>DALShift1, DARShift1</td>
<td>Double Arithmetic Left, Right Shift R, Q by 1 (fX = shift)</td>
</tr>
<tr>
<td>DLShift1, DRShift1</td>
<td>Double Left, Right Shift R, Q by 1 (fX = cycle or fY = cycle)</td>
</tr>
</tbody>
</table>

**aD.0 = 0 implies right shift**

\[
\begin{array}{c|c|c|c|c|c}
\hline
aD & fX & fY & \text{Shift} & \text{Cycle} \\
\hline
1 & & & 1 & 0 \\
1 & & & cycle & shift \\
0 & & & 0 & 1 \\
0 & & & cycle & shift \\
\hline
\end{array}
\]

\[
\begin{array}{c|c|c|c|c|c}
\hline
\text{Cin} & \rightarrow & R & \rightarrow & \text{Cin} \\
\hline
\text{Cout} & \rightarrow & R & \rightarrow & Q & \rightarrow & \text{Cin} \\
\hline
\text{Cin} & \rightarrow & R & \rightarrow & Q & \rightarrow & \sim \text{Cin} \\
\hline
\end{array}
\]

**Notes:**

1. The notation "SE+0", "SE+1", or "SE+pc16" is used to specify the shift ends. "SE+" is equivalent to "Cin+".

\[
\begin{align*}
\text{Reg} + \text{LShift1 Reg, SE+1, \{puts 1 into Reg.15\}} & \quad \text{c1;} \\
\text{Reg} + \text{DALShift1 Reg, SE+1, \{puts 1 into Q.15\}} & \quad \text{c1;}
\end{align*}
\]

2. A "DARShift1" shifts Cout into the left side of the double length R, Q. (This is used in the multiply instruction.) The 2901 can cause carries on logical operations (believe it or not). Therefore, if you want to shift a 0 into the left side you must specify an arithmetic operation which produces no carry.

\[
\begin{eqnarray*}
\text{Reg} + \text{DARShift1 (Reg + 0)} & \quad \text{c1;}
\end{eqnarray*}
\]

3. The single bit shifting operations use Cin for the shift ends. Therefore, if SU is being read and shifted, or SU is being read and there is a shift operation in the ALU, the shift ends must be zero. Note that SU cannot be written simultaneously with any type of shifting operation (A bypass and shift are not a legal aD combination). If shifting is combined with arithmetic, the shift ends must be 0 unless a +1 operation is desired. Note that "Reg−Reg" implies a SE of 1.

\[
\begin{align*}
\text{Reg} + \text{RShift1 (RegA + Reg), SE+1,} & \quad \text{c1;} \\
\text{Reg} + \text{RShift1 (Reg + 1), SE+1,} & \quad \text{c1;} \\
\text{Reg} + \text{RShift1 (RegA − Reg − 1), SE+0,} & \quad \text{c1;} \\
\text{Reg} + \text{RShift1 Ureg, SE+0,} & \quad \text{c1;}
\end{align*}
\]

4. LRotn, when used in conjunction with A bypass, allows the ALU to be used for other purposes. For instance, an R register can be rotated and placed onto the Xbus (where it can be branched on or sent to RH or IOOut) while arithmetic is performed in the ALU. Note that the R register given by rB must always be written when A bypass is used.
1. Out + RegA LRot8, Reg + Reg + 1, c1;
   rhReg + Reg LRot12, Reg + ~Reg, XDisp, c1;
   STK + RegA, rhReg + RegA LRot0, Reg + Reg + 1, c1;

5. An arbitrary 16 bit rotate takes 3 cycles (plus 1 to specify it). This example uses 2 R registers and assumes the shift count is in RH.

[[=rhReg, XDisp,
  T = LRot1 R, DISP4[Rot],

Rot:]
   GOTO[Shift0],
   R = T, GOTO[Shift0],
   R = LRot1 T, GOTO[Shift0],
   R = RRot1 R, GOTO[Shift0],
   GOTO[Shift4],
   R = LRot1 R, GOTO[Shift4],
   R = LRot1 T, GOTO[Shift4],
   R = RRot1 R, GOTO[Shift4],
   GOTO[Shift8],
   R = LRot1 R, GOTO[Shift8],
   R = LRot1 T, GOTO[Shift8],
   R = RRot1 R, GOTO[Shift8],
   GOTO[Shift12],
   R = LRot1 R, GOTO[Shift12],
   R = LRot1 T, GOTO[Shift12],
   R = RRot1 R, GOTO[Shift12],
   Shift0:]
   GOTO[Done], c1;
   Shift4: R = R LRot4, GOTO[Done], c1;
   Shift8: R = R LRot8, GOTO[Done], c1;
   Shift12: R = R LRot12, GOTO[Done], c1;
C. Link Registers & Subroutines

Link registers, besides being used for subroutines, can be used to store 4-bits of state information which can be branched on later. Constants or branch condition bits can be stored in Link registers. Later, current branch conditions can be simultaneously OR’d with the saved state bits. "pRetn" acts like a dispatch/branch (DispBr) and "pCalln" is used to load a link register. When either a "pRetn" or "pCalln" is specified, the following instruction must be constrained in some way.

Address bits of the following instruction are indicated by "IA.n", where n varies from 0 to 11 ("IA" stands for "instruction address"). "NIA" ("next instruction address") is the 12-bit quantity which addresses the control store -- while instruction "n" is executing, "n+1" is being accessed from the control store. "INIA" refers to the contents of the 12-bit microinstruction field. In the CP, "inia" is OR’d with the currently specified dispatch/branch bits to form "NIA."

1. Link registers are loaded from the low 4 bits of NIA--the control store address which is currently being used to fetch the next microinstruction. Notationally, the instruction after an "L1+=" must be constrained such that its low 4 bits equal the constant to be loaded into the Link register. In addition, IA.7 of the next instruction must be 0 (MASS does this allocation). "Ln+=" is equivalent to "pCalln". The "at" is NOT required.

Set[FlagBB, 6];
L1 + FlagBB. {loads a 6 into Link1} c1;
Noop, c2, a[8,10];

2. If the microinstruction before the "pCalln" specifies a branch, dispatch or "pRetn", then the specified bits will be OR'd into the value stored into the link register. The "at" is NOT required.

[] + Reg xor RegA, NegBr,
pCall5, BRANCH[Pos, Neg], {bit 3 of Link6 + IF negative THEN 1 ELSE 0} c1;
Noop, c2, a[0,2];

3. "LnDisp" is used to dispatch on the value of link register "n" and is equivalent to "pRetn". Branch or dispatches can be simultaneously specified. The instruction after a "pRetn" must be constrained so that the "BRANCH", "DISPn" or "RET" has the desired affect. In addition, IA.7 of the next instruction must be 1 (MASS does this allocation).

The following example dispatches on "(0,0,.,Ureg.8,.,Ureg.15) OR Link3" and places the result in Link2:

[] + Ureg, L3Disp, XLDisp,
pCall2, DISP4[Table], c1;
Noop, c2;

4. Each subroutine has an associated table of 16 possible return locations. On exit, the subroutine uses a link register (specific to the subroutine) to dispatch into the return table. Thus a subroutine usually only has 16 possible return locations (usually implying 16 possible call locations). Each location of the return table also has ia.7 set to 1 since the table is preceded by a pRetn. Similarly, each location of the call table has IA.7 set to 0 since each "CALL." is preceded by a "pCalln". Thus, the "pCalln" can not immediately precede a return point since IA.7 can not be resolved. (See example below). It is possible to have a "CALL." on a return point if RH registers are used instead of Link registers for subroutines (See C.6).
5. Since condition bits can be simultaneously specified with a "pRet", there can be conditional return points. The same is true of "pCall" so conditional entry points are possible. If the least significant bit of the return address is not masked, conditional calls always imply conditioned returns (since the condition bits are saved in the Link register).

6. RH registers can also be used for subroutine calling. This format is easier to use since there are less address constraints: The call points don't need to be "at'd" and return addresses need not have \( IA.7 = 1 \).
7. By using RH registers more than 16 return points can be accommodated through multiple return-dispatch tables. Two tables imply 32 call/return locations.

Sub: Noop,  c3;

Table 1:
[] ← rhRet, XRefBr,
[] ← rhRet, XDisp, BRANCH[Table 1, Table 2],
RET[ReturnA],  c2;
RET[ReturnA],  c3;

Table 2:
RET[ReturnB],  c1, at[0,2,Table2];
RET[ReturnC],  c1, at[1,2,Table1];
D. SU Registers

The Stack-U registers (abbreviated "SU") are addressed implicitly by the fs field. The "Cin" field determines whether SU will be read or written. The SU addressing mode and Cin field only effect the SU registers if the "EnSU" (Enable SU) field is 1.

1. When writing SU, Cin must be 1. If SU is being written via the A bypass, ALU arithmetic must assume a Cin of 1. Similarly, if SU is being placed on the X bus only, Cin must be 0.

\[
\begin{align*}
R\text{reg} & = U\text{reg} + Reg + RegA, & c3; \\
U\text{reg} & = Reg + RegA + 1, & c1; \\
\end{align*}
\]

2. If fs2 is 0, the SU address comes from the stack pointer (stackP). The fZ field is free to be interpreted as either fZNorm or a Nibble.

\[
\begin{align*}
STK & = RegA, Reg = RegA + OFF + 1, & c3; \\
STK & = STK, rhReg = RegA L\text{Rot}0, X\text{Disp}, & c1; \\
\end{align*}
\]

3. If fs2 is 1, the SU address is rA,,fZ. Since the aS value which combines a U with an R register is "DA" and since rA is also used to specify the high four U address bits, a given R register can only be combined (in one statement) with U registers which are in the block of 16 given by the value of the R register. If A bypass is used in a statement which uses a U register, the same restriction is true. Statements which only read or write U registers are not affected.

\[
\begin{align*}
\text{Reg} & = U\text{reg}, & c3; \\
\text{Reg} & = U\text{reg xor RegA}, \quad \{\text{RegA} = U\text{reg[0-3]}\} & c1; \\
\text{Ureg} & = RegA, \text{Reg} + \text{MD}, \quad \{\text{RegA} = U\text{reg[0-3]}\} & c1; \\
\end{align*}
\]

4. If fs2 is 1 and fZ of the previous instruction was "AltUaddr", then the SU address is rA,,Y[12-15]. Here Y[12-15] is for the previous instruction (the same one which contained the AltUaddr). This U register indexing mode can be used to efficiently load a block of 16 U registers from memory (such as from an IOCB). The individual U registers can be used later, one at a time. The following example assumes the 16 words in memory are hex aligned (rAddr is 0 mod 16).

MASS expects a register of type UY, where the 4-bit register number references the block of 16. "AltUaddr" cannot occur in c3, and a UY register should not be used in c1—the addressing mode can't be used across clicks.

\[
\begin{align*}
\text{RegDef}[\text{Ublock, UY, O}E]; \\
\text{Cont:} \\
\text{MAR} & = [r\text{Addr, rAddr}, r\text{Addr} + r\text{Addr} + 1, & c1; \\
[1] & = \sim 0 \oplus r\text{Addr, AltUaddr, NibCarryBr} \text{ [tests for 0 nibble],} & c2; \\
\text{Ublock} & + r\text{data, rData} + \text{MD, BRANCH[Cont, Exit]} & c3; \\
\end{align*}
\]

5. The Alternate U addressing mode can be used with IOXIn (but not fZNorm or Nibble).

\[
\begin{align*}
[1] & = \langle \text{Address}, \text{AltUaddr}, & c2; \\
\text{Ublock} & + r\text{Data, [1] = RH, XDisp}, & c3; \\
\end{align*}
\]
E. Mesa Stack

1. For the PrincOps stack, the stackP equals the number of words on the stack. Thus, the stackP=0 for an empty stack; and stackP=8 for a full stack. Also in the PrincOps stack, the stackP points one above the top of stack, thus a PrincOps Pop must decrement the stackP & return the top of stack and a PrincOps push must write first, then increment the stackP.

In the Dandelion, the top of stack is kept in TOS and TopOfStack-1 is kept on the top of the stack in the U register file (STK). The stackP always points at TOS-1 in the STK. Thus, to pop STK one moves STK[stackP] to TOS and decrements the stackP, and to push one increments the stackP and then moves TOS to STK[stackP]. In order to keep the values of stackP identical for the two Stack representations, PrincOps stack locations 1-8 should be mapped into U locations 2-9. For example, if the PrincOps stack has one entry, then TOS is full, stackP=1, and (with a stack push) TOS could be saved in STK[2]. If the stack is empty, then TOS is empty, stackP=0, and (with a stack push) TOS could be saved in STK[1].

Figure 4 shows the stack from empty to overflowing.

U[9] is necessary if one assumes one can always save TOS into STK, i.e., if the stack is full (stackP=8) and we save TOS, the place it will go is U[9]. For example, if JEQu is executed on a full stack, then U[9] is necessary (This would not be stack overflow). If a Mesa Push is tried on a full stack, the write into U[9] would occur before it could be stopped (This would be a stack overflow).

Since stackP=8 & Push does not define stack overflow, we define overflow to be stackP=9 and overflow. This implies that whenever a true Mesa Stack Push is desired, the stackP must be incremented twice and decremented once. The idea is that stackP can always be incremented once (to save TOS into STK) without fear of overflow, but if we are truly putting one more word on the Stack, we must increment it once more.

Stack underflow occurs when stackP=0 and a Pop is attempted.

The maximum sized Mesa stack is 14 words (overflow at stackP=15 and Push, underflow at stackP=0 and Pop).

If a stack error occurs, one additional emulator click beyond the one which errored can execute before the emulator begins executing control store location 0 in cl.

2. To ameliorate checking for stack overflow or underflow, the pop function fields have been asymmetrically encoded. The following tables show the allocation of pops and pushes among the function fields and their effect on the stackP and the StackErrProm when multiple pops and pushes are specified in the same microinstruction.

<table>
<thead>
<tr>
<th>Function</th>
<th>stackP</th>
<th>Check For</th>
</tr>
</thead>
<tbody>
<tr>
<td>pop</td>
<td>-1</td>
<td>underflow</td>
</tr>
<tr>
<td>push</td>
<td>+1</td>
<td>overflow</td>
</tr>
<tr>
<td>ixpop, push</td>
<td>0</td>
<td>underflow (simulates a Pop)</td>
</tr>
<tr>
<td>push, izpop</td>
<td>0</td>
<td>overflow (simulates a Push)</td>
</tr>
<tr>
<td>ixpop, izpop</td>
<td>-1</td>
<td>underflow (simulates a Pop-Pop)</td>
</tr>
<tr>
<td>ixpop, izpop, push</td>
<td>0</td>
<td>underflow (simulates a Pop-Pop)</td>
</tr>
</tbody>
</table>
3. In general, the previously executed Mesa instruction may complete executing without saving TOS into STK. Therefore each Mesa instruction implementation must, if necessary, save TOS into STK (at STK[stackP+1]) before it modifies TOS. According to PrincOps, if TOS is an argument to the bytecode, TOS should be saved away (so it can be recovered by a Mesa PUSH) if either the Mesa bytecode does not change the contents of the Stack or does not change the value of the stack pointer. SLn and JEQn are two examples.

Note that as a part of normal Stack maintenance, TOS must be saved into STK if the Mesa opcode is merely pushing data onto the Stack.
Figure 4. Mesa Stack examples

8 Jan 79
MesaStack.sil
F. Mesa Instruction Buffer

The instruction buffer holds a maximum of 3 bytes—the minimum number necessary to complete a Mesa instruction. Whenever a Mesa opcode completes and there are not 3 bytes in the buffer, a microcode trap is caused which results in refilling of the buffer. The so-called “refill” microcode executes in one click if 2 more bytes are needed and in two clicks if 4 are needed. The Refill code also dispatches on the front byte of the buffer (so if 4 bytes were fetched, 3 are retained).

The instruction buffer has four possible states as given by the 2-bit "ibPtr" register:

<table>
<thead>
<tr>
<th>state</th>
<th>ibPtr</th>
</tr>
</thead>
<tbody>
<tr>
<td>3 bytes (full)</td>
<td>0</td>
</tr>
<tr>
<td>2 bytes</td>
<td>1</td>
</tr>
<tr>
<td>1 byte</td>
<td>3</td>
</tr>
<tr>
<td>0 bytes (empty)</td>
<td>2</td>
</tr>
</tbody>
</table>

Thus the ibPtr counts 0, 1, 3, 2, 2, ....

"+ib", "IBDisp", and "AlwaysIBDisp" cause the ibPtr to "increment" by 1. However, "+ibNA", "+ibHigh", and "+ibLow" do not change the ibPtr. "IB−" causes ibPtr to be set to 1 if it was 2 and otherwise sets it to 0. "IBPtr=n" sets it to 1 and "IBPtr+1" sets it to 3 (See Figure 5).

As shown below, the 3 bytes of the buffer are labeled IB[0], IB[1], and ibFront. IB[0], IB[1] is parallel-loaded by a word from the X-bus when fy = “IB+”. ibFront is loaded if fy = “IBDisp”, "AlwaysIBDisp", "+ib" or "IB−". "IB−" only loads ibFront if the old ibPtr=2. When it is loaded, its value comes from IB[0] if the old ibPtr=1 was 0 and from IB[1] if the old ibPtr=1 was 1. When ibFront is loaded by fy = “IBPtr=n”, its value comes from IB[n].

![Figure 6a: Mesa Instruction Buffer States](image)

"+ib" and "+ibNA" cause ibFront to be placed onto the X-bus, while "+ibHigh" puts the high 4 bits, and "+ibLow" the low 4 bits, of ibFront onto the X-bus.

```c
MAR ← [rhl, L + ib],
rhT ← ibLow + 1,
```

c1;  
c2;

"IBDisp" causes a 256-way dispatch based on the value of ibFront. It can only occur in c2 (since only 4 bits of branch/dispatch bits are saved across clicks). The high 4 bits of ibFront replace INIA[4-7], while the low 4 bits of ibFront are OR’d with INIA[8-11]. INIA[0-3] are unaffected, so there are 16 possible 256-way dispatch tables which can be used. The macro "DISPNI" is equivalent to "GOTO".

```c
IBDisp,
DISPNI[OpTable],
```

c2;  
c3;
If IBDisp is executed and ibPtr does not equal 0 (full), a microcode trap is caused to location 400'x for a buffer-empty refill or 500'x for a buffer-not-empty refill. If there is a pending Mesa interrupt request (MInt=1), a microcode trap is caused to location 600'x or 700'x. If either trap occurs the buffer state does not change. "-ib", "IB-", "IBPtr+0", and "IBPtr+1" do not cause an instruction buffer trap. The Error microcode trap to location 0 has priority over the buffer traps.

"AlwaysIBDisp" will never trap--it ignores a pending Mesa interrupt request and a non-full buffer. AlwaysIBDisp is encoded by fY="IBDisp" and fZ="IBPtr+1". If the microinstruction before either an IBDisp or AlwaysIBDisp is a "MAR-" and a PageCross occurs, the IBDisp or AlwaysIBDisp will be canceled and the state of the buffer will remain unchanged.
Figure 7. Mesa Instruction Buffer Sequences
G. Memory

The memory address register is 18 bits wide (expandable to 20 bits) for real addresses and 22 bits for virtual addresses (expandable to 24). Using 16K chips, the memory size is 192K, using 64K chips it is 768K.

1. "MAR<-[rhReg, <arithPhrase>]]" implies a real address reference. "MAR<" can only occur in c1. The first argument specifies which RH register holds the high 2 address bits. The rB field is set to the value of rhReg. Notationally, <arithPhrase> is anything that can occur on the right side of an arithmetic statement.

An RH register should not be loaded simultaneously with a MAR<.

The memory data register can only be loaded in c2. A "MDR<" has no effect unless the previous c1 contained a MAR<. MDR can be loaded by any register, including SU and IOfn.

Data is only delivered from the memory in c3. A "MD<MD" has no effect unless the previous c1 executed a MAR<. Besides R and RH registers, MD can be loaded into IOOut or the Instruction Buffer or be the source of a branch/dispatch.

If a memory location is both read and written in the same click, the old contents of the location is returned:

\[\begin{array}{c}
\text{MAR} & \leftarrow [\text{rhReg}, \text{Reg} + 0], \\
\text{MDR} & \leftarrow \text{RegB}, \\
\text{RegB} & \leftarrow \text{MD},
\end{array}\]  

c1; c2; c3;

2. "MAR<" has 3 important side effects:

(a) "MAR<" forces aS="0,B" and aF="RorS" for the high half of the ALU. This causes the output of the high half of the ALU (bits 0-7) to equal the contents of the R register given by the rB field. Thus, if A bypass is not used, the upper 10 bits of the memory address (the page address) come from the RH-R pair given by the rB field, while the lower 8 bits (the displacement within a page) come from the source defined by <arithPhrase>.

\[\text{MAR} \leftarrow [\text{rhReg}, \text{RegA} + 0],\]  

c1;

causes

\[\begin{array}{l}
\text{YHbus} \leftarrow \text{rhReg}, \\
\text{Ybus[0-7]} \leftarrow \text{Reg[0-7]}, \\
\{\text{NOT RegA[0-7]}\} \\
\text{Ybus[8-15]} \leftarrow \text{RegA[8-15]},
\end{array}\]

If a bypass is used, the Y bus (and MAR) receive the complete R register given by the rA field, but the ALU still delivers rB in the high half and the high 2 bits of address still come from RH[rB]. Thus,

\[\text{MAR} \leftarrow [\text{rhReg}, \text{RegA}], \text{Reg} + \text{RegA} + 1, \{\text{Reg[0-7] unchanged}\}\]  

c1;

causes

\[\begin{array}{l}
\text{YHbus} \leftarrow \text{rhReg}, \\
\text{Ybus[0-7]} \leftarrow \text{RegA[0-7]}, \\
\{\text{NOT RegA} + 1\} \\
\text{Ybus[8-15]} \leftarrow \text{RegA[8-15]} + 1.
\end{array}\]
A consequence of forcing AS=’0.B’ and AF=’RorS’ in the high half of the ALU is that page carries do not propagate into the high half.

\[ \text{MAR} \leftarrow \text{Reg} \leftarrow [\text{rReg}, \text{Reg} + 0F + 1], \quad \{\text{Reg[0-7] unchanged}\} \quad c1; \]

(b) "MAR++" automatically specifies a PageCross branch (See Sec. A). Thus, a change in the flow of control will occur if a page boundary crossing has been indicated by the PageCross branch. This usually implies that a remapping of the real address is necessary.

The PageCross branch occurs in INIA.10 (Not the usual INIA.11 of 2-way branches). Thus, the "BRANCH[LabelX, LabelY, 1]" form must be used after a "MAR++".

\[ \begin{align*}
\text{MAR} & \leftarrow [\text{rReg}, \text{Reg} + 0F + 1], \quad c1; \\
\text{BRANCH}[\text{Continue}, \text{ReMap}, 1], & \quad c2; \\
\text{Continue} & \leftarrow \text{Reg} \leftarrow \text{MD}, \quad c3; \\
\text{ReMap} & \leftarrow \text{Noop}, \quad c3;
\end{align*} \]

The implied PageCross branch can be canceled by a "CANCELBR[Label, 2]".

\[ \begin{align*}
\text{MAR} & \leftarrow [\text{rCnt}, \text{RegA}], \text{Cnt} \leftarrow \text{Cnt} + 1, \quad c1; \\
\text{CANCELBR}[\text{Cont}, 2], & \quad c2; \\
\text{Cont} & \leftarrow \text{Reg} \leftarrow \text{MD} \text{xor Reg}, \quad c3;
\end{align*} \]

Since the 2901 produces carries on logical operations (believe it or not), you must explicitly say "MAR++ [rh, R+0]" (where "MAR++ [rh, R]" was desired) in order to prevent a PageCross branch. In general, if MASS doesn't see a "Reg+0", then the "MAR++" must be followed by either a BRANCH, DISPn, or a CANCELBR. If "Cinp+pc16" is present with "MAR++", then any "MAR++" is an implied branch.

\[ \begin{align*}
\text{MAR} & \leftarrow [\text{rhReg}, \text{Reg} + 0], \quad c1; \\
\text{Noop,} & \quad \{\text{CANCELBR not required here}\} \quad c2; \\
\end{align*} \]

Since the automatic PageCross branch occurs in INIA.10, other branch conditions can be simultaneously specified.

\[ \begin{align*}
\text{MAR} & \leftarrow \text{Reg} \leftarrow [\text{rReg}, \text{Reg} + 1], \text{ZeroBr}, \quad c1; \\
\text{DISP2[Table]}, & \quad c2; \\
\text{Table:} & \quad \text{GOTO[Cont],} \quad \{\text{here if no PgCross, Cnt # 0}\} \quad c3, \text{at[0,4,Table]}; \\
\text{GOTO[Done],} & \quad \{\text{here if no PgCross, Cnt = 0}\} \quad c3, \text{at[1,4,Table]}; \\
\text{GOTO[ReMap],} & \quad \{\text{here if PgCross, Cnt # 0}\} \quad c3, \text{at[2,4,Table]}; \\
\text{GOTO[Done],} & \quad \{\text{here if PgCross, Cnt = 0}\} \quad c3, \text{at[3,4,Table]};
\end{align*} \]

(c) If a PageCross occurs in a "MAR++", then a following "MDR++" or "IBDisp" is canceled. This prevents writing into the wrong memory page (if A bypass was not used in the MAR++) or dispatching on the next Mesa instruction if a page crossing has been indicated.

\[ \begin{align*}
\text{MAR} & \leftarrow \text{Reg} \leftarrow [\text{rReg}, \text{Reg} + 1], \quad c1; \\
\text{MDR} & \leftarrow \text{RegA, BRANCH[Cont, ReMap, 1],} \quad \{\text{MDR++ canceled if we go to ReMap}\} \quad c2;
\end{align*} \]

3. (a) A memory address can be incremented either before or after being sent to MAR.

\[ \begin{align*}
\text{MAR} & \leftarrow [\text{rhReg}, \text{Reg} + 1], \quad c1; \\
\text{MAR} & \leftarrow [\text{rhReg}, \text{Reg} \leftarrow \text{Reg} + 1, \quad c1;
\end{align*} \]

(b) The automatic PageCross branch can be used to indicate the end of a count sequence, where the initial Cnt equals 256-count. rhCnt is used for bank select:

\[ \begin{align*}
\text{MAR} & \leftarrow [\text{rhCnt}, \text{Reg}], \text{Cnt} \leftarrow \text{Cnt} + 1, \quad c1;
\end{align*} \]
H. The Map

The Map is a 16K linear table which is indexed by a 14-bit virtual page number and contains a 10-bit real page number (expandable to 12 bits) and some flags pertaining to the virtual page. The Map is located immediately above the low 64K display bank, real addresses 10000'x to 1JFFF'x. Figure 6 illustrates the mapping process.

1. References to the Map are notationally indicated by "Map+". Either the fX or fZ field must be set to "MapRef". "Map+" causes NO side affects (such as those caused by "MAR+").

"Map+" supplies to the memory system a 22-bit virtual address in YH,Y. An RH register, as addressed by field rB, holds the high 6 bits of the virtual address and an R register typically supplies the low 16 bits. The upper 14 bits of this address are used to index into the Map.

\[
\begin{align*}
\text{Map} + [r\text{Reg}, \text{Reg}] & \quad \text{ct} \\
\text{Map} + \text{Reg} + [r\text{Reg}, \text{Reg} + 0\text{F} + 1] & \quad \text{ct} \\
\text{Map} + [r\text{Reg}, \text{Reg}] & \quad \text{ct} \\
\end{align*}
\]

2. If "MapRef" is specified during a "MAR+", all of MAR+’s side affects occur (see above) and the MapRef correctly causes a read from the Map. If a "MDR+" or "MD" is executed without a preceding "Map+", the "MDR+" or "MD" have no effect.

3. A "Map+" will cause a microcode trap to location 0 (and set the EKEErr register) in the Emulator if either bits 0 or 1 of an Emulator virtual address are non-zero. (There is no trap for IO microcode). The flow of normal Mesa byte code execution stops and a Mesa Xfer occurs.

If a virtual address error occurs, one additional emulator click beyond the one which errored can execute before the emulator begins executing location 0 in cl. Since the Mesa PC and stackP can change in this additional click, they can not be backed up to their original values. The memory address is also lost.
Map Entry Format:

```
+-----------------+---------+---+---+-----------------+
| realPage[4-11]  | dp      | w | d | rp | realPage[0-3]   |
+-----------------+---------+---+---+-----------------+
| 0               |         | 7 |   |                | 15
```

Figure 5. Dandelion Map Reference

9 Feb 80

MapRef.sil
4. The following description is contained in [Iris]<Workstation>ExampleMap.mc.

A map entry has the following format:

```
  rp[4-11]   dp | w | d | rp | rp[0-3]
  0          8  9  10  11
```

where,

rp[0-11], [12-15] real page number. Implies max memory addr of 20 bits \(\Rightarrow\) 1,048,576 words

- dp 8 Dirty & Present
- w 9 Write Protected
- d 10 Dirty
- rp 11 Referenced & Present

The map flags have the following interpretation:

<table>
<thead>
<tr>
<th>W</th>
<th>D</th>
<th>R = RP</th>
<th>DP</th>
<th>Present</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>1</td>
</tr>
</tbody>
</table>

PrincOeps defines vacant to be \(W, D, R = 6\). (Version 1.0 has 6 or 7). Note that Referenced is equivalent to Referenced and Present.

In addition to the three standard bits, W, D, and RP, the bit DP is maintained by the microcode. DP is true if the page is Dirty and Present.

These bits are utilized as follows: When the microcode is doing a MapRef with the intent of reading a word from the page it branches on the RP bit (by using "XRefBr"). If RP is 0, then either the page has not been referenced yet or it is not present; in either case more work must be done. If RP is 1, the microcode can proceed assuming no map maintenance of any kind is required.

If RP was 0, then the microcode does a 4-way dispatch on \(W, D\) and acts according to the following table:

<table>
<thead>
<tr>
<th>W</th>
<th>D</th>
<th>action</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>set RP and do a real memory read</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>set RP and do a real memory read (changes flags = 2 to flags = 3)</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>set RP and do a real memory read</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>Page Fault</td>
</tr>
</tbody>
</table>

Similarly, when the microcode is referencing the Map with the intent of writing a word into the page it branches on the DP bit (by using "XDirtyDisp" = "XLDisp"). Note that the branch occurs in INIA.10). If DP is 0, then either the page has not been written yet, or it is write protected, or it is not present; in any case more work must be done. If DP is 1, the microcode can proceed assuming no map maintenance of any kind is required.

If DP was 0, then the microcode does a 4-way dispatch on \(W, D\) and acts according to the following table:

<table>
<thead>
<tr>
<th>W</th>
<th>D</th>
<th>action</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>set D and DP and do a real memory write</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>{Control shouldn’t reach here}</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>Write Protect Fault</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>Page Fault</td>
</tr>
</tbody>
</table>

IO microcode can quickly check whether the page is present and the flags are correct by using XDirtyDisp or XRefBr.

"MAR-"s feature of forcing "0 or D" in the high half of the ALU is used to combine the real page number from the Map with the displacement into the page given by the low byte of the virtual address.
Mesa Emulator Note: For PageFaults and WriteProtectFaults, the Mesa PC and the stackP must be restored to what they were at the beginning of the Mesa instruction. The STK (and TOS), along with the rest of the Mesa machine state, must be restored if it changed.

Example 1 assumes we have been given a virtual address in rhV.V and we want to map it into a real address and do a read. Example 2 is a possible implementation of the W1 opcode: a virtual write within an MDS. Example 2 uses a subroutine to update the map entry.

{Example 1:
Memory Read given virtual address in rhV.V. R holds real address and data from memory.
}

Start:
Map + [rhV, V],
Noop,
R + MD, rhR + MD, XRefBr,
c1; {rA, fY, fX or fZ unused}
c2;
c3; {rA, fZ unused}

{The MAR+ causes the ALU to output R[0-7],V[8-15] onto the Y bus, where R[0-7] holds the low byte of the real page number and V[8-15] holds the location in the page from the original virtual address. The high 4 real page bits are put onto the YH bus from rhR.}

RedoR:
MAR + [rhV, V], BRANCH[MapUpData, MapOK],
Noop,
R + MD, GOTO[ProcessData],
c1; {rA, fX, fY, fZ unused}
c2;
c3; {rA, fX, fY, fZ unused}

{Either this is the first time we have referenced the page, or it is mapped out}
MapUpDate:
Noop,
[ ] + LRot12 R, XDisp,
c1;
c2;
c3;

Map + [rhV, V], DISP4[FixRFflags, 8],
c1;

FixRFflags:
MDR + R or 10, GOTO[ReRead], {Set RP}
MDR + R or 10, GOTO[ReRead], {Set RP}
MDR + R or 10, GOTO[ReRead], {Set RP}
GOTO[RestoreStateAndPageFault],
c2, at[8,10,FixRFflags];
c2, at[0A,10,FixRFflags];
c2, at[0C,10,FixRFflags];
c2, at[0E,10,FixRFflags];

{Set a pending branch so the BRANCH at RedoRead goes to MapOK}
ReRead:
[ ] + 0, ZeroBr, GOTO[RedoR],
c1;
c2;
c3;

{Example 2: Mesa Opcode W1
Memory Write in MDS. Q holds virtual address. Subroutine to update map: Link0 holds the return address and Link1 encodes the necessary state fixup.}

W1:
Map + Q + [rhMDS, TOS + 1], L1 + n.
T + STK, pop, pCall0,
R + MD, rhR + MD, XDirtyDisp,
c1;
c2, at[n,10];
c3, at[0,10];

RedoW:
MAR + [rhR, Q + 0], {CALL}BRANCH[WMapUD, WMapOK, 1],
c1, at[0, RedoW];
WMaPOK:
MDR + TOS, TOS + T,
PC + PC + 0, Cin+pc16, DISP4[@Noop],
c2;
c3;

{Subroutine to do map updates for writes.}
WMaPUD:
Noop,
[ ] + LRot12 R, XDisp,
c2;
c3;

Map + [rhMDS, Q], L1Disp, DISP4[FixWFlags, 1],
c1;

FixWFlags:
MDR + R or 0A0, pRet0, CANCELBR[ReWrite, 0F],
MDR + R or 0A0, pRet0, CANCELBR[ReWrite, 0F],
T + aWriteProtect, DISP4[WTrap],
T + aPageFault, DISP4[WTrap],
c2, at[0,10,x];
c2, at[2,10,x];
c2, at[4,10,x];
c2, at[6,10,x];

ReWrite:
[ ] + 2, XDisp, RET[RedoW],
c3;

WTraP:
push, GOTO[PageFault],
c3, at[n,10,WTrap];
I. Bus Sinks

<table>
<thead>
<tr>
<th>Destination</th>
<th>Source Bus</th>
</tr>
</thead>
<tbody>
<tr>
<td>MAR+</td>
<td>Y,R,Y</td>
</tr>
<tr>
<td>MDR+</td>
<td>Y</td>
</tr>
<tr>
<td>STK+</td>
<td>Y</td>
</tr>
<tr>
<td>stackP+</td>
<td>Y</td>
</tr>
<tr>
<td>U+</td>
<td>Y</td>
</tr>
<tr>
<td>IB+</td>
<td>X</td>
</tr>
<tr>
<td>RH+</td>
<td>X</td>
</tr>
<tr>
<td>KOData+</td>
<td>X</td>
</tr>
<tr>
<td>XOData+</td>
<td>X</td>
</tr>
<tr>
<td>POData+</td>
<td>X</td>
</tr>
<tr>
<td>IOPOData+</td>
<td>X</td>
</tr>
<tr>
<td>DClitFifo+</td>
<td>Y</td>
</tr>
<tr>
<td>DBorder+</td>
<td>Y</td>
</tr>
<tr>
<td>KClt+</td>
<td>X</td>
</tr>
<tr>
<td>XClit+</td>
<td>X</td>
</tr>
<tr>
<td>MClt+</td>
<td>Y</td>
</tr>
<tr>
<td>IOPClit+</td>
<td>X</td>
</tr>
<tr>
<td>DClit+</td>
<td>X</td>
</tr>
<tr>
<td>PClit+</td>
<td>X</td>
</tr>
</tbody>
</table>

Memory Address Register
Memory Data Register
Stack (SU reg addr = 0.,stackP) (enSU = 1, Cin = 1, fS.2 = 0)
Stack Pointer
U register (SU reg addr = fA.,fZ) (enSU = 1, Cin = 1, fS.2 = 1)
Mesa instruction Buffer
RH registers
Rigid Disk Output Data reg
Xerox Wire Output Data reg
Printer Output Data reg
IOP Output Data reg
Display Output Data Control Fifo
Display Output Data border register
Rigid Disk Control reg
Xerox Wire Control reg
Memory Control reg
IOP Control reg
Display Control reg
Printer Control reg

J. Bus Sources

Note that Xbus bits [0-7] are set to zero when Nibble, Byte, or IOXIn values 8 through 0F are specified.

<table>
<thead>
<tr>
<th>Source</th>
<th>Destination Bus</th>
</tr>
</thead>
<tbody>
<tr>
<td>+Y LRot0</td>
<td>X</td>
</tr>
<tr>
<td>+Y LRot4</td>
<td>X</td>
</tr>
<tr>
<td>+Y LRot8</td>
<td>X</td>
</tr>
<tr>
<td>+Y LRot12</td>
<td>X</td>
</tr>
<tr>
<td>fZ</td>
<td>X</td>
</tr>
<tr>
<td>fY,fZ</td>
<td>X</td>
</tr>
<tr>
<td>+MD</td>
<td>X</td>
</tr>
<tr>
<td>+STK</td>
<td>X</td>
</tr>
<tr>
<td>+U</td>
<td>X</td>
</tr>
<tr>
<td>+RH</td>
<td>X</td>
</tr>
<tr>
<td>+ib</td>
<td>X</td>
</tr>
<tr>
<td>+ibNA</td>
<td>X</td>
</tr>
<tr>
<td>+ibLow</td>
<td>X</td>
</tr>
<tr>
<td>+ibHigh</td>
<td>X</td>
</tr>
<tr>
<td>+KIData</td>
<td>X</td>
</tr>
<tr>
<td>+XData</td>
<td>X</td>
</tr>
<tr>
<td>+IOPData</td>
<td>X</td>
</tr>
<tr>
<td>+ErrIBstkP</td>
<td>X</td>
</tr>
<tr>
<td>+KStatus</td>
<td>X</td>
</tr>
<tr>
<td>+XStatus</td>
<td>X</td>
</tr>
<tr>
<td>+PStatus</td>
<td>X</td>
</tr>
<tr>
<td>+MStatus</td>
<td>X</td>
</tr>
<tr>
<td>+IOPStatus</td>
<td>X</td>
</tr>
</tbody>
</table>

Left Rotate Y by 0
Left Rotate Y by 4
Left Rotate Y by 8
Left Rotate Y by 12
0-0F 4-bit constants
0-0F 8-bit constants
Memory Data
Stack (SU reg addr = 0.,stackP) (implies enSU = 1, Cin = 0, fS.2 = 0)
U register (SU reg addr = fA.,fZ) (implies enSU = 1, Cin = 0, fS.2 = 1)
RH registers
Instruction Buffer
Instruction Buffer (doesn’t advance IBPtr)
ib[4-7]
ib[0-3]
Rigid Disk Input Data
Xerox Wire Input Data
IOP Input Data
EmuErrAB, IBPtr, stackP
Rigid Disk Status
Xerox Wire Status
Printer Status
Memory Status (Syndrome, SE, DE, ErrLog)
IOP Status reg
K. Miscellaneous Functions

Carry in to ALU or into Shift Ends is pc16
Carry in to ALU or into Shift Ends is pc16
Carry in to ALU or into Shift Ends is 0
Carry in to ALU or into Shift Ends is 1
Carry in to ALU or into Shift Ends is 0
Carry in to ALU or into Shift Ends is 1
Causes a microcode trap to 600x or 700x if Mesaint pending,
trap to 400x or 500x if buffer not full.
Sets Mesaint Interrupt Request flag
ibFront = ib[0]
ibFront = ib[1] (With iBDisp, changes to AlwaysIBDisp)
Encoded by iY = iBDisp & iZ = iBPtr+1 (prevents iBDisp from
Clear error bits & interrupt req bit
Breakpoint - go to Kernel task
Leave Kernel task and commence regular task scheduling
Memory refresh (only in cycle 1, ignored in c2 or c3)
Increment stackP by 1 (trap to loc. 0 if overflow)
Decrement stackP by 1 (trap to loc. 0 if underflow)
Reset Display and Printer Requests
Reset Rigid Disk Flags
Reset IOP Request
Reset Xerox Wire Request
assure noop F fields
L. Microcode Conventions:

The file [Iris]<Workstation>mc<form.dm> embodies the following suggested conventions.

1. All microcode should be in either font Helvetica8 or Helvetica 10. (The [Hardcopy] section of user.cm should be updated to cause Empress to use this font instead of Gacha8. Returns should be used for spacing since Empress doesn't know about paragraph looks and will otherwise squish the paragraphs together. Empress will also change the tabs, but the source is still readable.)

2. The tab stops for Helvetica8 microcode are 140 pt., 160 pt., and 420 pt. The stop at 160 keeps lines with long labels from jumping over to the comment field when Bravo is not in hardcopy mode. (160-140 is less than the length of most statements.) In form.mc, the tab stops for comments are 140, 160, 180, 200, and 420 pts.)

3. The top of the microcode file should have at least the following information:

   {File name: <Name>.mc
    Description: <what file contains>
    Author: <being>,
    Created: <date>,
    Last Edited: <date & time>}

4. The general format of a line is:

   <Label> <Arrow clause>, <functions>, <DispBr>, <Call/Red>, <GOTOField>, <cycle>, <at>

   There should always be one space after a comma. Those statements which are executed together within the same click should be preceded and followed by a blank line. The <cycle> and <at> clauses should be located after the 420 pt. tab. It also helps to locate the last comma at the 420 stop. For example:

   TOS + TOS + 1, c3;
   Shift: [], PC + PC + 0, Cin = pc16, pCall1, TT + STK (TT + v), c1;
   TOS + T LRot1, DISP4[maskTbl], c1;

5. Comments can be embedded within a line.

6. MASS generates the required "at" phrases after BRANCH's, but not for dispatch tables. For readability, it helps if "at" clauses are explicitly placed to identify their partners when more than one line away. For example, the "at" clause of "MulLoop" points to the instruction "MLEnd", which then points back to MulLoop.

   MulLoop: [] + Q, YDisp,
            TT + TT - 1, ZeroBr, BRANCH[MPlier0, MPlier1, YOdd], c1, at[0,2,MLEnd];
            TT + TT - 1, ZeroBr, BRANCH[MPlier0, MPlier1, YOdd], c2;
   MPlier0: T + T, DRShift1, BRANCH[MulLoop, MLEnd], c3, at[];
   MPlier1: T + T + TOS, DRShift1, BRANCH[MulLoop, MLEnd], c3, at[];
   MLEnd: STK + T [long.high/rem], pop [point at long.low/quot], c1, at[1,2,MulLoop];

7. If names consist of multiple parts, the first letter of each sub-name should be capitalized. All names should not be all capitalized. For example: MulLoop, JumpSign, MapOK. "i" should be used instead of the capital form "I", which is indistinguishable from "i" (small L) and similar to "i" (one). Try to avoid a solitary "0" in a name since it looks like a "o" (zero).
8. All instances of user names and reserved words must follow the same capitalization, else MASS will not recognize them. (Searching problems in Bravo are reduced.)

9. U register names should start with the letter "U" or "u", and RH register names with "RH" or "rh". For example: Uinterrupt. uTemp, RHxK, rhRtemp.

10. Arithmetic clauses (i.e., arrow clauses) should have a space before and after the →. The → character can be used for the minus sign instead of -, which is microscopic in size (in all fonts). In Bravo type-in mode, this minus sign is entered by typing: n Ctrl-S ESC Look g (see p.57 Bravo manual). It looks like a - in bravo. For example:

\[\begin{align*}
T & \leftarrow 17 \ xor \ T, \ \text{stackP} \leftarrow 17, \\
T & \leftarrow TOS - T, \\
\text{c2} & , \\
\text{c3} & \end{align*}\]

11. The allocation and definition of registers used in the complete microcode system are given in [Iris]<Workstation>Dandelion.df.

Dandelion.df should ONLY be accessed via the Librarian Access Tool. Thus, "Mesa Access checkOutReason/r Dandelion/o" is used to check Dandelion.df out and move it to your disk, and "Mesa Access Dandelion/i" is used to return it to Iris.

The following files are needed on your disk: RunMesa.run, Mesa.image (or BasicMesa.image), and [Igor]<AlphaTools>Access.bcd.

Your user.cm should include the following entry:

```
[Librarian]
Server: Marion
NamePrefix: WorkStation
NameSuffix: df
```

M. MASS

1. Overview

There are two types of source files. There are macro-and-defs files and there are microcode source files. The macro-and-defs files have a ".df" extension and microcode files have a ".mc" extension. ".df" files do not contain microinstructions and are global to an assembly. ".mc" files may have macros and defs. Intermediate files are produced for each micro-code source file to allow subsequent partial re-assemblies: ".ml", ".si", and ".eb" files are produced from every ".mc" file.

The ".fb" is the Burdock loadable file of allocated microinstructions. The ".ft" is a human readable form of the ".fb" file. Burdock also reads the MASS produced ".st" symbol table file.

While MASS is running, the cursor is reversed once per statement processed. During the allocation phase, a square cursor is displayed with the number of enclosed dots increasing to six. If errors are found in the file, MASS pauses with the cursor showing "hit me". As soon as any keyboard character is struck, MASS returns to the Alto executive (which may continue on to Bravo if the S macro was previously invoked).

The ".er" status file contains MASS version information and the errors encountered. Along with the complete statement which is in error, the phrase which caused the error is written into the status file. MASS actually displays an error message for each possible (and unsuccessful) way to encode the statement.

2. File Summary

The following files are used (u) or generated (g) by MASS in pass 1 or 2.

<table>
<thead>
<tr>
<th>File</th>
<th>Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>.mc u</td>
<td>MicroCode source</td>
<td></td>
</tr>
<tr>
<td>.df u</td>
<td>macro and Defs source File</td>
<td></td>
</tr>
<tr>
<td>.eb ug</td>
<td>Early Binary (output of Pass 1, one per &quot;.mc&quot; file)</td>
<td></td>
</tr>
<tr>
<td>.ml ug</td>
<td>Label constraint records &amp; Reserves (output of Pass 1, one per &quot;.mc&quot; file)</td>
<td></td>
</tr>
<tr>
<td>.si ug</td>
<td>Symbol Intermediate (output of Pass 1, one per &quot;.mc&quot; file)</td>
<td></td>
</tr>
<tr>
<td>.fb ug</td>
<td>Final Binary (output of Pass 2)</td>
<td></td>
</tr>
<tr>
<td>.ft g</td>
<td>Final binary Text (output of Pass 2)</td>
<td></td>
</tr>
<tr>
<td>.st g</td>
<td>Symbol Table (output of Pass 2)</td>
<td></td>
</tr>
<tr>
<td>.er g</td>
<td>status-error (output of Pass 2)</td>
<td></td>
</tr>
</tbody>
</table>

3. Command Line Switches

- `noswitch` use name for ".mc" input file and ".eb", ".ml", & ".si" output files.
- `/d` designates a ".df" file
- `/o` use name for ".er", ".fb", ".ft", and ".st" output files
- `/2` pass 2 only for this file (it has been previously assembled)
- `/x` satisfy imports from this file & exclude its locations from allocation
4. Sample Assembly Descriptions

MASS regs/d macros/d acode aout/o

Pass1:
reads: regs.df, macros.df, acode.mc
writes: acode.eb, acode.ml, acode.si

Pass2:
reads acode.ml, acode.si
writes aout.st
allocates acode
reads acode.eb
form final binary of acode
writes aout.fb, aout.ft, aout.er

MASS config1a/d source1/2 source2 source3/x sourceout/o

Pass1:
reads: config1a.df, source2.mc
writes: source2.ml, source2.eb, source2.si

Pass2:
reads source1.ml, source1.si
source2.ml, source2.si
source3.fb
writes sourceout.st
allocates source1, source2
reads source1.eb, source2.eb
form final binary of source1, source2
writes sourceout.fb, sourceout.ft, sourceout.er

5. MASS Format descriptions:

Comments: All text between squiggly bracket pairs "{" and "}" is comment. The brackets nest, so they must be properly paired. (This is so sections of code which already contain comments can be commented out)

Labels & numbers: The case of letters in names and macros is important and must be consistent. Hexadecimal numbers are assumed, although a decimal number is specified by a trailing 'd' and an octal number by a trailing 'b'. Hex numbers which start with A-F must be prefixed by a zero: 0F = 0Fx = 15'd = 17'b.

Source Line Format: A line of source which defines a single microinstruction is a list of clauses which are terminated by semicolons. Clauses are separated from one another by commas. A clause is either a function field name, a macro invocation, or an arrow (arithmetic) clause. A name followed by a colon is the label of the microinstruction. Microinstructions can have multiple labels. Spaces are ignored (and this does not mean they are significant in variable names). Parentheses are used in arithmetic clauses (where they may improve readability but are ignored by MASS) and in macro definitions. Brackets are used to denote the argument list to macros (and used in "MAR+[..]", "MDR+[..]", and "MD").

Cycle Numbers: Each source line must have a cycle number macro: c1, c2, c3, or c*. c* inhibits wrong cycle warning from MASS. This is primarily used by the Kernel, but can be used in loops which are not a multiple of three microinstructions and short subroutines which don't contain cycle constrained operations. Such loops must always exit on the same cycle number and should not contain "MAR+", "MDR+", or "MD" since these depend on executing in cycle 1, 2, or 3, respectively. Burdock can not breakpoint instructions with a c*.
Register definition: All registers must be defined before their names are used. The macro "RegDef" is used to associate a name with a type (R, RH, U or UY) and a register number.

\[
\begin{align*}
\text{RegDef}[\text{Reg}, R, 4]; \\
\text{RegDef}[\text{Ureg}, U, 47]; \\
\text{RegDef}[\text{rhReg}, RH, 4]; \\
\text{RegDef}[\text{Ublock}, UY, 4];
\end{align*}
\]

Arrow formats: Arrow statements, like all clauses, are isolated by commas. They may have one or two destinations, as in

\[
\begin{align*}
B &= B \text{ or } A, \\
B &= SU \text{ or } B \text{ or } A,
\end{align*}
\]

The right side of an arrow clause (the source) consists of a single entry, a double entry with an operator, or a triple entry with two (identical) operators. It is also possible for an entry to have a qualifying unary operator ("-" or "+"). For example:

\[
\begin{align*}
B &= -A, \\
B &= A + 1, \\
B &= A + B + 1, \\
B &= A \text{ and } B,
\end{align*}
\]

Constants can be 4 or 8 bits long. Note that a special form must be used for $-1$:

\[
\begin{align*}
\text{Reg} &= 0FE; \\
\text{Reg} &= -0F \text{ xor Reg}, \\
\text{Reg} &= -5, \\
\text{Reg} &= \text{Reg xor Reg}, \quad \text{used instead of Reg} - 1, \text{ or Reg} - 0
\end{align*}
\]

MASS expects the single-bit-shift phrases to precede any arithmetic clauses and 4-bit rotate phrases (LRotn) to follow the quantity to be rotated. Parentheses are ignored. See sections A.1 and A.2.

\[
\begin{align*}
\text{Reg} &= \text{LSHIFT Reg}, \\
\text{Reg} &= \text{RSHIFT} \text{ (Reg and OF)}, \\
\text{Reg} &= \text{LRot} \text{ (Reg + 1)}, \\
\text{Reg} &= \text{Reg LRot4}, \\
\text{Reg} &= (\text{Reg LRot8}) \text{ or Reg}, \\
\text{rhReg} &= (\text{RegA + Reg}) \text{ LRot12},
\end{align*}
\]

Default Labels: A "$\$" can be used in BRANCH and CANCELBR macros to refer to the location of the next statement.

\[
\begin{align*}
0 \text{Begin BRANCH}[\text{MapUpdate, }\$], \\
\text{CANCELBR[}$\$],
\end{align*}
\]

External variables: The macro "IMPORT" indicates which labels of the ".me" file have been defined elsewhere. The "EXPORT" macro specifies labels which other modules will import. When MASS is assembling a group of files together, called an "assembly unit", (a combination of ".me" and/or ".ml"/".si" files) it is illegal to have the same label defined twice or imported and defined. Labels defined in IMPORT macros will be assigned values from the EXPORT of a previous assembly unit by using the /x switch on its name.

Macro Definition: A macro is defined by supplying a name for the macro and a text string which will replace the name. Macros return nothing or a single number or variable (never a pair of arguments, for instance). Arguments (up to 9) may be supplied in the macro call and are referred to in the expansion by "#n" for n = 0 to 9. The "#0" is replaced by the number of arguments in the macro call, and #1-9 will be replaced with the appropriate argument (or null if there is none). Parentheses must be used to enclose arguments containing commas. For example:
Macro Def[NewMacro, (OldMacro1[#1, OldMacro2[#2, #3])];

Macro Invocation: The invocation of a macro is caused by supplying the macro and optionally up to nine arguments. Arguments are separated by commas, and if an argument contains a comma, it must be enclosed within parentheses. Macro calls cannot appear on the left side of arrow clauses, and only unargumented macros (such as macros defined by Set) can appear on the right side.

If an "mc" file contains macro definitions, they remain valid for subsequent "mc" files which are part of the same assembly unit.

Reserved Names: None of the function field or builtin macro names should be used as user names. For example, RH, ib, Q, XStatus, c1, Refresh, xor, and pop are all reserved. Null and Apass are also reserved names. "*" should not be present in user names.

6. Bravo S macro:

The S macro facilitates ping ponging between Bravo and MASS. It assumes "standard" file naming conventions. In particular, corresponding to a "Name.mc" microcode file, there is a "Name.cm" command file, and errors are placed into a "Name.err" file. A typical "Name.cm" file is:

```
MASS Dandelion/d Name/o Name
```

The macro is invoked by "BRAVO/S Name". This causes the "mc" and "er" files to be fetched into bravo. When leaving bravo (and the top file window contains Name.mc), it is invoked by typing "QS<return>", which causes deletion of the old "er" file, execution of the "cm" command file, and re-entry to bravo via the S entry macro. [Iris]<Workstation>MASS>user.cmslice contains the bravo macros.

7. Conditional Code Generation

The "IfEqual", "IfGreater", "IfAndZero" and "SkipTo" macros can be used so that MASS will not assemble sections of code. The values of the variables used by the macros can be set on the command line by phrases of the form "[varName,value]" (no spaces). MASS treats the bracketed pair as an argument to the "Set" macro.

8. Builtin Macros:

```
Add[arg1, ..., arg9]  -- adds up to 9 arguments
Mul[arg1, ..., arg9]  -- multiplies up to 9 arguments
And[arg1, ..., arg9]  -- ands up to 9 arguments
Or[arg1, ..., arg9]   -- ors up to 9 arguments
Xor[arg1, ..., arg9]  -- xors up to 9 arguments
LShift[arg1, arg2]   -- left shift arg1 by arg2
RShift[arg1, arg2]   -- right shift arg1 by arg2
Sub[arg1, arg2]      -- arg1 — arg2

GOTO[label]
CALI[label]
GOTOABS[value]
BRANCH[[label0, label1, mask]  -- mask optional
CANCELBIT[label, mask]       -- mask optional
DISP2[label, mask]           -- 2-bit dispatch, mask optional
DISP3[label, mask]           -- 3-bit dispatch, mask optional
DISP4[label, mask]           -- 4-bit dispatch, mask optional
DISPN[label]                 -- 8-bit IDisp dispatch.
RET[label]
```
IFDEF\[x, y, equalVal, unequalVal\]  -- IF x = y THEN equalVal ELSE unequalVal
IF\[x > y\]  -- IF x > y THEN equalVal ELSE unequalVal
IF\[x \& y = 0\]  -- IF x \& y = 0 THEN equalVal ELSE unequalVal
SkipTo\[label\]  -- MASS skips code until label! (I must be appended to label)
EXPORT\[label1, ..., label9\]  -- Declares up to 9 values exportable
IMPORT\[label1, ..., label9\]  -- Declares up to 9 values importable
at\[offset, modulo, label\]  -- modulo defaults to 4096, label to current label
Reserve\[LowAddr, HighAddr\]  -- micro-informs can't be allocated within this range.
Set\[varName, value\]  -- sets variable named \varName\ to value of \value\; "-" not
לכאaned.
MacroDef\[macroName, expansion\]  -- print variable in "..cr" file
RegDef\[regName, regType, regAddress\]  -- \regType\ = \{U, R, RH, UY\}
PrintVar\[varName\]  -- puts "Message" in "..cr" file (only letters, numbers, and ".")
SetTask\[task\]  -- in effect until the next SetTask. task in [0..7]
StartAddress\[label\]  -- Burdock initializes task's TPC to this value.
N. Burdock-Kernel

1. Overview

This section deals with 4 of Burdock's numerous windows: the CP Panel, State Analyzer, Files, and an Empty window. The Empty window can be used to load source files such as "mc" or "rt" files. The Files window accepts the file names for the "fb", "st", and "cpr" files. The Analyzer window reads and formats the output of the Tektronics 7904 (DF1 formatter) logic analyzer. The CP Panel reads and writes CP registers and executes CP related commands.

2. CP Panel - Commands

The CP Panel can execute one of the following commands: Boot, Load, Start, Stop, Continue, Break, Unbreak, or LoadReal.

The "Boot" command boots the IOP, if necessary, and then loads the CP kernel (KernelDion.fb). All the TPC's are initialized to "0FDF", which currently is an microinstruction which loops on itself and resets the display controller. Note that while the kernel is executing at the kernel task level, no other tasks in the Dandelion can run.

"Load" loads the "fb" file into the control store and reads the "st" symbol table file into Burdock. The file name come from the Files window. If the original source files contained "SetTask" and "StartAddress" macros, the specified TPC's are initialized.

"Start" loads the emulator (task 0) TPC with the value of the given label and then causes the CP to stop executing the kernel. Normal task scheduling begins, and if multiple tasks are enabled, it is not known which will begin executing first (since the kernel will stop executing on an arbitrary click within a round).

Since the CP-task-specific register which holds condition bits between clicks (the TC register) can not be written, the microinstruction which will be started must contain a "CANCELBR[Label, 0F]", i.e., the address of the second instruction executed must be "at[0F,10]".

"Stop" causes the kernel task to run, thereby blocking all other tasks from running. The tasks which were running are interrupted on a click boundary, so the TPC's and TC's for the tasks remain valid. It is not possible to determine which click boundary the stop occurred on.

"Continue" exits the kernel given the current values of the TPC's. Like Start, it is not known which task will begin executing first.

"Break" sets a breakpoint at the address given by the label. Burdock saves the breakpointed microinstruction in an internal table and replaces it with the appropriate instruction which will cause entry into the kernel. If the control store is examined from Burdock, the breakpoints will be visible and not the original instructions. "UnBreak" restores the breakpointed location with its original contents.

There are 4 types of breakpoints: cycle1, cycle2, shortCycle3, and cycle3. Their breakpoint identification numbers are [10..1F], [20..2F], [1..7], and 0, respectively. Thus, there are 16 cycle1, 16 cycle2, and 8 cycle3 breakpoints available. (Burdock can be expanded to accommodate 256 cycle1 and 256 cycle2 breakpoints). The cycle1 and cycle2 breakpoints correctly save all possible pending dispatch/branch bits (i.e., they correctly breakpoint an instruction containing a DISP4] or RET]). However, there is only one c3 breakpoint (id_num 0) which has this property. The other seven c3 breakpoints (called shortCycle3 breakpoints) only correctly handle instructions containing at most 2-way BRANCH's, i.e., it may not be possible to correctly resume from a shortCycle3 breakpoint set on a statement which specifies a DISP4, DISP3, DISP2, or RET. (This restriction is necessary in order to save memory data if the click causes a memory read. The kernel could be changed in order to provide more versatile cycle3 breakpoints at the cost of losing memory data.)
Clicks which contain memory reads ("+MD") can be correctly restarted after a breakpoint, but cycle2 of clicks which contain "MDR+" can not be (since the memory address is lost). Clicks containing memory reads are restarted by loading main memory location 0 (0FFDD instead?) with the data which was arriving from memory during the breakpointed click and then restarting the memory to read from mem[0] when resuming from the breakpoint.

If a cycle1 breakpoint is executed, the kernel will always run in the following task. However, if a cycle2 or cycle3 breakpoint is executed, the breakpointed task must run for at least one more click in order for the breakpoint to take effect. This is never a problem with the emulator (since it will eventually run), but could be a problem with an IO task if the task’s request were disabled during the breakpointed click, thereby preventing it from executing again. Note that for cycle2 and cycle3 breakpoints it is not known which other tasks may have run between the breakpointed click and the kernel entry.

If "Start" is used instead of "Continue" after a breakpoint, the second instruction executed must be "al[0F,10]." This requirement is not necessary if the TC register of the task does not have any non-zero bits where the low 4 bits of the address of the second instruction has zero bits. For example, if the TC register is 0, then any microinstruction can be Started. If the TC register is 1, then only microinstructions which are followed by an instruction at an odd address can be started. (The TC register for a task can be written by executing a single microinstruction at the task level which sets the bits accordingly.)

3. CP Panel - Registers

After the CP has been booted, its registers can be read and written via absolute addresses. After a program has been loaded, Burdock can read and write registers given their symbolic names. Burdock ignores capitalization in all names. The control store, TPC, and TC registers can be read or written before the CP is booted.

The format for reading and writing a register with absolute addresses is ".name address" or "address .name". An address without a ".name" is assumed to be a real memory address. Burdock recognizes the following register names:

<table>
<thead>
<tr>
<th>.name</th>
<th>address range</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>.q</td>
<td>[0..0F]</td>
<td></td>
</tr>
<tr>
<td>.r</td>
<td>[0..0F]</td>
<td></td>
</tr>
<tr>
<td>.rh</td>
<td>[0..0FF]</td>
<td>read only</td>
</tr>
<tr>
<td>.u</td>
<td>[0..0FF]</td>
<td></td>
</tr>
<tr>
<td>.tc</td>
<td>[0..7]</td>
<td></td>
</tr>
<tr>
<td>.tpc</td>
<td>[0..7]</td>
<td></td>
</tr>
<tr>
<td>.link</td>
<td>[0..7]</td>
<td></td>
</tr>
<tr>
<td>.ioxin</td>
<td>[0..0F]</td>
<td>address is value of EZ--read only</td>
</tr>
<tr>
<td>.iout</td>
<td>[0..0F]</td>
<td>address is value of FY--write only</td>
</tr>
<tr>
<td>.cr0</td>
<td>[0..0FF]</td>
<td>control store real--first word</td>
</tr>
<tr>
<td>.cr1</td>
<td>[0..0FF]</td>
<td>control store real--second word</td>
</tr>
<tr>
<td>.cr2</td>
<td>[0..0FF]</td>
<td>control store real--third word</td>
</tr>
<tr>
<td>.mr</td>
<td>[0..3FFFF]</td>
<td>memory real address</td>
</tr>
<tr>
<td>.mv</td>
<td>[0..3FFFF]</td>
<td>memory virtual address</td>
</tr>
<tr>
<td>.map</td>
<td>[0..3FFF]</td>
<td>index into Map</td>
</tr>
<tr>
<td>.ib</td>
<td></td>
<td>doesn’t effect ibPtr</td>
</tr>
<tr>
<td>.ibPtr</td>
<td></td>
<td>can only be set to 1 or 3</td>
</tr>
<tr>
<td>.stackP</td>
<td></td>
<td></td>
</tr>
<tr>
<td>.pc16</td>
<td></td>
<td></td>
</tr>
<tr>
<td>.MInt</td>
<td></td>
<td>Mesa Interrupt</td>
</tr>
<tr>
<td>.EKErr</td>
<td></td>
<td>error register--read only</td>
</tr>
</tbody>
</table>
Currently, if the display task (1) was executing at the time of a breakpoint, a 2 (or 0) should be written into ".ioout 7" in order to disable the display controller, thereby preventing it from reading the low bank so that the low bank can be read/written correctly by the kernel. (This is also necessary for proper resumption after a cycle2 or cycle3 breakpoint of a click which reads memory.) This problem will be fixed.

4. State Analyzer Window

For CP debugging, the low 12 bits of the state analyzer are connected to the control store address lines (NIA[0..11]). The upper 4 bits are available for other inputs (such as cycle number, task number, etc). The Analyzer Tool can format, filter, and search the addresses for matches. The "mask" is and'd with all displayed addresses and the "xor" mask is used to invert appropriate bits. The default for the xor mask is 0FF since the control store address lines are inverted.

The addresses can be displayed in one of 4 modes: NIA (symbolic labels), hex, octal, or binary. Since NIA is displayed, the occurrence of a label corresponds to the cycle when the instruction is being read from the control store—the indicated instruction is executed in the following cycle. Thus, if cl is being displayed in the top bits, it actually corresponds to microinstructions labeled c2 in the source file.

5. Real Memory Loader

The LoadReal command loads real memory with the ".cpr" file given by the Files window. Such a file contains one or more blocks in the following format:

<count>,<addrHigh>,<addrLow>,<data1>,...<dataCount>.

A program exists (stored on [Iris]<DDavies>MakeBinFile.bcd) which translates text files into this format.

6. Command files

<Will be documented later>

7. Microcode Error Traps

If a microcode trap occurs to location 0, the kernel will treat it as a breakpoint (id=0FF'x). "Reserve[0, 0]" must be present in your program unless you are going to handle the error. The .EKEErr register identifies the microcode trap:

<table>
<thead>
<tr>
<th>Number</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>Control Store Parity Error</td>
</tr>
<tr>
<td>1</td>
<td>Emulator Double-bit Memory Error</td>
</tr>
<tr>
<td>2</td>
<td>Stack Overflow or Underflow</td>
</tr>
<tr>
<td>3</td>
<td>Emulator Virtual Address Out of Range (&gt;22 bits)</td>
</tr>
</tbody>
</table>
O. Timing Constraints - Allowable Bus Operations:

The following two figures should answer the question: "Can my microinstruction accomplish its task in time (i.e. before the end of the cycle)?" The first table deals with the X-bus, and the following with the Y-bus.

Note that ALL operations which do not output onto the X or Y bus (i.e. internal 2901 register to register operations) complete on time (see schematics for timing information). (Note that +1 or -1 do not imply use of the X-bus.)

If your microinstruction contains an X-bus operand, then Figure 7 will tell you whether that microinstruction will complete in time. At the intersection of the appropriate "X Source" column and the "X Operation" row, if the number is not in bold type (i.e. any cycle times less than 140 nS), then the microinstruction is OK from a timing viewpoint.

MASS checks for the timing violations given in the following tables. The macro "SuppressTimingWarning" can be used to prevent an error message where the timing is OK for a partial result (for example, addition in the low 4 bits).

The ALU performs arithmetic at 3 different speeds depending on which bits of the result you're looking at. Bits[0-7] are the slowest (they depend on a carry from the lookahead unit; Bits[8-11] are next (they depend on a ripple carry from the low nibble); and Bits[12-15] are fastest (Cin arrives very early relative to Xbus sources). The low nibble always has the timing of a corresponding ALU logic operation. For example "Reg + Ureg or Reg" has the same timing as "Reg + Ureg + Reg" for the low 4 bits only.

\[
\begin{align*}
\text{Rreg + Ureg - 1, YDisp, \{only low 4 bits OK\}} & \quad \text{c2;} \\
\text{Rreg + RHreg + 1, YDisp, \{only low 4 bits OK\}} & \quad \text{c2;} 
\end{align*}
\]
<table>
<thead>
<tr>
<th>X Source</th>
<th>Setup</th>
<th>SU</th>
<th>MD</th>
<th>RH</th>
<th>Nibble</th>
<th>Byte</th>
<th>IB</th>
<th>IffIBstkP</th>
<th>IOIn</th>
<th>A.Rotn</th>
<th>(A.or.B.)</th>
<th>L.Rotn</th>
<th>(A+B)</th>
</tr>
</thead>
<tbody>
<tr>
<td>X+</td>
<td>max (59, X+)</td>
<td>75</td>
<td>97</td>
<td>64</td>
<td>50</td>
<td>50</td>
<td>59</td>
<td>59</td>
<td>63</td>
<td>91</td>
<td>103</td>
<td>102</td>
<td></td>
</tr>
<tr>
<td>B+X.or.A</td>
<td>40</td>
<td>115</td>
<td>137</td>
<td>104</td>
<td>99</td>
<td>99</td>
<td>99</td>
<td>99</td>
<td>103</td>
<td>131</td>
<td>131</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X.or.A, ZeroBr</td>
<td>58</td>
<td>133</td>
<td>155</td>
<td>122</td>
<td>117</td>
<td>117</td>
<td>117</td>
<td>121</td>
<td>149</td>
<td>149</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X.or.A, NegBr</td>
<td>58</td>
<td>133</td>
<td>155</td>
<td>122</td>
<td>117</td>
<td>117</td>
<td>117</td>
<td>121</td>
<td>149</td>
<td>149</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>[]+X.or.A, YDisp</td>
<td>68</td>
<td>143</td>
<td>165</td>
<td>132</td>
<td>127</td>
<td>127</td>
<td>127</td>
<td>131</td>
<td>159</td>
<td>159</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X.or.A, LShift1</td>
<td>50</td>
<td>125</td>
<td>147</td>
<td>114</td>
<td>109</td>
<td>109</td>
<td>109</td>
<td>113</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X.or.A, L.Rotn</td>
<td>59</td>
<td>134</td>
<td>156</td>
<td>123</td>
<td>118</td>
<td>118</td>
<td>118</td>
<td>118</td>
<td>122</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MAR+X.or.A</td>
<td>78</td>
<td>—</td>
<td>142</td>
<td>137</td>
<td>137</td>
<td>137</td>
<td>137</td>
<td>141</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>MDR+X.or.A</td>
<td>46</td>
<td>120</td>
<td>—</td>
<td>109</td>
<td>104</td>
<td>104</td>
<td>104</td>
<td>108</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SU+X.or.A</td>
<td>87</td>
<td>—</td>
<td>184</td>
<td>151</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>159</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>IOYOut+X.or.A</td>
<td>64</td>
<td>139</td>
<td>161</td>
<td>128</td>
<td>123</td>
<td>123</td>
<td>123</td>
<td>127</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A</td>
<td>74</td>
<td>149</td>
<td>171</td>
<td>133</td>
<td>133</td>
<td>133</td>
<td>133</td>
<td>133</td>
<td>137</td>
<td>165</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>65</td>
<td>140</td>
<td>162</td>
<td>129</td>
<td>124</td>
<td>124</td>
<td>124</td>
<td>124</td>
<td>128</td>
<td>156</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>40</td>
<td>115</td>
<td>137</td>
<td>104</td>
<td>99</td>
<td>99</td>
<td>99</td>
<td>99</td>
<td>103</td>
<td>121</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, ZeroBr</td>
<td>95</td>
<td>170</td>
<td>192</td>
<td>154</td>
<td>154</td>
<td>154</td>
<td>154</td>
<td>154</td>
<td>158</td>
<td>186</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, NegBr</td>
<td>97</td>
<td>162</td>
<td>184</td>
<td>151</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>150</td>
<td>178</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, OvBr</td>
<td>90</td>
<td>165</td>
<td>187</td>
<td>154</td>
<td>149</td>
<td>149</td>
<td>149</td>
<td>149</td>
<td>153</td>
<td>181</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, NibCarry</td>
<td>58</td>
<td>133</td>
<td>155</td>
<td>122</td>
<td>117</td>
<td>117</td>
<td>117</td>
<td>117</td>
<td>121</td>
<td>149</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, PpCarryBr</td>
<td>65</td>
<td>140</td>
<td>162</td>
<td>129</td>
<td>124</td>
<td>124</td>
<td>124</td>
<td>124</td>
<td>128</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, PpCrossBr</td>
<td>77</td>
<td>152</td>
<td>174</td>
<td>141</td>
<td>136</td>
<td>136</td>
<td>136</td>
<td>136</td>
<td>140</td>
<td>168</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, CarryBr</td>
<td>60</td>
<td>149</td>
<td>171</td>
<td>145</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>144</td>
<td>147</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, YDisp</td>
<td>68</td>
<td>143</td>
<td>165</td>
<td>132</td>
<td>127</td>
<td>127</td>
<td>127</td>
<td>127</td>
<td>131</td>
<td>171</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, RShift1</td>
<td>89</td>
<td>164</td>
<td>186</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>152</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>60</td>
<td>155</td>
<td>177</td>
<td>144</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>143</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>50</td>
<td>125</td>
<td>147</td>
<td>114</td>
<td>109</td>
<td>109</td>
<td>109</td>
<td>109</td>
<td>113</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B+X+A, R.Rotn</td>
<td>99</td>
<td>174</td>
<td>196</td>
<td>158</td>
<td>158</td>
<td>158</td>
<td>158</td>
<td>158</td>
<td>162</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>90</td>
<td>165</td>
<td>187</td>
<td>154</td>
<td>149</td>
<td>149</td>
<td>149</td>
<td>149</td>
<td>153</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>60</td>
<td>135</td>
<td>157</td>
<td>124</td>
<td>119</td>
<td>119</td>
<td>119</td>
<td>119</td>
<td>123</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>MAR+X+A</td>
<td>78</td>
<td>—</td>
<td>142</td>
<td>137</td>
<td>137</td>
<td>137</td>
<td>137</td>
<td>137</td>
<td>141</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>MDR+X+A</td>
<td>77</td>
<td>152</td>
<td>—</td>
<td>136</td>
<td>136</td>
<td>136</td>
<td>136</td>
<td>136</td>
<td>135</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>70</td>
<td>144</td>
<td>—</td>
<td>134</td>
<td>129</td>
<td>129</td>
<td>129</td>
<td>129</td>
<td>133</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>45</td>
<td>120</td>
<td>—</td>
<td>109</td>
<td>104</td>
<td>104</td>
<td>104</td>
<td>104</td>
<td>108</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>SU+X+A</td>
<td>119</td>
<td>216</td>
<td>178</td>
<td>178</td>
<td>178</td>
<td>178</td>
<td>178</td>
<td>178</td>
<td>182</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>112</td>
<td>—</td>
<td>209</td>
<td>176</td>
<td>171</td>
<td>171</td>
<td>171</td>
<td>171</td>
<td>174</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>87</td>
<td>184</td>
<td>151</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>146</td>
<td>150</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>IOYOut+X+A</td>
<td>96</td>
<td>171</td>
<td>193</td>
<td>155</td>
<td>155</td>
<td>155</td>
<td>155</td>
<td>155</td>
<td>159</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>88</td>
<td>164</td>
<td>186</td>
<td>153</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>148</td>
<td>152</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>64</td>
<td>139</td>
<td>161</td>
<td>128</td>
<td>123</td>
<td>123</td>
<td>123</td>
<td>123</td>
<td>127</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>IOYOut+X+A</td>
<td>80</td>
<td>155</td>
<td>177</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>139</td>
<td>143</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>S374</td>
<td>73</td>
<td>148</td>
<td>170</td>
<td>137</td>
<td>132</td>
<td>132</td>
<td>132</td>
<td>132</td>
<td>136</td>
<td>—</td>
<td>—</td>
<td></td>
<td></td>
</tr>
<tr>
<td>[]+X, XDisp</td>
<td>32</td>
<td>107</td>
<td>129</td>
<td>96</td>
<td>91</td>
<td>91</td>
<td>91</td>
<td>91</td>
<td>94</td>
<td>123</td>
<td>134</td>
<td></td>
<td></td>
</tr>
<tr>
<td>RH+X</td>
<td>36</td>
<td>111</td>
<td>133</td>
<td>100</td>
<td>95</td>
<td>95</td>
<td>95</td>
<td>95</td>
<td>99</td>
<td>127</td>
<td>138</td>
<td></td>
<td></td>
</tr>
<tr>
<td>IB+X</td>
<td>37</td>
<td>112</td>
<td>134</td>
<td>101</td>
<td>96</td>
<td>96</td>
<td>96</td>
<td>96</td>
<td>100</td>
<td>128</td>
<td>139</td>
<td></td>
<td></td>
</tr>
<tr>
<td>IOXOut+X (L374)</td>
<td>22</td>
<td>97</td>
<td>117</td>
<td>84</td>
<td>79</td>
<td>79</td>
<td>79</td>
<td>79</td>
<td>83</td>
<td>111</td>
<td>122</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

* Timing for bits[0-7] of these sources is that of Nibble.  
stackP+ has timing of the slow IOYOut. 
The 3 numbers for arithmetic operations correspond to bits[0-7], bits[8-11], & bits[12-15], respectively.

Figure 7. Allowable X-bus Operations
9 Feb 80
Filed on <Workstation11D>AllowableXbus.sil
<Workstation11D>DMR_press
<table>
<thead>
<tr>
<th>Y Source</th>
<th>setup</th>
<th>A or. B</th>
<th>A (bypass)</th>
<th>A + B</th>
</tr>
</thead>
<tbody>
<tr>
<td>Y +</td>
<td></td>
<td>80</td>
<td>69</td>
<td>109</td>
</tr>
<tr>
<td>MAR +</td>
<td>36</td>
<td>116</td>
<td>105</td>
<td>114</td>
</tr>
<tr>
<td></td>
<td>11</td>
<td>91</td>
<td>80</td>
<td>119</td>
</tr>
<tr>
<td></td>
<td>36</td>
<td>116</td>
<td>105</td>
<td>119</td>
</tr>
<tr>
<td>MDR +</td>
<td>3</td>
<td>83</td>
<td>72</td>
<td>102</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>86</td>
</tr>
<tr>
<td>SU +</td>
<td>45</td>
<td>125</td>
<td>114</td>
<td>154</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>150</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>128</td>
</tr>
<tr>
<td>stackP +</td>
<td>6</td>
<td>86</td>
<td>75</td>
<td>115</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>112</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>89</td>
</tr>
<tr>
<td>[] +, YDisp</td>
<td>32</td>
<td>112</td>
<td>101</td>
<td>121</td>
</tr>
<tr>
<td>uaddr[4-7] +</td>
<td>15</td>
<td>95</td>
<td>84</td>
<td>124</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>120</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>98</td>
</tr>
<tr>
<td>IOYOOut +</td>
<td>6</td>
<td>86</td>
<td>75</td>
<td>115</td>
</tr>
<tr>
<td>(S374)</td>
<td></td>
<td></td>
<td></td>
<td>112</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>89</td>
</tr>
</tbody>
</table>

* Bits[0-7] have timing of Y = (B or. 0), except in the A bypass case.

The 3 numbers for arithmetic operations correspond to bits[0-7], bits[8-11], & bits[12-15], respectively.

Figure 8. Allowable Y-bus Operations
Appendix: Antithetical List of MicroInstructions

This appendix contains a list of some example microinstructions in addition to examples of illegal ones. Those microcode statements which can be written but don't work as intended have three possible reasons for their shortcomings:

1. Timing error,
2. Syntax error, or
3. Characteristic of a processor data path.

SU := STK I U register;
A := the R register addressed by rA;
B := the R register addressed by rB;
R := an R register addressed by either rA or rB;
RotI := LRot I RRot1;
ShiftI := LShift I RShift1;
LRotn := LRot0 I LRot4 I LRot8 I LRot12;
constant := Nibble | Byte;
ArithBr := NegBr | ZeroBr | OvBr | CarryBr | PgCarryBr;
LogicBr := NegBr | ZeroBr;
XDispBr := XpcDisp | XhDisp | XwdDisp | XdDisp | XdDisp;
* := alu logic operation (R or S, R and S, ¬R and S, R xor S, ¬R xor S);
± := alu arithmetic operation (R + S, S - R, R - S);
⊕ := alu arithmetic or logic operation;

General:

Possible:

B + R o R,
B + R ± R,
Q + R ⊕ R,
YBus + A, B + R ⊕ R,

Possible:

B + SU o A,
B + SU o Q,
SU + R o R,
B + SU + R o R,
SU + A, B + R ⊕ R,
SU + A, B + R + 1,
SU + A, B + R - R,
SU + A, B + R ± R, Cin+pc17,

Not Possible:

B + Q + R ⊕ R,
YBus + A, [] + R ⊕ R, (2)
YBus + A, Q + R ⊕ R, (2)

Not Possible:

B + SU ± A, (1)
SU + R ± R, (1)
SU + A, B + R + R, Cin+pc17,
SU + A, B + R + R + 1, (2)
SU + A, B + R - 1, (2)
SU + A, B + SU, (2) & (3)
SU + A, SU ⊕ A, (2) & (3)
Constants:

Possible:

\[
\begin{align*}
B & = \text{constant}, \\
B & = A \oplus \text{constant}, \\
B & = 0, \\
\{B + 100,\} & = B + \text{OFF} \times 1 \\
\{B + \text{OFF}00,\} & = B + \text{OFF} \\
\{B + \text{OFF}FF,\} & = B + \text{OFF} \times B \\
\{B + \text{7FF},\} & = B + \text{RShift} \times (\text{B xor B}) \\
\{B + \text{1FF},\} & = B + \text{LSHIFT} \times \text{OFF, SE+1} \\
SU & = 0 \\
\{SU + \text{OFF}FF,\} & = SU \times \text{~A xor A}
\end{align*}
\]

Not Possible:

\[
\begin{align*}
\text{STK} & = \text{constant}, \\
U & = \text{constant}, \\
B & = SU \oplus \text{constant},
\end{align*}
\]

(1)

(2)

(2) & (3)

RH registers:

Possible:

\[
\begin{align*}
B & = \text{RH}[B] \times A, \\
B & = \text{RH}[B] \pm A, \\
\text{RH}[B] & = SU, \\
\text{RH}[B] & = SU, B = SU \times A, \\
\text{RH}[B] & = SU, B = R \oplus R, \\
\text{RH}[B] & = \text{constant}, \\
\text{RH}[B] & = \text{constant}, B = R \oplus R, \\
\text{RH}[B] & = b, \\
\text{RH}[B] & = b, B = R \oplus R, \\
\text{STK} & = A, B = \text{RH}[B] \oplus A,
\end{align*}
\]

Not Possible:

\[
\begin{align*}
A & = \text{RH}[B], \\
\text{RH}[B] & = \text{RH}[B]
\end{align*}
\]

(2)

Memory:

Possible:

\[
\begin{align*}
R & = \text{MD}, \\
R & = R \times \text{MD}, \\
Q & = R \times \text{MD}, \\
\text{RH}[B] & = \text{MD}, \\
\text{RH}[B] & = \text{MD, B \times R \oplus R}, \\
\text{RH}[B] & = \text{MD, B \times MD, SU \times A,} \\
\text{MAR} & = [\text{rh}[B], A], B = B \oplus R, \\
\text{MAR} & = [\text{rh}[B], A], B = \text{IOin}, \\
\text{MAR} & = [\text{rh}[B], A], B = \text{RH}[B], \\
\text{MAR} & = [\text{rh}[B], A], B = \text{SU}, \\
\text{MAR} & = [\text{rh}[B], A], B = \text{constant}, \\
\text{MAR} & = [\text{rh}[B], B \oplus R], \\
\text{MAR} & = B = [\text{rh}[B], B \oplus R], \\
\text{Map} & = Q = [\text{rh}[B], R \oplus R], \\
\text{MAR} & = [\text{rh}[B]], \text{constant}, \\
\text{MDR} & = R \oplus R, \\
\text{MDR} & = A, B = R \oplus R, \\
\text{MDR} & = SU \times A, \\
\text{MDR} & = \text{RH}[B],
\end{align*}
\]

Not Possible:

\[
\begin{align*}
R & = R \pm \text{MD}, \\
R & = \text{MD Shift}, \\
SU & = \text{MD},
\end{align*}
\]

(1)

(1)

MAR = [rh[B], SU], (1)

MDR = SU \oplus A, \{OK in bits[12-15]\} (1)
IOIn/IOOut:

Possible:

B ← IOIn ⊕ A,
RH[B] ← IOIn,
MDR ← IOIn,
RH[B] ← IOIn, B ← IOIn ⊕ A,

IOOut ← (R ⊕ P) LRotn,
IOOut ← Q LRotn,
IOOut ← MDn,
IOOut ← IOIn,
IOOut ← SU,
IOOut ← A LRotn,
IOOut ← 1b,

stackP:

Possible:

stackP ← R ⊕ R,
stackP ← stackP ⊕ A,
stackP ← Nibble,
stackP ← IOIn,
stackP ← RH[B],
RH[B] ← stackP,

LRotn:

Possible:

B ← A LRotn, [A bypass]
[] ← (A ⊕ B) LRotn,
[] ← Q LRotn,
B ← A ⊕ (A LRotn),

RH[B] ← A LRotn, B ← R ⊕ R,
RH[B] ← (R ⊕ P) LRotn,

STK ← A, RH[B] ← A LRotn, B ← R + R + 1,
STK ← A, RH[B] ← (R ⊕ P) LRotn,

Not Possible:

SU ← (IOIn ⊕ A) LRotn,
IOOut ← (R ± R) LRotn,

stackP ← stackP + constant,
stackP ← MDn,

Not Possible:

B ← (R ⊕ R) LRotn,
Q ← A LRotn,
B ← Q LRotn,
B ← B ⊕ (A LRotn),
SU ← (R ⊕ P) LRotn,
SU ← A LRotn,
MDR ← A LRotn,
B ← SU LRotn,
B ← (A LRotn) Rot1,
B ← (R ⊕ R LRotn) Shift1,
RH[B] ← A LRotn,
RH[B] ← (R ± R) LRotn, [OK in [12-15]]
RH[B] ← (RH[B] ⊕ A) LRotn,
U ← A, RH[B] ← (R ⊕ P) LRotn,
Single Bit Shifting:

Possible:

\[ B \oplus (R \circ R) \text{Shift}1, \]
\[ B \oplus (R \circ R) \text{Rot}1, \]
\[ B \oplus (R \circ R) \text{Shift}1, SE=0, \]
\[ B \oplus (R - R-1) \text{Shift}1, SE=0, \]
\[ B \oplus (SU \circ A) \text{Shift}1, SE=0, \]
\[ B \oplus (\text{constant} \pm A) \text{Shift}1, \{\text{only [8-15]}\} \]
\[ B \oplus R \pm R, \text{DRShift}1, \]
\[ B \oplus R \pm R, \text{DLShift}1, \]

Not Possible:

\[ Q \oplus (R \circ R) \text{Shift}1, \]
\[ Q \oplus (R \circ R) \text{Rot}1, \]
\[ B \oplus (R \pm R) \text{Shift}1, \]
\[ B \oplus (R \pm R) \text{Rot}1, \]
\[ Y\text{Bus} \oplus A, B \oplus R \text{Shift}1, \]
\[ \text{[]} \oplus R \text{Shift}1, \]
\[ B \oplus (R + R + 1) \text{Shift}1, SE=0 \]
\[ B \oplus (R - R) \text{Shift}1, SE=0 \]
\[ B \oplus (SU \circ A) \text{Shift}1, SE=1, \]
\[ SU \oplus B \text{Shift}1, \]
\[ SU \oplus A, B \oplus R \text{Shift}, \]
\[ B \oplus (\text{constant} \pm A) \text{Rot}1 \{\text{OK in [12-15]}\} \]

Branching:

Possible:

\[ B \oplus R \oplus R, \text{Branch}, \]
\[ Y\text{Bus} \oplus A, B \oplus R \oplus R, \text{Branch}, \]
\[ B \oplus X\text{bus} \circ A, \text{LogicBr}, \]
\[ B \oplus X\text{bus} \pm A, \text{CarryBr}, \{\text{except for SU, MD}\} \]
\[ B \oplus X\text{bus} \pm A, \text{PcCarryBr}, \{\text{except for SU, MD}\} \]
\[ B \oplus R \pm 1, \text{ArithBr}, \]
\[ B \oplus R \oplus R, \text{YDisp}, \]
\[ B \oplus \text{RH}[B] \circ A, V\text{Disp}, \]
\[ Y\text{Bus} \oplus A, B \oplus R \oplus R, \text{YDisp}, \]
\[ \text{[]} \oplus \text{SU}, \text{XDispBr}, \]
\[ \text{[]} \oplus \text{IOin}, \text{XDispBr}, \]
\[ \text{[]} \oplus \text{constant}, \text{XDispBr}, \]
\[ \text{[]} \oplus \text{lb}, \text{XDispBr}, \]
\[ \text{[]} \oplus \text{RH}[B], \text{XDispBr}, \]
\[ \text{[]} \oplus \text{stackP}, \text{XDispBr}, \]
\[ \text{[]} \oplus \text{MD}, \text{XDispBr}, \]
\[ B \oplus A \text{LRotn}, \text{XDispBr}, \]
\[ B \oplus A \circ (A \text{LRotn}), \text{XDispBr}, \]
\[ \text{[]} \oplus (R \circ R) \text{LRotn}, \text{XDispBr}, \]

Not Possible:

\[ B \oplus MD \circ A, \text{LogicBr}, \]
\[ B \oplus X\text{bus} \pm A, \text{ZeroBr}, \]
\[ B \oplus X\text{bus} \pm A, \text{NegBr}, \]
\[ B \oplus SU \circ A, \text{YDisp}, \]
\[ B \oplus MD \circ A, \text{YDisp}, \]
\[ \text{[]} \oplus (R \pm R) \text{LRotn}, \text{XDispBr}, \{\text{bits[12-15] OK}\} (1) \]