%
Page Numbers: Yes First Page: 1
Heading:
TriconD.mcMarch 8, 1979
title[TriconD.mc - July 11, 1984]; ** diagnostics for the Dorado disk controller
TESTDESCRIPTION
The first part of these tests will run with no disk present
PreBeginChecks basic IO and Muffler input, and halts on an intentional parity err
Beginstart of main diagnostics
CheckStatechecks that all known muffler bits are in their correct state
TagRegChecks loading of the TAG register - static mode
RamTests the Format Ram - both addressing and contents
FifoTests writing and reading of the Fifo - also test write wake-up logic
FifoStepAllows to to step (in Pasive mode) the controller through Fifo operation
IsDiskSpinningLooks and the disk status and branches to done if not ready and on-line
The rest of the tests require a disk to be present and running
TagWrite Tag register, block, and wait for appropriate wake-up
WakeUpschecks basic sector and index wake-ups
SectorCountersLoads Sub-sector counter and checks for the right number of sectors
NOTE: In all testing, the bits that were found to be in orror are left in register "Errors"
As an example if Errors containd 200 and the test is for controller state KSTATE, then the signal on the muffler KSTATE "bit 200" (ie EnableRun) would have an unexpected value.
%
***** - - - - - - - - - - - - - - - - - - - - emulator code for CheckTAG test
*if the disk task is ever blocked, then this loop will run, and if EMUCount goes to zero then the DSK task will be woken up via Notify. It is up to the disk micro-codeto notice that there was no actual hardware wake-up.
set[xtask, 0]; top level; KnowRBase[Errors];
set[EMUlocx, 201]; mc[EMUloc, EMUlocx];*EMU task PC counter
EMULoop:
RBase ← RBase[EMUcount];
Pd← EMUcount, at[EMUlocx];**Test for already zero
SkpIf[alu#0];
Branch[EMUloop];
EMUcount ← (EMUcount)-1;**EMU count down loop
SkpIf[alu#0];
Wakeup[DSK];*Notify if just gone to Zero
Branch[EMULoop];
set[xtask, IP[DSK]];
***** - - - - - - - - - - - - - - - - - - - -
** check out the muffler reading scheme and compare the results
*** read by the microcode with what Midas reads
RestartDiagnostic:
T←DiskControl; *At this point set TIOA with Bmux to disk TIOA
TIOA←T;
T ← noRunEnable;
Out: Output ← T;
RBase←RBase[Errors];
Noop;
T←10C;
TIOA[DiskMuff];
Output ← T;
T ← TIOA&STKP;*Lets check that TIOA got set correctly
T ← T and (177400C);*mask TIOA
Errors ← T # (DiskMuff);
SkpIf[alu=0];
TIOAErr0:
Error;*Value in TIOA not equal to DiskMuff (011C)
InNoPck: Errors ← InputNoPe;
Noop;
Noop;
Noop;
InPck: Pd ← Input;
Noop;
Noop;
Noop;
CheckMuffs:
call[read20Muffs], T← StateMuffs;
Dmux100← T;
call[read20Muffs], T← StatusMuffs;
Dmux101← T;
call[read20Muffs], T← RamMuffs;
Dmux102← T;
call[read20Muffs], T← TagMuffs;
Dmux103← T;
call[read20Muffs], T← FifoMuffs;
Dmux104← T;
T←177400C;*At this point set TIOA with Bmux 377
TIOA←T;
PeCk: Errors ← Input;*There is no input for this TIOA address.
Noop;*This should therefore cause an IOin pari err
Good.PE.Hlt: Branch[Begin];*Compair Disk Mufflers against Midas Mufflers
PCKFailed:
Error;*Processor Input Parity test failed!!!
*No divice was selected to drive IOB
***** - - - - - - - - - - - - - - - - - - - -
** Check all signals on mufflers and see that they are reasonable at this point
** This means masking off all signals that we can’t be sure of.
Begin:
Noop;
Check.State:
call[clear-disk];
call[read20Muffs], T← StateMuffs;
T ← T # (70C);*Normalise the bits read
Errors←T AND (76377C);*mask of just the bits that are determinate
SkpIf[alu=0];
State.Errs:
Error;*Compire "1" bit(s) with muffler word KSTATE
call[read20Muffs], T← StatusMuffs;
T←T # (16000C);*Normalise the bits read
T←T # (3C);*Normalise the bits read
Errors←T and (177737C);*mask of just the bits that are determinate
SkpIf[alu=0];
Stat.Errs:
Error;*Compire "1" bit(s) with muffler word KSTAT
call[read20Muffs], T← FifoMuffs;
Errors ← T # (2000C);*Normalise the bits read
SkpIf[alu=0];
Fifo.Errs:
Error;*Compire "1" bit(s) with muffler word KFIFO
***** - - - - - - - - - - - - - - - - - - - -
** write the TAG register with various data.
** I must avoid values that make the disk to barf
** read value via the mufflers and compare
** the value written into the TAG register is immediately
** available for reading on the mufflers
** run as DiskTASK only - no blocks/wakeups
TagReg:
call[clear-disk];
Count← (100000C);*initialize counter and TAG value
TagRegL:
Noop;
TIOA[DiskTag];
Output ← Count;
call[read20Muffs], T← TAGmuffs;
T← T # (Count);
SkpIf[alu=0];*skip if ok
TagRegErr1:
Error;*Tag Register not equal to data written
Pd ← (Count) and (160000C);
SkpIf[alu=0];
Count ← (Count) rsh 1, Skip[];
Count← (Count) - 1;
LoopUntil[alu=0, TagRegL];*skip if around to zero again
***** - - - - - - - - - - - - - - - - - - - -
*Issue DriveSelect tag instructions to select drives values of 0 to 17
*and see that the controller selects the appropriate drives
* ( DrSel is a subroutine that isues a drive select from value in "T"
* and returns with select.0,,select.1 in bits 14,,15 of T)
DriveSelect:
call[clear-disk];*initialize controller & clear any wakeups
call[init-ram];*turn on RunEnable
call[read1Muff], T← RunEnableMuff;*Now check to see of RunEnable is on
SkpIf[alu#0];*skip if RunEnable set
DriveSelE:
Error;*RunEnable should be set at this point
Noop;
call[Set-CKDrive], Count← A0;*Select drive 0
SkpIf[alu=0];
DriveSelE0:
Error;***didn’t select 0
call[Set-CKDrive], Count← (Count)+1;*Select drive 1
Pd← (T) # (Count);
SkpIf[alu=0],Errors←T;
DriveSelE1:
Error;***didn’t select 1
call[Set-CKDrive], Count← (Count)+1;*Select drive 2
Pd← T # (Count);
SkpIf[alu=0],Errors←T;
DriveSelE2:
Error;***didn’t select 2
Noop;
ds3:
Noop;
call[Set-CKDrive], Count← (Count)+1;*Select drive 3 through 16b
Pd← T # (3C);
SkpIf[alu=0],Errors←T;
DriveSelE3:
Error;***didn’t select 3
Pd← (Count) # (17C);*Do this until we get to drive 16
LoopUntil[alu=0, ds3];
Noop;
call[Set-CKDrive], Count← (Count)+1;*Select drive 17b
SkpIf[alu=0],Errors←T;
DriveSelE17:
Error;***didn’t select 0
***** - - - - - - - - - - - - - - - - - - - -
** check that the format ram is being written correctly by writing a random pattern
*** write the RAM once; then check those values and write in new ones
Ram:
call[clear-disk];*initialize controller & clear any wakeups
Call[SendCommand],T←A0;*make sure write addr is 0 for format RAM
KScr← 123C;*start reading with some value
KScr2← 123C;*start writing with some value
Count← 20C;*counter
RamInitL:*Write the Ram once to get known data in it
Noop;
Call[read1Muff], T← RunEnableMuff;*Check that RunEnable has not been set yet
SkpIf[alu=0];
RamWErr0:
Error;*RunEnable should still be cleared at this point
*The format ram has been writen "Count" times
TIOA[DiskRam];
Call[MakeRandom],T← KScr2;*Get a new random number for next write
KScr2← Output← T;*Write new number
call[read1Muff], T← IOBPErrMuff;*Check that Output didn’t cause a parity error
SkpIf[alu=0];
RamPerr0:
Error;*Controller detected IOB parity error
*due to the previous write
Count← (Count) - 1;
LoopUntil[alu=0, RamInitL];
call[read1Muff], T← RunEnableMuff;
SkpIf[alu#0];
RamWErr1:
Error;*runenable should be set at this point
Noop;
RamLoop:
Noop;
Call[MakeRandom],T←KScr;*Update random number for next read
KScr← T;*Make up what were are really going to read
KScr3← T and (7777C);*The ram contents in bits 4,,15
T ← LSH[Count,14];
KScr3← (KScr3) or T;*and the ram address in bits 0,,3 of Count
call[read20Muffs], T← RamMuffs;*Now go and read the muffler
Errors← (KScr3) # T;
SkpIf[alu=0];
RamErr:
Error;*bad value from Ram (KScr3 = good value)
*bits 0,,3 are address; 4,,15 are RAM data
TIOA[DiskRam];*Output next value and
Call[MakeRandom],T← KScr2;*Get a new random number for next write
KScr2← Output← T;*Write new number
call[read1Muff], T← IOBPErrMuff;*Check that Output didn’t cause a parity error
SkpIf[alu=0];
RamPerr1:
Error;*Controller detected IOB parity error
*due to the previous write
Count← (Count) + 1;
LoopUntil[alu=0, RamLoop];
***** - - - - - - - - - - - - - - - - - - - -
*First initialize the controller into an active Write command and check the mufflers
*If the controller doesn’t run this test right, you might try the StepFifo routine
FiFo:
Noop;
KScr← 123C;*Initialize read value
KScr2← 123C;*Initialize write value
call[clear-disk];
TIOA[DiskTag], T ← T-T;*Make sure Tag register is 0
Output← T;
call[init-ram];
Call[SetDrive], T←37C;*Deselect drive so command is not executed
T← 100C;*write header command
call[SendCommand],T← T + (1000C);*set Debug mode so controller becomes Active
call[read20Muffs], T← StateMuffs;
T ← T # (354C);*Normalise the bits read
Errors←T and (374C);*mask of just the bits that are determinate
SkpIf[alu=0];
FifoErr0:
Error;*Controller in wrong state after Write command
*Compire "1" bit(s) with muffler KSTATE
FifoOut:
Noop;
call[read20Muffs], T← FifoMuffs;*First see that R/W address pointers are zero
Errors←T and (377C);
SkpIf[alu=0];
FifoOutE0:*The FiFo address counters are wrong
Error;*Compair should be zero
Call[MakeRandom],T← KScr2;*Get a new random number for next write
TIOA[DiskData];
KScr2← Output← T;*Write new number [ KFIFO = 6000 ]
call[read1Muff], T← IOBPErrMuff;*Check that Output didn’t cause a parity error
SkpIf[alu=0];
FifoOutE1:
Error;*Controller detected IOB parity error
*due to the previous write
call[read20Muffs], T← FifoMuffs;*Check R/W address pointers again
T←T and (5777C);*CntDone’ is not determinant
T←T # (5000C);
Errors←T # (21C);
SkpIf[alu=0];
FifoOutE2:*The FiFo state is wrong KFIFO should be 7021
Error;*The FiFo address counters should be 21
*OutRegFull should be 1 (2000 bit)
KScr3← 21C;*Initialize Read/write address pointer
*****Now start the main loop
FifoLoop:
Count ← 16C;
*****First do 14 writes checking parity,fifo address pointers, and wake-up
FifoWLoop:
Noop;
Call[MakeRandom],T← KScr2;*Get a new random number for next write
TIOA[DiskData];
KScr2← Output← T;*Output and save the new number
call[read1Muff], T← IOBPErrMuff;*Check that Output didn’t cause a parity error
SkpIf[alu=0];
FifoWriteE0:
Error;*Controller detected IOB parity error
*due to the previous write
KScr3 ← (KScr3) + (20C);*now update the R/W address register
KScr3 ← (KScr3) and (377C);
call[read20Muffs], T← FifoMuffs;*Now see that R/W address counters are correct
T←T and (377C);
Errors ← T # (KScr3);
SkpIf[alu=0];
FifoWriteE1:*The FiFo address counters are wrong
Error;*Compair KFIO (bits 8,,15) with KScr3
T ← RSH[KScr3,4];*Now compute the difference between
Errors ← T;*words in Fifo = (Waddr-Raddr)&17b
T ← (KScr3) and (17C);
T ← (Errors) - T;
Errors ← T and (17C);
Call[Read1Muff], T ← WriteTWmuff;*Read state of Write Task Wakeup
Pd ← (Errors) - (15C);*See if the value should be on or off
SkpUnless[alu<0], Errors ← T;*save read value in Errors
T← (T) # (1C);*normalize value
SkpIf[alu=0];*and skip if is what it should be
Branch[FifoErrCheck];
Branch[FifoWritecont];
FifoErrCheck:
T← (Count) # (3C);
SkpIf[alu=0];
FifoWriteE2:*Write wakeup Error
Error;*Value read is in Errors
FifoWritecont:
Count← (Count) - 1;
LoopUntil[alu=0, FifoWLoop];
Noop;
Count ← 16C;
*****Now do 14 reads checking data and fifo address pointers
FifoRLoop:
Noop;
Call[MakeRandom],T←KScr;*Update random number for reading
KScr← T;
TIOA[DiskData];
T← Input;
Errors← (KScr) # T;
SkpIf[alu=0];
FifoReadE0:
Error;*bad value from Fifo (KScr = good value)
T ← (KScr3) and (17C);*update the Read portion of R/W addr register
Pd ← T # (17C);
SkpUnless[alu=0], KScr3 ← (KScr3) + 1;
KScr3 ← (KScr3) - (20C);
Noop;
call[read20Muffs], T← FifoMuffs;*Now see that R/W address counters are correct
T←T and (377C);
Errors ← T # (KScr3);
SkpIf[alu=0];
FifoReadE1:
Error;*Compire "1" bit(s) with muffler word KFIFO
Pd← (KScr) # (123C);
SkpUnless[alu=0];*exit test if we get our original random number
Branch[FifoDone];
Count← (Count) - 1;
LoopUntil[alu=0,FifoRLoop];*Keep reading some words
Branch[FifoLoop];*Go back and write some more words
FifoDone:
Call[clear-disk];*Clear out Debug Mode
Call[clear-disk];*Turn off Active
Branch[IsDiskSpinning];
***** - - - - - - - - - - - - - - - - - - - -
**Set the controller up so that the midas STEP command can be used to observe
**the state of the controller as a word is loaded into the controller and progresses
**to the output register
FifoStep:
Noop;
call[clear-disk];
TIOA[DiskTag], T ← T-T;
Output← T;
call[init-ram];*Turn on RunEnable so Fifo can be loaded
*[ KSTATE = 354 ]
*Fifo is initialized and empty[ KFIFO = 2000 ]
T ← 1C;
Set-Pasive:*Midas must be in Pasiv mode for this test
TIOA[DiskData], Breakpoint;*TIOA = 12
Noop;*InRegister is loaded [ KFIFO = 2000 ]
S-out:Output← T;*Write data (see ALUA) [ KFIFO = 2000 ]
S-IR:Noop;*InRegister is loaded [ KFIFO = 2400 ]
S-FiFo:Noop;*Fifo ← InRegister[ KFIFO = 2020 ]
Noop;*FifoEmpty←0 (no muff)[ KFIFO = 2020 ]
S-OR:Noop;*OutRegister ← Fifo[ KFIFO = 3021 ]
S-In:KScr←Input;*IOB←OutRegister[ KFIFO = 3021 ]
S-Ck:B ← KScr, Breakpoint;*Read data (see ALUA) [ KFIFO = 2021 ]
T ← T lcy 1, Branch[S-out];*Change data and do again
%
On sucessive passes through this loop, KFIFO is incremented by 21 as the two 4 bit address counters increment modulo 16. The date written and read is 1 left shifted 1 for each pass, and is on the "BMux" so you can see in in passive mode on the ALUA muffler.
The valid values for instruction "S-Ck" are:
KFIFO=3021 3042 3063 3104 3125 3146 3167 3210 3231 3252 3273 3314 3335 3356 3377 3000
ALUA=0001 0002 0004 0010 0020 0040 0100 0200 0400 1000 2000 4000 10000 20000 40000 100000
%
***** - - - - - - - - - - - - - - - - - - - -
**Now check things that require a disk to be up and spinning
IsDiskSpinning:
Call[clear-disk];*Clear out previous states
call[init-ram];*turn on RunEnable
call[read20Muffs], T← StatusMuffs;
T ← T and (16000C);
SkpIf[alu=0];
Branch[Done];*Done if the disk is not up and ready
***** - - - - - - - - - - - - - - - - - - - -
** write the TAG register with incrementing data, thru all values
*** read value via the mufflers and compare
** start as DiskTASK then block and wait for Hardware Wakeup
**The emulator has a timeout loop which will generate a Wakeup to be detected as an error
** this checks out clearTWs, block, and TAG wakeups
Tag:
call[clear-disk];
CountS←5C;*read the 5 TWmuffs
call[ReadMuffs], T← IndexTWmuff;*address of 1st muffler to read
SkpIf[alu=0];*skip if wakeups cleared
TagE0:
Error;*wakeups not cleared
call[init-ram];*turn on RunEnable
call[read1Muff], T← RunEnableMuff;*Now check to see of RunEnable is on
SkpIf[alu#0];*skip if RunEnable set
TagE1:
Error;*RunEnable should be set at this point
Count← (1C);*shift 1 bit - primarily tests wake-ups
TagLoop:
Noop;
Pd ← (Count) and (70000C);*See if the new number should enable Wake-Up
Branch[NoWkUp,alu=0];
Noop;*Micro placement
call[SendTag], T← Count;*returns via Wakeup
SkpIf[alu#0],Errors←T;*Test to see that return was not due to time out
TagWkUpE0:
Error;*Was woken up by EMU timeout
Branch[TagCk];
NoWkUp:
call[SendTag], T← Count;*returns via time-out
SkpIf[alu=0],Errors←T;*Test to see that return was not due to time out
TagWkUpE1:
Error;*Was woken up by Hardware
TagCk:
T← TAGmuffs;*Micro placement
call[read20Muffs];*Now see if the data is correct
Errors← T # (Count);
SkpIf[alu=0];*skip if ok
TagE2:
Error;*Wrote "Count" into TAG register
Count← (Count) lsh 1;
LoopUntil[alu=0, TagLoop];*test for done
********** - - - - - - - - - - - - - - - - - - - -
**Try the basic Sector and Index wake-up logic
WakeUps:
call[clear-disk];
call[init-ram];*turn on RunEnable
EMUcount←A1;
WakeIdx:
Noop;
Call[DoClear], T← clearSTW;
Block;*block for wake-up -- timeout @ 17ms
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (SectorTWBit);
LoopUntil[alu=0,WakeIdx], Errors←T;*Loop untill not a Sector wake-up
Pd ← T # (IndexTWBit);
SkpIf[alu=0];
WakeIdxErr:
Error;*No INDEX wake-ups???
EMUcount ← A1;
WakeSec:
Noop;
Call[DoClear], T← clearITW;
Block;*block for wake-up -- timeout after 17ms
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
LoopUntil[alu=0,WakeSec], Errors←T;*Loop untill not an Index wake-up
Noop;
Pd ← T # (SectorTWBit);
SkpIf[alu=0];
WakeSecErr:
Error;*No SECTOR wake-ups???
********** - - - - - - - - - - - - - - - - - - - -
**Now put all reasonable values of SubSector counts and check for the right number of Sectors
SectorCounters:
call[clear-disk];
call[init-ram];*turn on RunEnable
Sectors117:
Call[SetSector], T ← A0;
Call[DoClear], T← clearATWs;
Call[SendCommand],T←ReadyAndIndex;
EMUcount ← A1,Block;*block for wake-up -- timeout after 17ms
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
SkpIf[alu#0], Errors←T;
WakeSecErr117:
Error;*Wake-Up was NOT an Index
*WaitRillIndex command probably failed
Errors ← A0;
117Loop:
Call[DoClear], T← clearATWs;
EMUcount ← (1000C),Block;*block for wake-up -- timeout after 150us
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
LoopUntil[alu#0,117Loop], Errors ← (Errors)+1;
Pd ← (Errors) # (117D);
SkpIf[alu=0];
SectorErr117:*With a value of 0 in the Sub-Sector counters,
Error;*there should have been a count of 117d sectors.
Sectors30:
Call[SetSector], T ← AltoCount;
Call[DoClear], T← clearATWs;
Call[SendCommand],T←ReadyAndIndex;
EMUcount ← A1,Block;*block for wake-up -- timeout after 17ms
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
SkpIf[alu#0], Errors←T;
WakeIdxErr30:
Error;*Wake-Up was NOT an Index
*WaitRillIndex command probably failed
Errors ← A0;
30Loop:
Call[DoClear], T← clearATWs;
EMUcount ← (4000C),Block;*block for wake-up -- timeout after 600us
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
LoopUntil[alu#0,30Loop], Errors ← (Errors)+1;
Pd ← (Errors) # (30D);
SkpIf[alu=0];
SectorErr30:*With a value of 3 in the Sub-Sector counters,
Error;*there should have been a count of 30d sectors.
Sectors9:
Call[SetSector], T ← TriCount;
Call[DoClear], T← clearATWs;
Call[SendCommand],T←ReadyAndIndex;
EMUcount ← A1,Block;*block for wake-up -- timeout after 17ms
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
SkpIf[alu#0], Errors←T;
WakeIdxErr9:
Error;*Wake-Up was NOT an Index
*WaitRillIndex command probably failed
Errors ← A0;
9Loop:
Call[DoClear], T← clearATWs;
EMUcount ← 20000C,Block;*block for wake-up -- timeout after 2000us
CountS←5C;
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
Pd ← T and (IndexTWBit);
LoopUntil[alu#0,9Loop], Errors ← (Errors)+1;
Noop;
Pd ← (Errors) # (9D);
SkpIf[alu=0];
SectorErr9:*With a value of 0 in the Sub-Sector counters,
Error;*there should have been a count of 9d sectors.
********** - - - - - - - - - - - - - - - - - - - -
call[clear-disk];
branch[Done];*Done with diagnostics
********** - - - - - - - - - - - - - - - - - - - -
subroutine;*****
********** - - - - - - - - - - - - - - - - - - - -
read1Muff:*muffler number is in T
TIOA[DiskMuff], Global;
Output← T;
noop;
T← Input;
Pd← T, Return;*returns with alu test hanging
********** - - - - - - - - - - - - - - - - - - - -
read20Muffs:*muffler number is in T
CountS← 20C, Global;*Set count to 20
Branch[foo], KScrS← A0;*clear out starting value
********** - - - - - - - - - - - - - - - - - - - -
ReadMuffs:*T plus count in CountS
KScrS← A0,Global;*clear out starting value
foo:TIOA[DiskMuff],KScr4←T;
loopRdMf:
Output← KScr4, KScr4← (KScr4) + 1;
CountS← (CountS) - 1;
branch[.+3, alu<0], KScrS← (KScrS) lcy 1;
T← input;
KScrS← (KScrS) OR T, branch[loopRdMf];
KScrS← T← (KScrS) rcy 1, return;*values are returned in KScrS and T
********** - - - - - - - - - - - - - - - - - - - -
DoClear:*bits to be cleared are in T
*Noop,Global;
TIOA[DiskMuff],Global;
Output← T; Noop; Noop;*Noops required for wake-ups to clear
return;
********** - - - - - - - - - - - - - - - - - - - -
*** returns when a seek/tagTW is actually seen - or when the EMU task times out
***An actual hardare wakeup will cause the subroutine to return -1
***An EMU timeout will return 0
SendTag:*TAG value is in T
KScrL← Link;
top level;*****
TIOA[DiskTag];
Output← T;
call[read1Muff], T← IOBPErrMuff;*Check that Output didn’t cause a parity error
SkpIf[alu=0];
OutPerr:
Link←KScrL,branch[.], breakpoint;*Controller detected IOB parity error
*due to the previous write
Noop;
SndTagL:
Call[DoClear], T← clearALLbutTagTW;
EMUcount ← 40C,Block;*block for wake-up --3us maximum
CountS←5C;*block for wake-up
Call[ReadMuffs], T← IndexTWmuff;*See what woke up the task
SkpUnless[alu=0], KScrS←T;*Loop back and wait for another wakeup
Branch[TagTWseen];*Loop back and wait for another wakeup
Pd ← T and (4C);
LoopWhile[alu=0, SndTagL];*Loop back and wait for another wakeup
Noop;
TagTWseen:
Call[DoClear], T← clearALL;
subroutine;*****
Link← KScrL;
EMUcount ← A0;*stop the Emulator task from counting
T ← KScrS, Return;
********** - - - - - - - - - - - - - - - - - - - -
SetDrive:
TIOA[DiskTag],Global;
Output ← T;*First set the value in the 12 bit register
T ← T OR (DrvSelectTAG);*drive select plus DriveTag bit
Output← T;*Now turn on the strobe
T ← T and (7777C);*drive select only
Output← T, Return;*Finally turn off the strobe
********** - - - - - - - - - - - - - - - - - - - -
SetSector:
KScrL ← Link,Global;
top level;*****
Call[SetDrive];
T ← T or (SectorCntEnable);
Call[SetDrive];
subroutine;*****
Link← KScrL;
Return;
********** - - - - - - - - - - - - - - - - - - - -
SendCommand:*command is in T
TIOA[DiskControl],Global;
Output← T, return;
********** - - - - - - - - - - - - - - - - - - - -
clear-disk:
RBase←RBase[Errors],Global;*Not needed
KScrL ← Link;
Top Level;****
T← EMUloc;*set the EMU PC to its loop
Link← T;
LdTPC← EMU;
EMUcount←A0;*Clear EMU counter
T←DiskControl;*Set TIOA with Bmux to a disk TIOA
TIOA←T;
T←NoRunEnable;*Turn off RunEnable
subroutine;*****
Link← KScrL;
Output← T, Return;
********** - - - - - - - - - - - - - - - - - - - -
init-ram:*init format Ram in Diablo format
TIOA[DiskRam];
T← 1C;*header count - 1
Output← T;
T← 7C;*label count - 1
Output← T;
T← 377C;*data count - 1
Output← T, T← T - T;
Output← T;*unused block = 0
T← 104C;*look for D1Trident sync pattern/read command
Output← T;
T← 204C;*write command
Output← T;
T← 4C;*init command
Output← T, T← T - T;
Output← T;*reset command is 0
T← 33C;*write delay first block
Output← T;
T← 6C;*write delay successive blocks
Output← T;
T← 11C;*read delay first block
Output← T;
T← 2C;*read delay successive blocks
Output← T;
Output← T, T← T - 1;*head select delay is also 2
Output← T, T← T - 1;*no. of ECC words - 1 = 1
Output← T;*the constant 1 - 1 = 0
Output← T;*must write last word to turn on RunEnable
TIOA[DiskControl];*no wakeups ’til Ready&Index
T← ReadyAndIndex;
Output← T;
TIOA[DiskMuff];
T← clearALL;*clearErrors + clearTWs and returns
Output← T, Return;
********** - - - - - - - - - - - - - - - - - - - -
**And now my own favorite random number generator
**You give it some number, and it returns a new number
MakeRandom:*Old value is in T
Pd ← T,Global;
Branch[.+3,alu<0], T ← T LSH 1;*Skip to return
T ← T # (77000C);
T ← T # (213C);*XOR by value 77213
Return;*and Return
Set-CKDrive:
KScrL ← Link,Global;
top level;*****
Call[SetDrive] T←Count;
Call[read1Muff], T←DiskSelectMuff;*muffler number is in T
KScrS←T lsh 1;*Read the two DriveSelect mufflers
Call[read1Muff], T←DiskSelectMuff+1;*muffler number is in T
subroutine;
Link← KScrL;*****
T← (KScrS) or T, Return;
top level;*****