; Copyright 1996 Acorn Computers Ltd ; ; Licensed under the Apache License, Version 2.0 (the "License"); ; you may not use this file except in compliance with the License. ; You may obtain a copy of the License at ; ; http://www.apache.org/licenses/LICENSE-2.0 ; ; Unless required by applicable law or agreed to in writing, software ; distributed under the License is distributed on an "AS IS" BASIS, ; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ; See the License for the specific language governing permissions and ; limitations under the License. ; ; > $.Source.PMF.osinit ; ***************************************************************************** ExecuteInit ROUT Push R14 ; Point to OsbyteVars ; and initialise them BYTEWS WsPtr LDRB R1, LastBREAK ; 0 => soft, 1 => power-on, 2 => hard CMP R1, #1 ADRCC R2, SoftResetVars ADREQ R2, PowerOnResetVars ADRHI R2, HardResetVars LDRCCB R3, NoIgnore ; preserve NoIgnore over soft reset MOVCS R3, #0 ; if hard or power-on reset, zero it STRCS R3, TimerAlpha +0 ; and zero both copies of TIME STRCS R3, TimerAlpha +4 STRCS R3, TimerBeta +0 STRCS R3, TimerBeta +4 MOV R4, R1 ; preserve LastBREAK MOV R1, WsPtr ; start at the beginning ADR R11, ByteVarInitTable ByteVarInitLoop LDRB R0, [R11], #1 ; copy a byte from table STRB R0, [R1], #1 ; to vars TEQ R1, R2 ; at end ? BNE ByteVarInitLoop ; [no, then loop] STRB R3, NoIgnore ; put NoIgnore back STRB R4, LastBREAK ; put LastBREAK back ; Initialise buffer pointers LDR R0, =ZeroPage+4*(NBuffers-1) ; index to pointer MOV R1, #0 ; value to store MOV R2, #NBuffers-1 BuffPtrInitLoop STR R1, [R0, #BuffInPtrs] STR R1, [R0, #BuffOutPtrs] SUB R0, R0, #4 SUBS R2, R2, #1 BPL BuffPtrInitLoop ; mark printer as dormant STR R1, PrinterActive ; (R1=0) ; Initialise event semaphores ADR R0, EventSemaphores ADD R2, R0, #32 10 STR R1, [R0], #4 ; clear all 32 event semaphores TEQ R0, R2 BNE %BT10 STRB R1, FlashState STRB R1, SerialInHandle ; zero serial handles STRB R1, SerialOutHandle ; Initialise LatchB and soft copy ; MOV R1, #0 ; AND with 0 (was omitted on earlier MOS) ; R1 already zero MOV R0, #0 ; EOR with 0 BL UpdateLatchB [ HAL ; This has already been done much earlier in initialisation | ; set up IOC timer 0 before we read from CMOS (cos it uses timer for delays) MOV R1, #IOC LDR R0, =20000-1 ; R0 = Timer delay (units of 0.5 microsecond) ; 20000*0.5E-6 = 0.01 Seconds 100Hz ; TMD 21-May-93: "-1" correction applied STRB R0, [R1, #Timer0LL] ; Set up the delay MOV R0, R0, LSR #8 STRB R0, [R1, #Timer0LH] STRB R0, [R1, #Timer0GO] ; and start the ticks ] LDRB R0, LastBREAK TEQ R0, #0 BEQ %FT20 BL ReadMachineType BL ReadUniqueID BL ReadHardCMOSDefaults 20 BL ReadCMOSDefaults [ :LNOT: HAL [ StorkPowerSave BL PowerHardware ;On Stork, ensure Combo chip, Winnie, Floppy etc are powered ] [ Keyboard_Type <> "RCMM" ; On RCMM machines, this is done in SetUpKbd. [ STB BL ConfigureCombo | BL Configure37C665 ;RiscPC, Kryten and Stork use only SMC 37C665 ] ] ; <> "RCMM" ] [ HAL Push "r9,r12" AddressHAL MOV R0, #0 CallHAL HAL_TimerDevice MOV R4, R0 CallHAL HAL_IRQClear ; clear timer 0 IRQ MOV R0, R4 CallHAL HAL_IRQEnable ; enable timer 0 IRQ Pull "r9,r12" | MOV R1, #IOC MOV r0, #timer0_bit :OR: vsync_bit STRB R0, [R1, #IOCIRQCLRA] ; clear pending tim0, vsync irqs (+ pack irq if appropriate) LDRB R0, [R1, #IOCIRQMSKA] ORR R0, R0, #timer0_bit :OR: vsync_bit STRB R0, [R1, #IOCIRQMSKA] ; Enable timer 0 + vsync irqs ] [ :LNOT: HAL MOV R0, #0 STRB R0, [R1, #IOMD_ATODICR] ; power down the A to D convertor ] ; The RTC driver is later on in the module chain, or missing, for now default the ; system time to UNIX epoch + 1 day (so time() doesn't return -1) secs0070 * (86400*(365*70+18)) ; from time() in risc_oslib.c.armsys LDR R7, =(secs0070 * 100) ; centiseconds LSW MOV R8, #&33 ; centiseconds MSW BL Store5ByteInRealTime ; insert soft key 10 MOV R2, #&CA BL RDCHS Pull PC LTORG ; ***************************************************************************** ; ; ReadHardCMOSDefaults - Read CMOS values for a hard/power-on reset ; NB must be called in supervisor mode ReadHardCMOSDefaults Push R14 MOV R0, #PigCMOS BL Read STRB R0, PrinterIgnore MOV R0, #PSITCMOS BL Read TST R0, #2 ; NoIgnore bit MOVEQ R1, #0 MOVNE R1, #&80 STRB R1, NoIgnore MOV R1, R0, LSR #5 ; printer type now in bits 0..2 STRB R1, PrinterDrivType MOV R0, #MODETVCMOS BL Read MOV R2, R0, LSR #4 ; bit0:=interlace, bits 1-3 := vertical AND R1, R2, #1 STRB R1, TVInterlace MOV R2, R2, LSL #31-3 ; bits 29-31 := vertical MOV R2, R2, ASR #29 ; sign extend STRB R2, TVVertical MOV R0, #DBTBCMOS BL Read LDRB R1, StartOptions TST R0, #&10 ; bit 4 = boot bit ORREQ R1, R1, #8 ; noboot => set bit 3 BICNE R1, R1, #8 ; boot => clear bit 3 STRB R1, StartOptions LDR R2, =ZeroPage+VduDriverWorkSpace+CursorFlags ANDS R1, R0, #8 ; noscroll bit - put 0 or 1 MOVNE R1, #1 ; in bottom byte of CursorFlags STRB R1, [R2] ; leave other bytes alone MOV R0, #CountryCMOS ; read country CMOS and store in var BL Read ; but don't bind 'Default' to a fixed STRB R0, Country ; country at this stage BL SetUpPrinterBuffer Pull PC ; ***************************************************************************** ; ; ReadCMOSDefaults - Read CMOS values for any reset ; NB must be called in supervisor mode ReadCMOSDefaults Push R14 MOV R0, #DBTBCMOS BL Read TST R0, #2 ; NZ => loud MOVEQ R1, #&D0 ; (quiet) MOVNE R1, #&90 ; (LOUD) STRB R1, BELLinfo MOV R0, #StartCMOS BL Read MOVS R1, R0, LSL #(32-5) ; bit 5 -> carry, bit 4 -> N bit MOVPL R1, #KBStat_NoShiftLock + KBStat_ShiftEnable ; SHCAPS MOVMI R1, #KBStat_NoShiftLock + KBStat_NoCapsLock ; NOCAPS MOVCS R1, #KBStat_NoShiftLock ; CAPS TST R0, #1:SHL:7 ; [NO]NUM ORRNE R1, R1, #KBStat_NoNumLock BICEQ R1, R1, #KBStat_NoNumLock STRB R1, KeyBdStatus MOV R0, #SystemSpeedCMOS BL Read TST R0, #&20 ; Cache off when b5 set LDR R2, =:NOT: (MMUC_I + MMUC_C + MMUC_W) LDRNE R1, =0 LDREQ R1, =MMUC_I + MMUC_C + MMUC_W MOV R0, #MMUCReason_ModifyControl BL MMUControlSub [ ModeSelectors BL Read_Configd_MonitorType LDR r1, =ZeroPage+VduDriverWorkSpace+CurrentMonitorType ; set current to default STR r0, [r1] ] Pull R14 ; and drop thru to ... ReadKeyDefaults Push R14 MOV R0, #KeyDelCMOS ; Read the default out of CMOS RAM BL Read ; comes back in R0 STRB R0, KeyRepDelay MOV R0, #KeyRepCMOS ; Read the default out of CMOS RAM BL Read ; comes back in R0 STRB R0, KeyRepRate Pull PC ; ***************************************************************************** ; ; PostInit - Called by Sam after modules have been initialised ; PostInit ROUT Push R14 BYTEWS WsPtr LDRB R0, LastBREAK ; get reset type TEQ R0, #0 BEQ %FT10 ; [soft reset, skip] [ StorkPowerSave SWI XPortable_ReadFeatures BVC %FT01 ; MOV R0, #0 MOV R1, #0 SWI XPortable_Speed ; attempt to make the portable go fast! MOVVC R1, #PortableFeature_Speed MOVVS R1, #0 01 AND R1, R1, #(PortableFeature_Speed :OR: PortableFeature_Idle :OR: PortableFeature_Stop) LDR R0, =ZeroPage STRB R1, [R0, #PortableFlags] | LDR r0, =ZeroPage ; allow SWI Portable_Speed to be issued ASSERT (ZeroPage :AND: 255) = 0 STRB r0, [r0, #PortableFlag] ] [ BleedinDaveBell MOV R0, #1 ; indicate keyboard UK MOV R1, #101 ; indicate alphabet Latin1 | MOV R0, #2 ; indicate keyboard Master MOV R1, #100 ; indicate alphabet Bfont ] STRB R0, Keyboard STRB R1, Alphabet STRB R1, KeyAlphabet ; alphabet corresponding to keyboard 10 Pull R14 B KeyPostInit ; ***************************************************************************** ; ; UpdateLatchB - update latch B and soft copy ; ; LATCHB := (LATCHB AND R1) EOR R0 ; UpdateLatchB Push "R2, R3, R14" PHPSEI ; disable IRQ LDR R2, =ZeroPage LDRB R3, [R2, #LatchBSoftCopy] AND R3, R3, R1 EOR R3, R3, R0 STRB R3, [R2, #LatchBSoftCopy] PLP Pull "R2, R3, PC" [ :LNOT: STB :LAND: :LNOT: HAL ; ***************************************************************************** ; ; UpdateMonitorTypeLatch - update monitor type latch and soft copy ; ; Returns the result in R4 UpdateMonitorTypeLatch Push "R2, R3, R14" MRS R14, CPSR ORR R2, R14, #I32_bit MSR CPSR_c, R2 ; disable IRQ LDR R2, =ZeroPage LDRB R3, [R2, #CLine_Softcopy] MOV R3, #1 ;Set our bit only STRB R3, [R2, #CLine_Softcopy] MOV R2, #IOMD_Base STRB R3, [R2, #IOMD_CLINES] ;Write the new reg out LDRB R3, [R2, #IOMD_CLINES] ;Read it back AND R4, R3, #1 ;Clear all but our bit into R4 MSR CPSR_cf, R14 ;Re-enable IRQ, restore flags Pull "R2, R3, PC" ; ***************************************************************************** ] ; The initial values for all of the osbyte variables ; as decreed by arthur. ByteVarInitTable ; The main osbyte variables, accessed ; via calls &A6 to &FF DCW OsbyteVars-&A6 ; VarStart # 2 ; &A6,&A7 = 0,0 ; ROMPtr # 2 ; &A8,&A9 = 0,0 ; ROMInfo # 2 ; &AA,&AB = 0,0 ; KBTran # 2 ; &AC,&AD = 0,0 ; VDUvars # 2 ; &AE,&AF ; = 0 ; CFStime # 1 ; &B0 = 0 ; InputStream # 1 ; &B1 = &FF ; KeyBdSema # 1 ; &B2 ; = &00 ; ROMPollSema # 1 ; &B3 = &80 ; OSHWM # 1 ; &B4 (hi-byte of &8000) ; = 1 ; RS423mode # 1 ; &B5 = 0 ; NoIgnore # 1 ; &B6 = &00 ; CFSRFS # 1 ; &B7 = &00,&00 ; VULAcopy # 2 ; &B8,&B9 ; = &00 ; ROMatBRK # 1 ; &BA = &FF ; BASICROM # 1 ; &BB ; = &04 ; ADCchanel # 1 ; &BC = &04 ; ADCmaxchn # 1 ; &BD = &00 ; ADCconv # 1 ; &BE ; = &FF ; RS432use # 1 ; &BF = &42 ; RS432conflag # 1 ; &C0 ; = &19 ; FlashCount # 1 ; &C1 = &19 ; SpacPeriod # 1 ; &C2 = &19 ; MarkPeriod # 1 ; &C3 ; = &32 ; KeyRepDelay # 1 ; &C4 = &08 ; KeyRepRate # 1 ; &C5 ; = &00 ; ExecFileH # 1 ; &C6 = &00 ; SpoolFileH # 1 ; &C7 ; = &00 ; ESCBREAK # 1 ; &C8 (200) ; = &00 ; KeyBdDisable # 1 ; &C9 = &30 ; KeyBdStatus # 1 ; &CA ; = &11 ; RS423HandShake # 1 ; &CB = &00 ; RS423InputSupr # 1 ; &CC = &00 ; RS423CFSFlag # 1 ; &CD ; = &00 ; EconetOScall # 1 ; &CE = &00 ; EconetOSrdch # 1 ; &CF = &00 ; EconetOSwrch # 1 ; &D0 ; = &00 ; SpeechSupr # 1 ; &D1 = &00 ; SoundSupr # 1 ; &D2 ; = &01 ; BELLchannel # 1 ; &D3 = &90 ; BELLinfo # 1 ; &D4 = &64 ; BELLfreq # 1 ; &D5 = &06 ; BELLdur # 1 ; &D6 ; = &81 ; StartMessSupr # 1 ; &D7 ; = &00 ; SoftKeyLen # 1 ; &D8 ; = &00 ; PageModeLineCount # 1 ; &D9 ; = &00 ; VDUqueueItems # 1 ; &DA ; = &09 ; TABch # 1 ; &DB = &1B ; ESCch # 1 ; &DC ; = &01,&D0,&E0,&F0 ; IPbufferCh # 4 ; &DD,&DE,&DF,&E0 = &01,&80,&90,&00 ; RedKeyCh # 4 ; &E1,&E2,&E3,&E4 ; = &00 ; ESCaction # 1 ; &E5 = &00 ; ESCeffect # 1 ; &E6 ; = &00 ; u6522IRQ # 1 ; &E7 = &00 ; s6850IRQ # 1 ; &E8 = &00 ; s6522IRQ # 1 ; &E9 ; = &00 ; TubeFlag # 1 ; &EA ; = &00 ; SpeechFlag # 1 ; &EB ; = &00 ; WrchDest # 1 ; &EC = &00 ; CurEdit # 1 ; &ED ; = &30 ; KeyBase ; &EE = &01 ; Shadow ; &EF = &00 ; Country ; &F0 ; = &00 ; UserFlag # 1 ; &F1 ; = &64 ; SerULAreg # 1 ; &F2 ; = &05 ; TimerState # 1 ; &F3 ; = &FF ; SoftKeyConsist # 1 ; &F4 ; = &01 ; PrinterDrivType # 1 ; &F5 = &0A ; PrinterIgnore # 1 ; &F6 ; = &01,&00,&00 ; BREAKvector # 3 ; &F7,&F8,&F9 ; = &00 ; DRIVER ; &FA = &00 ; DISPLAY ; &FB ; = &FF ; LangROM # 1 ; &FC ; = &01 ; LastBREAK # 1 ; &FD ; = &0F ; KeyOpt # 1 ; &FE ; = &08 ; StartOptions # 1 ; &FF ; ; ByteVarInitTableEnd ByteVarInitTableSize * ByteVarInitTableEnd - ByteVarInitTable ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ LTORG oldirqowner & IRQ ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ; ; ReadMachineType - Determine machine type and store it in IOSystemType ; ReadMachineType Entry "r0-r12" [ HAL LDR r1, =ZeroPage MOV r0, #4_3330 ; Assume VGA during osinit STRB r0, [r1, #MonitorLeadType] EXIT | MOV r12, #IOMD_Base LDRB r0, [r12, #IOMD_ID0] LDRB r11, [r12, #IOMD_ID1] ORR r0, r0, r11, LSL #8 LDR r11, =IOMD_Original TEQ r0, r11 MOVEQ r11, #0 ; assume Medusa MOVNE r11, #IOST_7500 ; else assume Morris TEQ r11, #IOST_7500 ; and set EQ if Morris to do conditional stuff below ; ; On Kryten, Morris pin Event2 is tied low so bit Nevent2 is a ONE ; On Stork, Morris pin Event2 is tied high so bit Nevent2 is a ZERO ; On STB/NCD, Morris pin Event2 is tied high so bit Nevent2 is a ZERO, but we don't have a LCD ; controller or battery manager so we'll not set IOST_BATMAN ; LDREQB r0, [r12, #IOMD_IRQSTD] ;EQ, Morris [ BatManSupport TSTEQ r0, #IOMD_Nevent2_bit ORREQ r11, r11, #IOST_BATMAN ;EQ, Stork ie Morris & BATMAN ] ORR r0, r11, #IOST_IOEB ; pretend we've got IOEB ; ; r11 holds 0 for IOMD (Risc PC) ; or IOST_7500 for Morris (Kryten, Falcon, Omega) ; or IOST_7500 + IOST_BATMAN for Morris (Stork) MOV r1, #0 ; normal Hsync and address pointer LDRB r3, [r1, #IOSystemType] AND r3, r3, #IOST_COMBOMASK ; preserve combo type which may already ORR r3, r3, r0 ; be in there on RCMM machines STRB r3, [r1, #IOSystemType] ; now read monitor lead type. TEQ r0, #0 ; no IOEB MOVEQ r2, #&FF ; then return all ones for monitor lead type BEQ %FT90 [ MPEGPoduleNTSCNotPALMask <> 0 ! 1, "Sorry, I don't do MPEGPoduleNTSCNotPALMask any more" ] [ STB LDR r0, =VIDC ; on VIDC20 we invert HSYNC by writing to External Register [ IOMD_C_MonitorType <> 0 ASSERT (IOMD_C_MonitorType = (1<<0)) ; this code only understands auto-detect in bit 0 ] TST R11, #IOST_7500 LDREQ r3, =IOMD_MonitorType ; Not Morris, address is in old place LDRNE r3, =(IOMD_Base + IOMD_CLINES) LDR r1, =VIDCExternal+Ext_InvertCompVSYNC+Ext_DACsOn+Ext_ERegExt ; normal HSYNC value STR r1, [r0] ORR r2, r1, #Ext_InvertHSYNC ; value for inverted HSYNC LDRB r4, [r3] ; base value [ MorrisSupport [ IOMD_C_PALNTSCType <> 0 ASSERT (IOMD_C_PALNTSCType = (1<<4)) ; this code only understands PAL/NTSC auto-detect in bit 4 AND r4, r4, #(IOMD_C_MonitorType :OR: IOMD_C_PALNTSCType) ORR r4, r4, r4, LSR #1 ; Shift PAL/NTSC bit into bit 3 AND r4, r4, #&0F ; only use bits 0..3 | AND r4, r4, #IOMD_C_MonitorType ; only one bit ] | AND r4, r4, #&0F ; only use bits 0..3 ] MOV r5, #0 ; bits 0..3 = bits which have ever changed ; bits 4..7 = bits whose deinverted value was high last time ; bits 8..11 = bits whose deinverted value just went high-low ; bits 12..15 = bits whose deinverted value just went high-low-low ; bits 16..19 = bits which could be hsync ; ie after every high-low there was low-low (after deinversion) ; bits 20..23 = bits which are definitely random MOV r10, #&0F ; used inside CheckBits MOV r12, #256 ; number of iterations 20 STR r2, [r0] LDRB r6, [r3] ; read value with inverted sync STR r1, [r0] LDRB r7, [r3] ; read value with normal sync [ MorrisSupport [ IOMD_C_PALNTSCType <> 0 AND r6, r6, #(IOMD_C_MonitorType :OR: IOMD_C_PALNTSCType) ORR r6, r6, r6, LSR #1 ; Shift PAL/NTSC bit into bit 3 AND r6, r6, #&0F ; only use bits 0..3 AND r7, r7, #(IOMD_C_MonitorType :OR: IOMD_C_PALNTSCType) ORR r7, r7, r7, LSR #1 ; Shift PAL/NTSC bit into bit 3 AND r7, r7, #&0F ; only use bits 0..3 | AND r6, r6, #IOMD_C_MonitorType ; only one bit AND r7, r7, #IOMD_C_MonitorType ] | AND r6, r6, #&0F ; only use bits 0..3 AND r7, r7, #&0F ] EOR r8, r6, r4 ; bits which have changed from steady value to inverted one ORR r5, r5, r8 ; OR into mask of bits which have ever changed EOR r8, r7, r4 ; bits which have changed from steady value to normal one ORR r5, r5, r8 ; OR into mask of bits which have ever changed EOR r6, r6, #&0F ; deinvert inverted value BL CheckBits ; call check routine with first value MOV r6, r7 BL CheckBits ; call check routine with second value SUBS r12, r12, #1 BNE %BT20 | ; STB LDR r0, =VIDC ; on VIDC20 we invert HSYNC by writing to External Register LDR r3, =IOMD_MonitorType LDR r1, =VIDCExternal+Ext_InvertCompVSYNC+Ext_DACsOn+Ext_ERegExt ; normal HSYNC value STR r1, [r0] ORR r2, r1, #Ext_InvertHSYNC ; value for inverted HSYNC [ MorrisSupport TST R11, #IOST_7500 BLNE UpdateMonitorTypeLatch ;Result back in R4 LDREQB r4, [r3] | LDRB r4, [r3] ; base value ] AND r4, r4, #&0F ; only use bits 0..3 MOV r5, #0 ; bits 0..3 = bits which have ever changed ; bits 4..7 = bits whose deinverted value was high last time ; bits 8..11 = bits whose deinverted value just went high-low ; bits 12..15 = bits whose deinverted value just went high-low-low ; bits 16..19 = bits which could be hsync ; ie after every high-low there was low-low (after deinversion) ; bits 20..23 = bits which are definitely random MOV r10, #&0F ; used inside CheckBits MOV r12, #256 ; number of iterations 20 STR r2, [r0] [ MorrisSupport TST R11, #IOST_7500 BLNE UpdateMonitorTypeLatch ;Result back in R4 MOVNE r6, r4 LDREQB r6, [r3] | LDRB r6, [r3] ; read value with inverted sync ] STR r1, [r0] [ MorrisSupport TST R11, #IOST_7500 BLNE UpdateMonitorTypeLatch ;Result back in R4 MOVNE r7,r4 LDREQB r7,[r3] | LDRB r7, [r3] ; read value with normal sync ] AND r6, r6, #&0F AND r7, r7, #&0F EOR r8, r6, r4 ; bits which have changed from steady value to inverted one ORR r5, r5, r8 ; OR into mask of bits which have ever changed EOR r8, r7, r4 ; bits which have changed from steady value to normal one ORR r5, r5, r8 ; OR into mask of bits which have ever changed EOR r6, r6, #&0F ; deinvert inverted value BL CheckBits ; call check routine with first value MOV r6, r7 BL CheckBits ; call check routine with second value SUBS r12, r12, #1 BNE %BT20 ] ; STB ; now process result LDR r1, =VIDCExternal+Ext_InvertCompVSYNC+Ext_InvertCompHSYNC+Ext_DACsOn+Ext_ERegExt ; put back default value STR r1, [r0] BIC r4, r4, r5 ; don't put port value in for bits that have changed BIC r5, r5, r5, LSR #16 ; make bits 0..3 of r5 indicate random bits ANDS r2, r4, #1 ; for each bit pair 00 => low, 01 => high, 10 => Hsync, 11 => random TST r5, #1 :SHL: 16 MOVNE r2, #2 :SHL: 0 TST r5, #1 MOVNE r2, #3 :SHL: 0 TST r4, #2 ORRNE r2, r2, #1 :SHL: 2 TST r5, #2 :SHL: 16 ORRNE r2, r2, #2 :SHL: 2 TST r5, #2 ORRNE r2, r2, #3 :SHL: 2 TST r4, #4 ORRNE r2, r2, #1 :SHL: 4 TST r5, #4 :SHL: 16 ORRNE r2, r2, #2 :SHL: 4 TST r5, #4 ORRNE r2, r2, #3 :SHL: 4 TST r4, #8 ORRNE r2, r2, #1 :SHL: 6 TST r5, #8 :SHL: 16 ORRNE r2, r2, #2 :SHL: 6 TST r5, #8 ORRNE r2, r2, #3 :SHL: 6 [ {FALSE} ASSERT IOMD_MonitorIDMask = 1 AND r2, r2, #3 ; only bit 0 of ID valid on IOMD-based systems ] 90 MOV r1, #0 STRB r2, [r1, #MonitorLeadType] EXIT ] CheckBits ROUT AND r8, r10, r5, LSR #12 ; bits that were H-L-L BIC r8, r8, r6 ; bits that are H-L-L-L ORR r5, r5, r8, LSL #16 ; OR into bits that could be hsync ORR r8, r5, r5, LSR #4 AND r8, r6, r8, LSR #8 ; bits that just went H-L-H or H-L-L-H AND r8, r8, r5, LSR #16 ; bits that just went H-L-H or H-L-L-H and could have been hsync ORR r5, r5, r8, LSL #20 ; they're definitely random now BIC r5, r5, r8, LSL #16 ; and they're definitely not hsync now AND r8, r5, #&FF :SHL: 4 ; get H bits, and H-L bits BIC r8, r8, r6, LSL #4 ; knock out bits that were H and are now H BIC r8, r8, r6, LSL #8 ; knock out bits that were H-L and are now H BIC r5, r5, #&FF :SHL: 8 ; knock out all H-L and H-L-L bits ORR r5, r5, r8, LSL #4 ; put in new H-L and H-L-L bits BIC r5, r5, #&F :SHL: 4 ; knock out old H bits ORR r5, r5, r6, LSL #4 ; put in new H bits BIC r5, r5, r5, LSR #20 ; knock out H bits if we know it's random MOV pc, lr ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ; ; TranslateMonitorLeadType - Determine monitor type and default mode + sync from monitor lead type ; ; in: Monitor lead type in variable MonitorLeadType (surprisingly!) ; ; out: r3 = default mode to use ; r4 = default monitortype to use ; r5 = default sync to use ; TranslateMonitorLeadType Entry "r0-r2" [ ZeroPage = 0 MOV r1, #Service_MonitorLeadTranslation LDRB r2, [r1, #MonitorLeadType-Service_MonitorLeadTranslation] | LDR r2, =ZeroPage MOV r1, #Service_MonitorLeadTranslation LDRB r2, [r2, #MonitorLeadType] ] SWI XOS_ServiceCall TEQ r1, #0 ; if service claimed, then exit with these numbers EXIT EQ ADR r0, MonitorLeadList 10 LDR r14, [r0], #4 EOR r1, r2, r14, LSR #24 ; differences EOR r14, r14, #&FF000000 ; make don't cares into zero TST r14, #&C0000000 BICEQ r1, r1, #&C0 ; knock out difference pairs if don't care TST r14, #&30000000 BICEQ r1, r1, #&30 TST r14, #&0C000000 BICEQ r1, r1, #&0C TST r14, #&03000000 BICEQ r1, r1, #&03 TEQ r1, #0 ; if still have differences, then loop BNE %BT10 MOV r0, #&FF AND r3, r0, r14 ; mode in bits 0..7 AND r4, r0, r14, LSR #8 ; monitortype in bits 8..15 AND r5, r0, r14, LSR #16 ; sync in bits 16..23 ; Give the current GraphicsV driver a chance to specify a better mode ; than whatever we've picked here Push "r4" MOV r0, r3 VDWS r4 LDR r4, [r4, #CurrentGraphicsVDriver] MOV r4, r4, LSL #24 ORR r4, r4, #GraphicsV_StartupMode BL CallGraphicsV Pull "r4" MOV r3, r0 EXIT MACRO MonitorLeadItem $lead, $mode, $monitortype, $sync ASSERT $lead < 256 ASSERT $mode < 256 ASSERT $monitortype < 256 ASSERT $sync < 256 DCD (($lead):SHL:24):OR:(($sync):SHL:16):OR:(($monitortype):SHL:8):OR:($mode) MEND MonitorLeadList [ STB [ IOMD_C_MonitorType = 0 ; no auto-detect bit [ IOMD_C_PALNTSCType = 0 ; no PAL/NTSC bits: MonitorLeadItem 4_3333, 12, 0, 1 ; PAL TV assumed | ; IOMD_C_PALNTSCType = 0 MonitorLeadItem 4_0333, 12, 0, 1 ; PAL TV MonitorLeadItem 4_1333, 46, 8, 1 ; NTSC TV ] ; IOMD_C_PALNTSCType = 0 | [ :LNOT: ChrontelSupport [ IOMD_C_PALNTSCType = 0 ; no PAL/NTSC bits: MonitorLeadItem 4_3331, 12, 0, 1 ; PAL TV assumed | ; IOMD_C_PALNTSCType = 0 ; wealth of bits: MonitorLeadItem 4_0331, 12, 0, 1 ; PAL TV MonitorLeadItem 4_1331, 46, 8, 1 ; NTSC TV ] ; IOMD_C_PALNTSCType = 0 ] ; :LNOT: ChrontelSupport ] ; IOMD_C_MonitorType = 0 MonitorLeadItem 4_3333, 28, 3, 0 ; VGA-capable monitors 256 colours | ; STB ; KJB - changed default modes to 256 colours MonitorLeadItem 4_3330, 28, 3, 0 ; VGA-capable monitors MonitorLeadItem 4_3111, 28, 5, 0 ; Nothing - try LCD (fudge fudge) MonitorLeadItem 4_3333, 15, 0, 1 ; Others - assume TV standard ]; STB [ StorkPowerSave :LAND: :LNOT: HAL ; ; List of latch addresses and initial values. ; PowerTab DCD HWLatchPA, InitLatchPA DCD HWLatchPB, InitLatchPB DCD HWLatchMC, InitLatchMC DCD HWLatchMA, InitLatchMA DCD 0 PowerHardware ;On Stork, ensure Combo chip, Winnie, Floppy etc are powered Entry "r0,r1" MOV r0, #0 LDRB lr, [r0, #IOSystemType] TST lr, #IOST_BATMAN EXIT EQ ;EQ, not Stork, so hardware already powered ; ; On Stork. ; ; Now would be a good time to hit the power control latches ; to ensure everything is powered up. ; ADR R14, PowerTab 05 LDMIA R14!, {R0, R1} TEQ R0, #0 STRNEB R1, [R0] BNE %BT05 10 EXIT ] [ :LNOT: HAL [ STB ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ; ; ConfigureCombo - Configure SMC 665/669 or UMC 8669 if present ; ComboBase * &03010000 ; Base address of combo chip = PC/AT I/O 000H ; SMC-type stuff SMC_CSR * &03F0 * 4 ; Configuration Select Register (CSR) SMC_CSRalt669 * &0370 * 4 ; Alternative Configuration Select Register on SMC 669 ; (this register is used if RTS2 is high on reset, which we don't have direct control of) SMC_config * &55 ; value to write to enter configuration mode SMC_endconfig * &AA ; value to write to end configuration mode SMC_IDreg * &0D ; device ID register SMC_data * &03F1 * 4 ; Configuration Access Port (data) SMC_665 * &65 ; 665 ID SMC_669 * &03 ; 669 ID ; UMC-type stuff UMC_CSR * &0108 * 4 ; Configuration Select Register (CSR) UMC_config * &AA ; value to write to enter configuration mode UMC_endconfig * &55 ; value to write to end configuration mode UMC_data * &0109 * 4 ; Configuration Access Port (data) ; ; Configure SMC 37C665/669 or UMC8669 combo chip ; ConfigureCombo Entry "r0-r2" WritePSRc SVC_mode + I_bit + F_bit, R0 ; Disable FIQ and IRQ LDR r0, =ComboBase ; R0-> Base address of combo chip ; See if we have a SMC665/669 and try to configure it ADD r2, r0, #SMC_CSR ; first try in normal place MOV lr, #SMC_config STRB lr, [r2] ; Write &55 to CSR twice STRB lr, [r2] ; to enter configuration mode MOV lr, #SMC_IDreg STRB lr, [r2] LDRB lr, [r2, #SMC_data-SMC_CSR] TEQ lr, #SMC_665 ; 665 ? ADREQ r1, ConfigSMC665Table BEQ %FT20 TEQ lr, #SMC_669 ; 669 ? BEQ %FT18 ; Not a SMC chip at the normal place, but the SMC 669 may move its configuration registers ; to &370 if RTS2 was high on trailing edge of reset, so try there ADD r2, r0, #SMC_CSRalt669 MOV lr, #SMC_config STRB lr, [r2] ; Write &55 to CSR twice STRB lr, [r2] ; to enter configuration mode MOV lr, #SMC_IDreg STRB lr, [r2] LDRB lr, [r2, #SMC_data-SMC_CSR] TEQ lr, #SMC_669 ; 669 ? BEQ %FT18 ; Not a SMC chip that we recognise, maybe we have a UMC8669 ADR r1, ConfigUMC8669Table 10 LDRB lr, [r1], #1 ; get "Index Entry Valid" STRB lr, [r0, #UMC_CSR] TEQ lr, #UMC_endconfig ; end of table? BEQ %FT12 LDRB lr, [r1], #1 ; get config index STRB lr, [r0, #UMC_CSR] LDRB lr, [r1], #1 ; get config data STRB lr, [r0, #UMC_data] ; and write it BNE %BT10 12 ; UMC8669 is not self-identifying, so see if what we wrote is still there MOV lr, #UMC_config STRB lr, [r0, #UMC_CSR] ; Write &AA to enter config mode MOV lr, #&C0 STRB lr, [r0, #UMC_CSR] ; CR0 LDRB r2, [r0, #UMC_data] MOV lr, #UMC_endconfig ; maybe don't need to do this STRB lr, [r0, #UMC_CSR] ; exit config mode TEQ r2, #2_00111110 ; Value for CR0 from ConfigUMC8669Table MOVEQ r1, #IOST_UMC669 MOVNE r1, #0 ; Don't know what this is, give up B %FT30 ; SMC config loop 18 ADR r1, ConfigSMC669Table 20 LDRB lr, [r1], #1 ; get config index STRB lr, [r2] TEQ lr, #SMC_endconfig ; end of table? LDRNEB lr, [r1], #1 ; if not then get config data STRNEB lr, [r2, #SMC_data-SMC_CSR] ; and write it BNE %BT20 LDRB r1, [r1] ; pull chip ID out of table ; Record type of chip found 30 MOV r0, #0 LDRB lr, [r0, #IOSystemType] BIC lr, lr, #IOST_COMBOMASK ORR lr, lr, r1 STRB lr, [r0, #IOSystemType] WritePSRc SVC_mode + I_bit, lr ; Restore IRQ/FIQ state EXIT ConfigSMC665Table [ ComboIRQsActiveHigh DCB &01, 2_10010111 ; Enable config, COM3@338, COM4@238, IRQs active hi, ; // is extended & powered @278 | DCB &01, 2_10000111 ; Enable config, COM3@338, COM4@238, IRQs active low, ; // is extended & powered @278 ] DCB &02, 2_11011100 ; COM2 powered & enabled @2F8, COM1 powered & enabled @3F8 ; (default) DCB &03, 2_01111000 ; floppy stuff (default) DCB &04, 2_00000011 ; EPP v1.9, MIDI disabled, normal //floppy, ; // uses ECP & EPP modes DCB &05, 0 ; 4 drive support, don't swap drives, normal density, ; FDC burst mode, IDE@1F0-7,3F6-7, FDC@3F0-7 (default) ; DCB &06, &FF ; floppy drive types (default) ; DCB &07, 0 ; don't auto-powerdown anything (default) ; DCB &08, 0 ; ADRA7:4 address decode (default) ; DCB &09, 0 ; ADRA10:8 address decode (default) ; DCB &0A, 0 ; FIFO threshold for ECP // = ??? (default) ; DCB &0B, 0 ; floppy data rates (default) ; DCB &0C, 0 ; UART2 & UART1 standard speed, UART2 standard mode, ; UART2 full duplex, XMIT active hi, RCV active hi (default) DCB &00, 2_10111011 ; Valid config, OSC & BR on, FDC enabled & powered, ; IDE AT & enabled DCB SMC_endconfig ; Exit config mode DCB IOST_37C665 ; Handy place to keep ID ConfigSMC669Table DCB &01, 2_10010100 ; Enable config, // is extended, // is powered DCB &02, 2_10001000 ; COM2 powered, COM1 powered (default) DCB &03, 2_01110000 ; floppy stuff (bit 3 now reserved) DCB &04, 2_00000011 ; IR rx&tx on COM2 rx & tx pins, EPP v1.9, MIDI disabled, ; normal //floppy, uses ECP & EPP modes DCB &05, 0 ; 4 drive support, don't swap drives, normal density, ; FDC burst mode (default) ; DCB &06, &FF ; floppy drive types (default) ; DCB &07, 0 ; don't auto-powerdown anything ; DCB &08, 0 ; ADRA7:4 address decode (default) ; DCB &09, 0 ; ADRx disabled, ADRA10:8 address decode (default) ; DCB &0A, 0 ; FIFO threshold for ECP // = ??? (default) ; DCB &0B, 0 ; floppy data rates (default) ; DCB &0C, 0 ; UART2 & UART1 standard speed, UART2 standard mode, ; UART2 full duplex, XMIT active hi, RCV active hi (default) [ ComboClock14MHz ; DCB &10, 2_00000000 ; 14.318MHz input to PLL (default) | DCB &10, 2_01000000 ; 24MHz input to PLL ] ; DCB &1E, &80 ; GAMECS disabled (default) ; DCB &1F, 0 ; floppy drive types (default) DCB &20, &FC ; FDC@3F0-7 DCB &21, &7C ; IDE@1F0-7 DCB &22, &FD ; IDE Alternate Status Register @3F6 DCB &23, &9E ; //@278 DCB &24, &FE ; COM1@3F8 DCB &25, &BE ; COM2@2F8 DCB &26, 0 ; no FDC DMA, no // DMA (default) DCB &27, 2_01100101 ; FDC uses IRQ_F, // uses IRQ_E DCB &28, 2_01000011 ; UART1 uses IRQ_D, UART2 uses IRQ_C DCB &29, 0 ; IRQIN does not use any IRQ_x (default) DCB &00, 2_10001010 ; Valid config, FDC powered, IDE enabled DCB SMC_endconfig ; Exit config mode DCB IOST_37C669 ConfigUMC8669Table DCB UMC_config, &C0, 2_00111110 ; IR full-duplex, games off, IDE on, ; // in EPP & ECP mode, UART2 on, UART1 on, FDC off DCB UMC_config, &C1, 2_00101111 ; Direct access PnP register, Disable PnP, ; IDE@1F0-7,3F6-7, //@278, COM2@2F8, COM1@3F8, ; FDC@3F0-7 DCB UMC_config, &C2, 2_10000001 ; Not supspended, IR unselected/disabled, ; don't swap floppy, IBM mode floppy, ; floppy is R/W (default) DCB UMC_endconfig, 0 ; Exit config mode ALIGN | ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ; ; Configure82C710 - Configure 82C710/82C711/SMC 665 if present ; ; 82C710 stuff CnTbase * &03010000 ; Base address of 82C710 = PC/AT I/O 000H CRI710 * &0390 CRI710Off * CRI710*4 ; 82C710 Configuration Register Index port CAP710Off * CRI710Off +4 ; 82C710 Configuration Access Port (data) ConRegA710 * &02FA*4 ConRegB710 * &03FA*4 CRI711Off * &03F0*4 ; 82C711 Configuration Register Index port CAP711Off * CRI711Off +4 ; 82C711 Configuration Access Port (data) ConfigSMCTable DCB &01, 2_10000111 ; Enable config read, IRQ active low, parallel powered/extended, default addr. DCB &02, 2_00011100 ; 2nd serial port disabled, 1st enabled at &3F8 DCB &03, &78 ; extra stuff for SMC DCB &04, 2_00000011 ; allow extended parallel port modes DCB &05, 0 DCB &06, &FF DCB &07, 0 DCB &08, 0 DCB &09, 0 DCB &0A, 0 DCB &00, 2_10111011 ; Valid config, OSC/BR on, FDC enabled/powered, IDE AT,enabled DCB &AA, 0 ; Exit config mode ALIGN ; ; Simplified version of Configure82C710 programs SMC 37C665 only. ; Configure37C665 Entry "r0,r1" WritePSRc SVC_mode + I_bit + F_bit, r0 ; Disable FIQ and IRQ LDR r0, =CnTbase ; R0-> SMC 665 base address ; First try to configure the SMC665 MOV lr, #&55 STRB lr, [r0, #CRI711Off] ; Write &55 to CRI711 twice STRB lr, [r0, #CRI711Off] ; to enter configuration mode MOV lr, #&0D ; Check for SMC 665 STRB lr, [r0, #CRI711Off] LDRB lr, [r0, #CAP711Off] TEQ lr, #&65 MOVNE r1, #0 ;NE: not a SMC 665 this should never happen BNE %FT30 ;NE: on a RiscPC, Kryten or Stork ADR r1, ConfigSMCTable ; R1-> SMC 665 configuration data 20 LDRB lr, [r1], #1 ; get config index STRB lr, [r0, #CRI711Off] TEQ lr, #&AA ; end of table? LDRNEB lr, [r1], #1 ; if not then get config data STRNEB lr, [r0, #CAP711Off] ; and write it BNE %BT20 MOV r1, #IOST_37C665 30 MOV r0, #0 LDRB lr, [r0, #IOSystemType] BIC lr, lr, #IOST_COMBOMASK ORR lr, lr, r1 STRB lr, [r0, #IOSystemType] WritePSRc SVC_mode + I_bit, lr ; Restore IRQ/FIQ state EXIT ] ] ; :LNOT: HAL ; +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ ; ; ReadUniqueID - Read unique machine ID ; [ HAL ReadUniqueID Entry "r0-r3,r9,r12" AddressHAL ; Check for extended ID first MOV r0, #0 CallHAL HAL_ExtMachineID CMP r0, #0 BEQ %FT10 MOV r2, sp SUB r0, r2, r0 BIC r0, r0, #3 MOV sp, r0 Push "r0,r2" ; Remember old SP, buffer pointer CallHAL HAL_ExtMachineID Pull "r1" MOV r2, #0 MOV r3, #0 05 ; Construct the 7 byte machine ID using this simple algorithm: ; EOR each extended ID byte with the low byte of the 7 byte ID, ; then rotate left 7 bits LDRB r12, [r1], #1 EOR r2, r2, r12 AND r12, r3, #&fe0000 MOV r3, r3, LSL #7 ORR r3, r3, r2, LSR #25 MOV r2, r2, LSL #7 ORR r2, r2, r12, LSR #17 SUBS r0, r0, #1 BNE %BT05 LDR sp, [sp] ; Restore SP [ ZeroPage <> 0 LDR r0, =ZeroPage ] STR r2, [r0, #RawMachineID+0] STR r3, [r0, #RawMachineID+4] ; Abuse CheckCRC to calculate CRC byte BL CheckCRC LDR r0, =ZeroPage STRB r2, [r0, #RawMachineID+7] EXIT 10 CallHAL HAL_MachineID LDR r3, =ZeroPage STR r0, [r3, #RawMachineID+0] STR r1, [r3, #RawMachineID+4] ; now fall through and check the CRC below | defaultlatch * 20000-1 ; TMD 21-May-93: "-1" correction applied Tsyc * 5 ; time between frames - minimum of 1µs, so give it a bit more Trstl * 500 ; time reset pulse held low - minimum of 480µs, so give it a bit more Trsth * 500 ; time reset pulse held high - minimum of 480µs, so give it a bit more Tlow0 * 80 ; time for write0 low - min 60µs, max 120µs Tlow1 * 5 ; time for write1 low - min 1µs, max 15µs Tslot * 90 ; time for total read/write slot - min 60µs, max 120µs Trdlow * 5 ; time for read slot low before release - min 1µs, max 15µs Trddat * 3 ; time after read slot high before read it ASSERT Tslot-Tlow0 > Tsyc ASSERT Trdlow+Trddat < 15 ; Macro to set wire to a given state, and optionally count transitions (starting at low) while waiting for a given time MACRO SetWire $hilo, $time, $monstate, $count LCLS reg [ "$hilo"="LOW" reg SETS "r4" | ASSERT "$hilo"="HIGH" reg SETS "r5" ] [ ($time) = 0 STRB $reg, [r1, #IOCControl] ; set appropriate level on line | ASSERT ($time) < 32768 MOV r12, #(($time)*2):AND:&FF STRB r12, [r1, #Timer0LL] ; program low latch MOV r12, #(($time)*2):SHR:8 STRB r12, [r1, #Timer0LH] ; program high latch STRB $reg, [r1, #IOCControl] ; set appropriate level on line STRB r12, [r1, #Timer0GO] ; and start timer LDRB r12, [r1, #IOCIRQSTAA] ; dummy instruction to avoid bug in IOC LDRB r12, [r1, #IOCIRQSTAA] ; dummy instruction (repeated for FE) STRB r11, [r1, #IOCIRQCLRA] ; immediately clear IRQ bit [ "$monstate"<>"" MOV $monstate, #0 ] [ "$count"<>"" MOV $count, #0 ] 10 LDRB r12, [r1, #IOCIRQSTAA] TST r12, r11 [ "$count"<>"" ADDEQ $count, $count, #1 ] [ "$monstate"="" BEQ %BT10 ; not timed out, so just loop | BNE %FT30 ; timed out LDRB r12, [r1, #IOCControl] TST r12, #IOEB_unique_machine_ID_bit BEQ %BT10 ; if still low then loop to 10 ADD $monstate, $monstate, #1 ; increment number of transitions 20 LDRB r12, [r1, #IOCIRQSTAA] TST r12, r11 [ "$count"<>"" ADDEQ $count, $count, #1 ] BNE %FT30 ; timed out LDRB r12, [r1, #IOCControl] TST r12, #IOEB_unique_machine_ID_bit BNE %BT20 ; if still high then loop to 20 ADD $monstate, $monstate, #1 ; increment number of transitions B %BT10 30 ] ] MEND RestoreIOCState Entry STRB r3, [r1, #IOCControl] ; put back old value MOV r12, #defaultlatch :AND: &FF STRB r12, [r1, #Timer0LL] ; and restore old timer 0 latch values MOV r12, #defaultlatch :SHR: 8 STRB r12, [r1, #Timer0LH] STRB r12, [r1, #Timer0GO] WritePSRc SVC_mode + I_bit, lr ; restore old interrupt state EXIT SendResetPulse ROUT SetWire HIGH, Tsyc SetWire LOW, Trstl,,r6 SetWire HIGH, Trsth,r10 TEQ r6, #0 [ :DEF: DebugOneWireBus ADREQ r0, IOCBugHappenedError ] BEQ %FT05 CMP r10, #3 ; H-L-H is ok MOVEQ pc, lr ; V clear [ :DEF: DebugOneWireBus ADRHI r0, TooManyTransitionsError ; H-L-H-L... CMP r10, #2 ADREQ r0, NeverWentHighAgainError ; H-L CMP r10, #1 ADREQ r0, NeverWentLowError ; H ADRCC r0, NeverWentHighError ; stayed low permanently even though we released it ] 05 SETV MOV pc, lr [ :DEF: DebugOneWireBus NeverWentHighError = "Never went high", 0 NeverWentLowError = "Never went low", 0 NeverWentHighAgainError = "Never went high again", 0 TooManyTransitionsError = "Too many transitions", 0 IOCBugHappenedError = "IOC bug happened", 0 ALIGN ] SendCommandWord ROUT CLRV LDR r6, =&10F ; &0F is command word 10 MOVS r6, r6, LSR #1 MOVEQ pc, lr BCS SendOne SetWire LOW, Tlow0 SetWire HIGH, Tslot-Tlow0 B %BT10 SendOne SetWire LOW, Tlow1 SetWire HIGH, Tslot-Tlow1 B %BT10 GetAByte ROUT MOV r6, #&80 10 SetWire LOW, Trdlow SetWire HIGH, Trddat LDRB r10, [r1, #IOCControl] SetWire HIGH, Tslot-Trdlow-Trddat MOVS r10, r10, LSR #IOEB_ID_bit_number+1 ; move bit into carry MOVS r6, r6, RRX BCC %BT10 MOV r6, r6, LSR #24 MOV pc, lr ReadUniqueID Entry "r0-r12" MOV r0, #0 LDR r1, =IOC WritePSRc SVC_mode + I_bit + F_bit, r3 LDRB r3, [r0, #IOCControlSoftCopy] BIC r4, r3, #IOEB_unique_machine_ID_bit ; r4 is value to pull ID line low ORR r5, r3, #IOEB_unique_machine_ID_bit ; r5 is value to pull ID line high MOV r11, #timer0_bit BL SendResetPulse BVS ResetFailed BL SendCommandWord MOV r7, #-8 ; -no. of bytes to store = 8 bits type + 48 bits ID + 8 bits checksum 10 BL GetAByte STRB r6, [r7, #RawMachineID+8] ADDS r7, r7, #1 BNE %BT10 BL RestoreIOCState ] ; HAL BL CheckCRC BVS IDError EXIT [ :LNOT: HAL ResetFailed BL RestoreIOCState ] IDError DebugTX "Machine ID duff,zero substituted" MOV r0, #0 [ ZeroPage = 0 STR r0, [r0, #RawMachineID+0] ; indicate no ID by putting zero here STR r0, [r0, #RawMachineID+4] | LDR lr, =ZeroPage STR r0, [lr, #RawMachineID+0] ; indicate no ID by putting zero here STR r0, [lr, #RawMachineID+4] ] EXIT CheckCRC ROUT ; Note: artificial ID generator relies on the required CRC being returned in R2! LDR r1, =ZeroPage+RawMachineID ; pointer to current byte MOV r2, #0 MOV r3, #7 ; number of bytes to do 10 LDRB r4, [r1], #1 EOR r2, r2, r4 MOV r4, #8 ; number of bits to do 20 MOVS r2, r2, LSR #1 ; shift bit out into carry EORCS r2, r2, #&8C ; feedback carry into other bits SUBS r4, r4, #1 ; one less bit to do BNE %BT20 ; loop until done whole byte SUBS r3, r3, #1 ; one less byte to do BNE %BT10 ; loop until done all 7 bytes LDRB r4, [r1], #1 ; read CRC CMP r4, r2 ; if correct MOVEQ pc, lr ; exit (V clear) RETURNVS ; else exit indicating error LTORG END