mirror of
https://github.com/elliotnunn/mac-rom.git
synced 2025-01-01 11:29:27 +00:00
0ba83392d4
Resource forks are included only for .rsrc files. These are DeRezzed into their data fork. 'ckid' resources, from the Projector VCS, are not included. The Tools directory, containing mostly junk, is also excluded.
1662 lines
60 KiB
Plaintext
1662 lines
60 KiB
Plaintext
;
|
|
; File: ClockPRAMPrimitives.a
|
|
;
|
|
; Contains: low-level routines for accessing the realtime clock and parameter RAM
|
|
;
|
|
; Written by: Steve Christensen
|
|
;
|
|
; Copyright: © 1991-1993 by Apple Computer, Inc. All rights reserved.
|
|
;
|
|
; This file is used in these builds: ROM
|
|
;
|
|
; Change History (most recent first):
|
|
;
|
|
; <SM14> 12/13/93 PN Roll in KAOs and Horror changes to support Malcom and AJ
|
|
; machines.
|
|
; <SM13> 6/23/93 RC Go back to the Ludwig roll-in since PDM reallly needs this code
|
|
; from Cyclone - stay tuned for further fixes/changes from Kurt
|
|
; <SM12> 6/21/93 kc Roll out SM10 so that Quadras would wake up with the right time.
|
|
; <SM11> 6/14/93 kc Roll in Ludwig.
|
|
; <LW3> 1/14/93 GS (RBM) Fix the occassional hang at boot. When outputing from
|
|
; the VIA to Cuda, the interrupt will occur prior to the rising
|
|
; edge of CB1. Delay long enough for the edge to occur before
|
|
; acknowledging the shift register interrupt.
|
|
; <SM10> 5/20/93 RM EgretReadTime should not call _EgretDispatch. The lowmem global
|
|
; 'Time' is self correcting for Egret/Caboose/Cuda. The one second
|
|
; interrupt handler for Egret / Caboose / Cuda maintains the
|
|
; lowmem global 'Time' in a self correcting manner. This routine
|
|
; need only return the lowmem 'Time'. A call to _EgretDispatch
|
|
; will cause the AutoPoll timer within Egret / Caboose / Cuda to
|
|
; reset, resulting in deferring ADB auto polling. If an
|
|
; application calls ReadTime in a tight loop, ADB polling could be
|
|
; deferred indefinitely (making the mouse/keyboard appear to hang
|
|
; or respond intermittently). Since 'Time' is self correcting,
|
|
; the routine just returns 'Time' as the result placed into
|
|
; 'GetParam'.
|
|
; <SM9> 12/11/92 fau Backed out <SM8> cause it broke Cyclone.
|
|
; <SM8> 12/10/92 GS This patch bypasses the physical read of Egret Chip when reading
|
|
; the time. The time is automatically updated in the background,
|
|
; therefore the low mem global always reflects the accurate time.
|
|
; This patch pertains to all Egret Chip based firmware in
|
|
; Egret8/9, Caboose, and Cuda.
|
|
; <SM7> 11/30/92 SWC Rolled in the rest of the Horror changes, except for HcMac code.
|
|
; Added the build conditionals back in.
|
|
; <H6> 9/3/92 SWC Fixed a couple of bugs in the RTC-based routines (you mean no
|
|
; one had any problems until now?). Thanks, Helder.
|
|
; <SM6> 10/21/92 fau Cleaned up a MMCExists that was there from Pandora's timeframe:
|
|
; comment P5. Cuda works now, so we don't need to bypass stuff.
|
|
; <SM5> 6/26/92 GS (RBM) Rewrite of 'CudaRdXByte' & 'CudaWrXByte' to avoid looking
|
|
; for collisions. This is possible because a 'SyncAck' cycle
|
|
; occurs immediately after 'InitVias' and disables all
|
|
; asynchronous message sources within Cuda. These routines are
|
|
; being used after the Cuda manager is installed which will cause
|
|
; a loss of data from Cuda so I also made the routine dispatch to
|
|
; a trap based function using the Cuda manager once the Cuda
|
|
; manager is installed. This is not a particularly clean
|
|
; implementation and needs to be looked at further!!!
|
|
; <SM4> 5/28/92 kc Add ClkNoMem.
|
|
; <SM3> 5/25/92 RB Removed a forCyclone conditional. This WAS the last one left. A
|
|
; check for MMC is used instead to skip over RTCClkNoMem.
|
|
; <SM2> 5/21/92 RB Making changes for Cyclone. Original comments below: <P6>
|
|
; 2/10/92 GS Changed the PRAM IOPrimitives to use the full word of
|
|
; a register to set up the address to R/W. <P5> 1/30/92 RMP
|
|
; Temporarly disabled RTCClock read/write PRAM since Cuda isn't
|
|
; working (I know it doesn't make sense but it's 2:13 in the
|
|
; morning). <P4> 1/16/92 GS Updated the PRAMIO routine to use a
|
|
; full word as the address to Read/Write <P4> 1/16/92 GS Update
|
|
; VIA control bit var names. <P3> 1/16/92 GS Update to include the
|
|
; new file CudaMgr Clock and PRAM routines. <P2> 01/14/92 jmp
|
|
; (SWC,H5) Fixed PMgrWrXByte (registers were getting trashed).
|
|
; <1> 5/17/92 kc first checked in
|
|
; <SM0> 5/2/92 kc Roll in Horror. Comments follow:
|
|
; <H5> 1/13/92 SWC Fixed PMgrWrXByte (registers were getting trashed).
|
|
; <H4> 11/25/91 CCH Used standardized equates for VIA pins.
|
|
; <H3> 11/7/91 SWC Re-wrote the Egret code. It was working on an emulator but
|
|
; stopped working when the Mac was running stand-alone. Converted
|
|
; existing implementations to use a "standard" PramIO routine that
|
|
; calls implementation specific read/write PRAM byte routines so
|
|
; there's one less routine to write for new PRAM models that can
|
|
; use it.
|
|
; <H2> 11/6/91 SWC Re-wrote PMGRXPRamIO to fix problems too numerous to mention.
|
|
; <H1> 10/15/91 SWC Adding this file into the build.
|
|
|
|
|
|
PRINT OFF
|
|
LOAD 'StandardEqu.d'
|
|
INCLUDE 'HardwarePrivateEqu.a'
|
|
INCLUDE 'UniversalEqu.a'
|
|
IF hasPwrControls THEN
|
|
INCLUDE 'PowerPrivEqu.a'
|
|
ENDIF
|
|
IF hasEgret THEN
|
|
INCLUDE 'EgretEqu.a'
|
|
ENDIF
|
|
INCLUDE 'IOPrimitiveEqu.a'
|
|
PRINT ON
|
|
PRINT NOMDIR
|
|
MACHINE MC68020
|
|
|
|
|
|
|
|
ClockPRAMPrimitives PROC EXPORT
|
|
|
|
;ÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑÑ
|
|
;
|
|
; Clock/PRAM primitives vector table
|
|
;
|
|
; Tables pointed to by the universal ProductInfo record (ClockPRAMPtr) for low-level
|
|
; clock and PRAM routines dependent on the hardware. There should be a table for each
|
|
; supported ProductInfo. Note that if a machine has a combination of the "standard"
|
|
; PRAM and clock models, it should have a machine-specific table added below
|
|
|
|
|
|
IF hasVIAClock THEN
|
|
EXPORT RTCClockPRAM
|
|
DC.W 0 ; flags
|
|
DC.W (RTCTableEnd-RTCClockPRAM)/4 ; number of entries
|
|
RTCClockPRAM
|
|
DC.L RTCInitHW-RTCClockPRAM ; initialize the hardware (RAM-less)
|
|
DC.L RTCWrProtOff-RTCClockPRAM ; write-enable PRAM (RAM-less)
|
|
DC.L RTCWrProtOn-RTCClockPRAM ; write-protect PRAM (RAM-less)
|
|
DC.L RTCRdXByte-RTCClockPRAM ; read PRAM byte (RAM-less)
|
|
DC.L RTCWrXByte-RTCClockPRAM ; write PRAM byte (RAM-less)
|
|
DC.L StandardXPramIO-RTCClockPRAM ; read/write PRAM byte[s] (no traps) <H3>
|
|
DC.L RTCXParam-RTCClockPRAM ; read/write PRAM byte[s] (trap-based)
|
|
DC.L RTCReadTime-RTCClockPRAM ; read the clock (uses RAM)
|
|
DC.L RTCWriteTime-RTCClockPRAM ; write to the clock (uses RAM)
|
|
RTCTableEnd
|
|
ENDIF
|
|
|
|
|
|
IF hasPwrControls THEN
|
|
EXPORT PMGRClockPRAM
|
|
DC.W 0 ; flags
|
|
DC.W (PMGRTableEnd-PMGRClockPRAM)/4 ; number of entries
|
|
PMGRClockPRAM
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L PMGRRdXByte-PMGRClockPRAM ; read PRAM byte (RAM-less)
|
|
DC.L PMGRWrXByte-PMGRClockPRAM ; write PRAM byte (RAM-less)
|
|
DC.L StandardXPramIO-PMGRClockPRAM ; read/write PRAM byte[s] (no traps) <H3>
|
|
DC.L PMGRXParam-PMGRClockPRAM ; read/write PRAM byte[s] (trap-based)
|
|
DC.L PMGRReadTime-PMGRClockPRAM ; read the clock (uses RAM)
|
|
DC.L PMGRWriteTime-PMGRClockPRAM ; write to the clock (uses RAM)
|
|
PMGRTableEnd
|
|
ENDIF
|
|
|
|
|
|
IF hasEgret THEN
|
|
EXPORT EgretClockPRAM
|
|
DC.W 0 ; flags
|
|
DC.W (EgretTableEnd-EgretClockPRAM)/4 ; number of entries
|
|
EgretClockPRAM
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L EgretRdXByte-EgretClockPRAM ; read PRAM byte (RAM-less)
|
|
DC.L EgretWrXByte-EgretClockPRAM ; write PRAM byte (RAM-less)
|
|
DC.L StandardXPramIO-EgretClockPRAM ; read/write PRAM byte[s] (no traps) <H3>
|
|
DC.L EgretXParam-EgretClockPRAM ; read/write PRAM byte[s] (trap-based)
|
|
DC.L EgretReadTime-EgretClockPRAM ; read the clock (uses RAM)
|
|
DC.L EgretWriteTime-EgretClockPRAM ; write to the clock (uses RAM)
|
|
EgretTableEnd
|
|
|
|
EXPORT CudaClockPRAM
|
|
DC.W 0 ; flags
|
|
DC.W (CudaTableEnd-CudaClockPRAM)/4 ; number of entries
|
|
CudaClockPRAM
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L CudaRdXByte-CudaClockPRAM ; read PRAM byte (RAM-less)
|
|
DC.L CudaWrXByte-CudaClockPRAM ; write PRAM byte (RAM-less)
|
|
DC.L StandardXPramIO-CudaClockPRAM ; read/write PRAM byte[s] (no traps) <H5>
|
|
DC.L EgretXParam-CudaClockPRAM ; read/write PRAM byte[s] (trap-based)
|
|
DC.L EgretReadTime-CudaClockPRAM ; read the clock (uses RAM)
|
|
DC.L EgretWriteTime-CudaClockPRAM ; write to the clock (uses RAM)
|
|
CudaTableEnd
|
|
ENDIF
|
|
|
|
|
|
IF hasProtectedPRAM THEN
|
|
EXPORT NoPRAMClockPRAM
|
|
DC.W 0 ; flags
|
|
DC.W (NoPRAMTableEnd-NoPRAMClockPRAM)/4 ; number of entries
|
|
NoPRAMClockPRAM
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L 0 ; not used
|
|
DC.L NoPRAMRdXByte-NoPRAMClockPRAM ; read PRAM byte (RAM-less)
|
|
DC.L NoPRAMWrXByte-NoPRAMClockPRAM ; write PRAM byte (RAM-less)
|
|
DC.L NoPRAMXPramIO-NoPRAMClockPRAM ; read/write PRAM byte[s] (no traps)
|
|
DC.L NoPRAMXParam-NoPRAMClockPRAM ; read/write PRAM byte[s] (trap-based)
|
|
DC.L NoPRAMReadTime-NoPRAMClockPRAM ; read the clock (uses RAM)
|
|
DC.L NoPRAMWriteTime-NoPRAMClockPRAM ; write to the clock (uses RAM)
|
|
NoPRAMTableEnd
|
|
ENDIF
|
|
|
|
|
|
|
|
IF hasVIAClock | hasPwrControls | hasEgret | hasPowerMgr THEN ; <H3>
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: StandardXPramIO
|
|
;
|
|
; Inputs: A0 - pointer to table of base addresses
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A3 - pointer to PRAM I/O buffer
|
|
; D1 - flags indicating which external features are valid
|
|
; D3 - [r/w (bit 31)][number of PRAM bytes to read/write][starting PRAM address]
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: A0,A1,A2,A3,D0,D1,D2,D3
|
|
;
|
|
; Function: Reads/writes byte[s] from paramter RAM before traps are set up. This calls
|
|
; the implementation-specific RdXByte and WrXByte routines to read or write
|
|
; each PRAM byte.
|
|
;________________________________________________________________________________________
|
|
|
|
IMPORT ClkWPOff,RdXByte,WrXByte
|
|
|
|
StandardXPramIO
|
|
MOVEM.L D7/A4-A6,-(SP)
|
|
|
|
MOVEA.L DecoderInfo.VIA1Addr(A0),A2 ; get VIA 1's base address
|
|
|
|
BCLR #31,D3 ; read or write?
|
|
BEQ.S @StartRead ; -> read
|
|
|
|
;BigBSR6 ClkWPOff,A5 ; write-enable PRAM
|
|
BRA.S @StartWrite
|
|
|
|
@WriteLoop
|
|
MOVE SR,-(SP) ; save the status register
|
|
ORI #$0700,SR ; and disable all interrupts
|
|
BigBSR6 ClkWPOff,A5 ; write-enable PRAM
|
|
SWAP D3 ; get the PRAM address
|
|
MOVE.W D3,D1 ; move entire address into D1
|
|
MOVE.B (A3)+,D2 ; and a data byte from the caller's buffer
|
|
MOVE.L A3,D7 ; (save the buffer pointer)
|
|
BigBSR6 WrXByte,A5 ; write a byte to PRAM
|
|
MOVEA.L D7,A3 ; (restore the buffer pointer)
|
|
ADDQ.W #1,D3 ; next byte
|
|
MOVE (SP)+,SR ; restore the status register
|
|
bsr.l StdXPRAM2Patch ; call the patch
|
|
@StartWrite
|
|
SWAP D3 ; get the count
|
|
DBRA D3,@WriteLoop ; -> keep looping until done
|
|
MOVEA.L A1,A6 ; point to this machine's product info
|
|
ADDA.L ProductInfo.ClockPRAMPtr(A6),A6 ; and get the address of its clock/PRAM table
|
|
MOVE.L 4*cpWrProtOn(A6),D0 ; get the offset to the write-protect routine
|
|
BEQ.S @Done ; -> it's not supported for this implementation
|
|
;ADDA.L D0,A6 ; calculate the routine's address
|
|
;LEA @Done,A5 ; (simulate a BSR5)
|
|
;JMP (A6) ; and call it to turn on PRAM write-protect
|
|
bra.s @Done ; we're done
|
|
|
|
@ReadLoop SWAP D3 ; get the PRAM address
|
|
MOVE.W D3,D1 ; move entire address into D1
|
|
MOVE.L A3,D7 ; (save the buffer pointer)
|
|
BigBSR6 RdXByte,A5 ; read a byte from PRAM
|
|
MOVEA.L D7,A3 ; (restore the buffer pointer)
|
|
MOVE.B D1,(A3)+ ; and save it in the caller's buffer
|
|
ADDQ.W #1,D3 ; next byte
|
|
@StartRead SWAP D3 ; get the count
|
|
DBRA D3,@ReadLoop ; -> keep looping until done
|
|
|
|
@Done MOVEM.L (SP)+,D7/A4-A6
|
|
RTS
|
|
ENDIF
|
|
|
|
|
|
|
|
IF hasVIAClock THEN
|
|
;¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥ RTC ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥
|
|
|
|
IMPORT ReadTime
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCReadTime
|
|
;
|
|
; Inputs: none
|
|
;
|
|
; Outputs: D0 - result code
|
|
; "GetParam" contains the 32-bit time value read from the clock
|
|
;
|
|
; Trashes: A0,A2,A3,D1,D2,D4,D5, 8 bytes at "GetParam"
|
|
;
|
|
; Function: reads the time from the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
RTCReadTime MOVEM.L A5/A6,-(SP)
|
|
MOVEA.L VIA,A2 ; get the VIA 1 base address for RTCClkNoMem
|
|
|
|
MOVEQ #2-1,D4 ; try reading twice in case the time rolls over
|
|
@RetryLoop MOVEQ #$9D-256,D5 ; command to read the high byte of the time
|
|
LEA GetParam,A0 ; use clock utility storage
|
|
MOVEA.L A0,A3 ; and save a copy for later
|
|
|
|
@ReadLoop MOVE.B D5,D1 ; get the command into D1
|
|
BSR5 RTCClkNoMem ; read a byte of the time
|
|
MOVE.B D1,(A0)+ ; and save it
|
|
SUBQ.B #4,D5 ; next byte
|
|
BMI.S @ReadLoop ; -> keep looping until all 8 bytes are read
|
|
|
|
MOVE.L (A3)+,D0 ; do the two reads of the time match?
|
|
SUB.L (A3)+,D0
|
|
DBEQ D4,@RetryLoop ; -> keep looping if not, or for two tries
|
|
BEQ.S @Done ; -> no error
|
|
|
|
MOVEQ #ClkRdErr,D0 ; note the problem
|
|
@Done MOVEM.L (SP)+,A5/A6
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCWriteTime
|
|
;
|
|
; Inputs: "Time" contains the 32-bit value to write to the clock
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0,D1,D2,D3,D4,D5
|
|
;
|
|
; Function: writes a new time value to the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
RTCWriteTime
|
|
MOVEM.L D6/A2/A5/A6,-(SP)
|
|
MOVEA.L VIA,A2 ; get the VIA 1 base address for RTCClkNoMem
|
|
|
|
BSR5 RTCWrProtOff ; write-enable the clock chip
|
|
|
|
MOVEQ #2-1,D5 ; give it two tries
|
|
MOVE.L Time,D3 ; get the time to write
|
|
|
|
@RetryLoop MOVEQ #$01,D6 ; command to write the low byte of the time
|
|
MOVEQ #4-1,D4
|
|
@WriteLoop MOVE.L D3,D1 ; put the byte to write into the upper word
|
|
SWAP D1
|
|
MOVE.B D6,D1 ; and the command in the lower word
|
|
BSR5 RTCClkNoMem ; write a byte of the time
|
|
ADDQ.B #4,D6 ; address the next register
|
|
ROR.L #8,D3 ; get the next byte
|
|
DBRA D4,@WriteLoop
|
|
|
|
BIGJSR ReadTime,A0 ; read the time to verify it was written correctly
|
|
BNE.S @Done
|
|
CMP.L Time,D3 ; do they match?
|
|
DBEQ D5,@RetryLoop ; -> keep looping if not, or for two tries
|
|
BEQ.S @Done ; -> they match so we're done
|
|
MOVEQ #ClkWrErr,D0 ; note the error
|
|
|
|
@Done MOVE.L D0,D3 ; save the result code
|
|
BSR5 RTCWrProtOn ; turn write-protect back on <H6>
|
|
MOVE.L D3,D0 ; restore the result code
|
|
MOVEM.L (SP)+,D6/A2/A5/A6
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCXParam
|
|
;
|
|
; Inputs: A0 - pointer to user's buffer
|
|
; D0 - [number of bytes to transfer][starting PRAM address]
|
|
; D1 - bit 12: 1=read, 0=write
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0,A1,A2,A3,D1,D2,D3
|
|
;
|
|
; Function: reads/writes byte[s] from RTC-based paramter RAM
|
|
;________________________________________________________________________________________
|
|
|
|
RTCXParam MOVEA.L A0,A3 ; get the pointer to the caller's buffer
|
|
MOVEA.L UnivInfoPtr,A1 ; point to the ProductInfo table <H3>
|
|
MOVEA.L A1,A0 ; and to the decoder table <H3>
|
|
ADDA.L ProductInfo.DecoderInfoPtr(A0),A0
|
|
MOVE.L D0,D3 ; get the starting PRAM address and count
|
|
BTST #12,D1 ; is it a read or write?
|
|
BNE.S @ItsRead
|
|
BSET #31,D3 ; mark it as a write
|
|
|
|
@ItsRead BSR.S StandardXPramIO ; go transfer the PRAM bytes <H3>
|
|
|
|
MOVEQ #0,D0 ; always return "no error"
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCRdXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to read
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: D1 - byte read from PRAM
|
|
;
|
|
; Trashes: D0,D2,A5,A6
|
|
;
|
|
; Function: reads a byte from RTC-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
RTCRdXByte ANDI.W #$00FF,D1 ; [oooo oooo 7654 3210]
|
|
ASL.W #3,D1 ; [oooo o765 4321 0ooo]
|
|
ROR.B #1,D1 ; [oooo o765 o432 10oo]
|
|
ROR.W #8,D1 ; [o432 10oo oooo o765]
|
|
ORI.B #$B8,D1 ; [o432 10oo ioii i765]
|
|
|
|
BSR5 RTCClkNoMem ; read the PRAM byte
|
|
|
|
ANDI.W #$00FF,D1 ; zero-extend the byte
|
|
JMP (A3)
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCWrXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to write
|
|
; D2 - byte to write to PRAM
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A5,A6
|
|
;
|
|
; Function: writes a byte to RTC-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
RTCWrXByte SWAP D1 ; copy the byte to write
|
|
MOVE.B D2,D1 ; to the upper word of D1
|
|
SWAP D1
|
|
|
|
ANDI.W #$00FF,D1 ; [oooo oooo 7654 3210]
|
|
ASL.W #3,D1 ; [oooo o765 4321 0ooo]
|
|
ROR.B #1,D1 ; [oooo o765 o432 10oo]
|
|
ROR.W #8,D1 ; [o432 10oo oooo o765]
|
|
ORI.B #$38,D1 ; [o432 10oo ooii i765]
|
|
|
|
MOVEA.L A3,A5 ; (fake out the return address)
|
|
BRA.S RTCClkNoMem ; write the PRAM byte
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCInitHW
|
|
;
|
|
; Inputs: A1 - pointer to ProductInfo record for this machine
|
|
; A6 - return address (BSR6)
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A0,A2,A5,A6
|
|
;
|
|
; Function: initializes the RTC by taking it out of test mode
|
|
;________________________________________________________________________________________
|
|
|
|
RTCInitHW MOVEA.L A6,A0 ; save the return address
|
|
MOVEA.L A1,A2 ; point to the decoder info
|
|
ADDA.L ProductInfo.DecoderInfoPtr(A2),A2
|
|
MOVEA.L DecoderInfo.VIA1Addr(A2),A2 ; get VIA 1's base address <H6>
|
|
|
|
BSR5 RTCWrProtOff ; write-enable the clock chip
|
|
|
|
MOVEQ #(0<<16)+($31<<0),D1 ; write zero to test bit in clock chip
|
|
BSR5 RTCClkNoMem
|
|
|
|
MOVEA.L A0,A5 ; restore the return address
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCWrProtOn [RTC utility routine]
|
|
;
|
|
; Inputs: A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A5 - return address (BSR5)
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A6
|
|
;
|
|
; Function: write-protects RTC-based PRAM when we're done modifying it
|
|
;________________________________________________________________________________________
|
|
|
|
RTCWrProtOn MOVE.L #($D5<<16)+($35<<0),D1 ; use 'set write protect' command
|
|
BRA.S RTCClkNoMem ; write it to the clock chip
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCWrProtOff [RTC utility routine]
|
|
;
|
|
; Inputs: A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A5 - return address (BSR5)
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A6
|
|
;
|
|
; Function: write-enables RTC-based PRAM so we can write bytes to it
|
|
;________________________________________________________________________________________
|
|
|
|
RTCWrProtOff
|
|
MOVE.L #($55<<16)+($35<<0),D1 ; use 'reset write protect' command
|
|
* BRA.S RTCClkNoMem ; write it to the clock chip
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: RTCClkNoMem [RTC utility routine]
|
|
;
|
|
; Inputs: D1 - [x] [byte to write] [<extended command>] [command (bit 7: 1=read, 0=write)]
|
|
; A2 - VIA1 base address
|
|
; A5 - return address (BSR5)
|
|
;
|
|
; Outputs: D1 - data byte read
|
|
;
|
|
; Trashes: D0,D1,D2,A6
|
|
;
|
|
; Function: reads or writes one 8-bit clock register
|
|
;________________________________________________________________________________________
|
|
|
|
RTCClkNoMem MOVE SR,D2 ; save the SR
|
|
ORI #HiIntMask,SR ; and disable interrupts
|
|
SWAP D2
|
|
|
|
MOVE.B D1,D2 ; save a copy of the command
|
|
|
|
BCLR #vRTCEnb,vBufB(A2) ; enable the clock
|
|
|
|
MOVEQ #$78,D0 ; mask off the code bits
|
|
AND.B D1,D0
|
|
CMPI.B #$38,D0 ; is this an extended command?
|
|
BNE.S @NotExtended ; -> nope
|
|
|
|
BSR6 SendToClk ; send the first command byte
|
|
LSR.W #8,D1 ; move the second command byte into position
|
|
|
|
@NotExtended
|
|
TST.B D2 ; is this a read or write command?
|
|
BMI.S @ItsRead ; -> read
|
|
|
|
BSR6 SendToClk ; send the command byte
|
|
SWAP D1 ; get the data byte
|
|
BSR6 SendToClk ; and send it
|
|
BRA.S @Done
|
|
|
|
@ItsRead BSR6 SendToClk ; send the command byte
|
|
BSR6 ReadFrClk ; and read the data byte
|
|
|
|
@Done BSET #vRTCEnb,vBufB(A2) ; disable the clock
|
|
SWAP D2
|
|
MOVE D2,SR ; re-enable interrupts
|
|
RTS5
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: SendToClk [RTC utility routine]
|
|
;
|
|
; Inputs: D1 - command/data byte to write to RTC
|
|
; A2 - VIA1 base address
|
|
; A6 - return address (BSR6)
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1
|
|
;
|
|
; Function: writes a command or data byte to the RTC (assumes interrupts masked to $2700)
|
|
;________________________________________________________________________________________
|
|
|
|
SendToClk MOVEQ #8-1,D0 ; shifting out 8 bits
|
|
|
|
@ShiftLoop SWAP D0 ; save the count
|
|
MOVE.B vBufB(A2),D0 ; get the VIA register's contents
|
|
ROR.B #1,D0 ; shift the bits to position the RTC data bit (bit 0)
|
|
ADDX.B D1,D1 ; shift the next data bit into the extend bit
|
|
ADDX.B D0,D0 ; shift the data bit from the extend bit to the RTC data bit
|
|
MOVE.B D0,VBufB(A2) ; stuff it back into the VIA register
|
|
BCLR #vRTCClk,vBufB(A2) ; and clock the bit out
|
|
BSET #vRTCClk,vBufB(A2)
|
|
SWAP D0 ; get the bit counter back
|
|
DBRA D0,@ShiftLoop ; -> keep looping until all bits are shifted out
|
|
RTS6
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: ReadFrClk [RTC utility routine]
|
|
;
|
|
; Inputs: A2 - VIA1 base address
|
|
; A6 - return address (BSR6)
|
|
;
|
|
; Outputs: D1 - data byte read from RTC
|
|
;
|
|
; Trashes: D0
|
|
;
|
|
; Function: reads a data byte from the RTC (assumes interrupts masked to $2700)
|
|
;________________________________________________________________________________________
|
|
|
|
ReadFrClk BCLR #vRTCData,vDirB(A2) ; set the RTC data bit's direction to input
|
|
MOVEQ #1,D1 ; initialize the result with a flag bit
|
|
|
|
@ShiftLoop BCLR #vRTCClk,vBufB(A2) ; clock the next bit in
|
|
BSET #vRTCClk,vBufB(A2)
|
|
MOVE.B vBufB(A2),D0 ; read the register
|
|
LSR.B #vRTCData+1,D0 ; shift the RTC data bit into the carry/X
|
|
ADDX.B D1,D1 ; shift in the new bit towards the msb
|
|
BCC.S @ShiftLoop ; -> keep looping until the flag bit shifts out
|
|
|
|
BSET #vRTCData,vDirB(A2) ; restore the RTC data bit's direction to output
|
|
RTS6
|
|
|
|
|
|
ENDIF ; {hasVIAClock}
|
|
|
|
|
|
|
|
IF hasPwrControls THEN
|
|
;¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥ Power Manager ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥
|
|
WITH PMCommandRec
|
|
|
|
IMPORT USTPmgrSendCommand, USTPMgrSendByte, USTPMGRRecvByte
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: PMGRReadTime
|
|
;
|
|
; Inputs: none
|
|
;
|
|
; Outputs: D0 - result code
|
|
; "GetParam" contains the 32-bit time value read from the clock
|
|
;
|
|
; Trashes: A0
|
|
;
|
|
; Function: reads the time from the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
PMGRReadTime
|
|
PEA GetParam ; pmRBuffer
|
|
MOVE.L (SP),-(SP) ; pmSBuffer (not used)
|
|
CLR.W -(SP) ; pmLength = 0
|
|
MOVE.W #TimeRead,-(SP) ; pmCommand = read time
|
|
MOVE.L SP,A0 ; point to the parameter block
|
|
_PmgrOp ; read the time
|
|
LEA pmRBuffer+4(SP),SP ; clean up the stack
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: PMGRWriteTime
|
|
;
|
|
; Inputs: "Time" contains 32-bit quantity to write to clock
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0
|
|
;
|
|
; Function: writes a new time value to the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
PMGRWriteTime
|
|
PEA Time ; pmRBuffer (not used)
|
|
MOVE.L (SP),-(SP) ; pmSBuffer
|
|
MOVE.W #4,-(SP) ; pmLength = 4 bytes of time
|
|
MOVE.W #TimeWrite,-(SP) ; pmCommand = write time
|
|
MOVE.L SP,A0 ; point to the parameter block
|
|
_PmgrOp ; read the time
|
|
LEA pmRBuffer+4(SP),SP ; clean up the stack
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: PMGRXParam
|
|
;
|
|
; Inputs: A0 - pointer to user's buffer
|
|
; D0 - [number of bytes to transfer][starting PRAM address]
|
|
; D1 - bit 12: 1=read, 0=write
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0,A1,A2,D1,D2,D3
|
|
;
|
|
; Function: reads/writes byte[s] from PMGR-based parameter RAM
|
|
;________________________________________________________________________________________
|
|
|
|
PMGRXParam MOVEA.L SP,A2 ; point to the current top of stack
|
|
MOVE.L D0,D2 ;
|
|
SWAP D2 ; D2.L=[starting PRAM address][count]
|
|
ADDQ.W #2+1,D2 ; make space on the stack for a buffer and 2 parameter bytes
|
|
BCLR #0,D2 ; and round up to the next word
|
|
SUBA.W D2,SP
|
|
MOVEA.L SP,A1
|
|
|
|
MOVE.L A0,-(SP) ; pmRBuffer
|
|
MOVE.L A1,-(SP) ; pmSBuffer
|
|
|
|
MOVE.B D0,(A1)+ ; put the PRAM address
|
|
SWAP D0
|
|
MOVE.B D0,(A1)+ ; and count at the start of the send buffer
|
|
|
|
MOVEQ #xPramRead,D3 ; assume it's a read
|
|
MOVEQ #2,D2 ;
|
|
BTST #12,D1 ; is this a write command?
|
|
BNE.S @NoWrBuffer ; -> no
|
|
MOVEQ #xPramWrite,D3
|
|
ADD.W D0,D2 ; adjust the count to include the bytes to be written
|
|
BRA.S @StartCopy
|
|
@CopyLoop MOVE.B (A0)+,(A1)+ ; copy the write buffer into the stack buffer
|
|
@StartCopy DBRA D0,@CopyLoop
|
|
|
|
@NoWrBuffer MOVE.W D2,-(SP) ; pmLength
|
|
MOVE.W D3,-(SP) ; pmCommand
|
|
MOVEA.L SP,A0
|
|
_PMgrOp ; have the PMGR read/write PRAM
|
|
|
|
MOVEA.L A2,SP ; restore the stack pointer
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: PMGRWrXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to write
|
|
; D2 - byte to write to PRAM
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: A0,A3,A4,A5,A6,D0,D1,D2
|
|
;
|
|
; Function: writes a byte to PMGR-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
PMGRWrXByte MOVEA.L A3,A4 ; save the return address
|
|
|
|
EXG D3,D1 ; save D3 in D1, put PRAM address in the upper word of D3
|
|
SWAP D3
|
|
EXG D4,D2 ; save D4 in D2, put data byte in the upper word of D4
|
|
SWAP D4
|
|
MOVE.L D6,D0 ; save D6
|
|
|
|
MOVE.W #(xPramWrite<<0)|\ ; pmCommand
|
|
(-1<<8),D3 ; pmLength = -1 (just send command byte)
|
|
|
|
MOVEA.L A1,A0 ; point to the decoder table
|
|
ADDA.L ProductInfo.DecoderInfoPtr(A0),A0
|
|
BigBSR6 USTPmgrSendCommand,A5 ; send the command to the PMGR
|
|
BNE.S @PMgrDone ; -> error
|
|
|
|
MOVE.B #3,D3 ; send pmLength = 3 (address, count, data)
|
|
BigBSR5 USTPMgrSendByte,A6
|
|
BNE.S @PMgrDone ; -> error
|
|
|
|
SWAP D3 ; send the PRAM address
|
|
BigBSR5 USTPMgrSendByte,A6
|
|
BNE.S @PMgrDone ; -> error
|
|
|
|
MOVEQ #1,D3 ; send the PRAM byte count
|
|
BigBSR5 USTPMgrSendByte,A6
|
|
BNE.S @PMgrDone ; -> error
|
|
|
|
MOVE.L D4,D3 ; send the PRAM data byte
|
|
SWAP D3
|
|
BigBSR5 USTPMgrSendByte,A6
|
|
|
|
@PMgrDone MOVE.L D1,D3 ; restore registers
|
|
MOVE.L D2,D4
|
|
MOVE.L D0,D6
|
|
JMP (A4)
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: PMGRRdXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to read
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: D1 - byte read from PRAM
|
|
;
|
|
; Trashes: A0,A3,A4,A5,A6,D0,D2
|
|
;
|
|
; Function: reads a byte from PMGR-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
PMGRRdXByte MOVEA.L A3,A4 ; save the return address
|
|
|
|
EXG D3,D1 ; save D3 in D1, put command/data bytes in D3
|
|
MOVE.L D4,D0 ; save other registers
|
|
MOVE.L D6,D2
|
|
|
|
ANDI.L #$000000FF,D3 ; mask the PRAM address and build the command:
|
|
ORI.L #(xPramRead<<16)|\ ; pmCommand
|
|
(2<<24)|\ ; pmLength
|
|
(1<<8),D3 ; count = 1 byte
|
|
SWAP D3 ; D1.L=[count][addr][pmLength][pmCommand]
|
|
|
|
MOVEA.L A1,A0 ; point to the decoder table
|
|
ADDA.L ProductInfo.DecoderInfoPtr(A0),A0
|
|
BigBSR6 USTPmgrSendCommand,A5 ; send the command to the PMGR
|
|
BNE.S @PMgrError ; -> error
|
|
|
|
BigBSR5 USTPMGRRecvByte,A6 ; read a byte back
|
|
BNE.S @PMgrError ; -> error
|
|
CMPI.B #xPramRead,D3 ; is it the reply byte?
|
|
BNE.S @NoReplyByte ; -> no, new protocol
|
|
|
|
BigBSR5 USTPMGRRecvByte,A6 ; read the count byte
|
|
BNE.S @PMgrError ; -> error
|
|
@NoReplyByte
|
|
BigBSR5 USTPMGRRecvByte,A6 ; read the PRAM byte
|
|
BEQ.S @PMgrDone ; -> got it
|
|
|
|
@PMgrError MOVEQ #0,D3 ; return a zero if an error occurs
|
|
@PMgrDone EXG D1,D3 ; restore D3, return PRAM byte in D1
|
|
MOVE.L D0,D4 ; restore other registers
|
|
MOVE.L D2,D6
|
|
|
|
TST.B D1 ; set up the CCR if anybody cares
|
|
JMP (A4)
|
|
|
|
|
|
ENDWITH ; {PMgrRec}
|
|
ENDIF ; {hasPwrControls}
|
|
|
|
|
|
|
|
IF hasEgret THEN
|
|
;¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥ Egret ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥
|
|
WITH EgretPB
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretReadTime
|
|
;
|
|
; The one second interrupt handler for Egret/Caboose/Cuda maintains the lowmem global <sm10>
|
|
; 'Time' in a self correcting manner. This routine need only return the lowmem 'Time'.
|
|
; A call to _EgretDispatch will cause the AutoPoll timer within Egret/Caboose/Cuda to
|
|
; reset, resulting in deferring ADB auto polling. If an application calls ReadTime
|
|
; in a tight loop, ADB polling could be deferred indefinitely (making the mouse/keyboard
|
|
; appear to hang or respond intermittently). Since 'Time' is self correcting, the
|
|
; routine just returns 'Time' as the result placed into 'GetParam'.
|
|
;
|
|
; Inputs: none
|
|
;
|
|
; Outputs: D0 - result code
|
|
; "GetParam" contains the 32-bit time value read from the clock
|
|
;
|
|
; Trashes: A0
|
|
;
|
|
; Function: reads the time from the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
EgretReadTime
|
|
clr.l d0 ; result code = noErr <sm10>
|
|
move.l Time,GetParam ; return current time <sm10>
|
|
; CLR.L -(SP) ; pbCompletion = nil <sm12>
|
|
; LEA pbCmdType-pbCompletion(SP),SP ; <sm12>
|
|
; MOVE.W #(pseudopkt<<8)+\ ; pbCmdType = pseudo type <sm12>
|
|
; (RdTime<<0),(SP) ; pbCmd = read time <sm12>
|
|
; MOVEA.L SP,A0 ; point to the parameter block <sm12>
|
|
; _EgretDispatch ; call Egret to read the time <sm12>
|
|
; MOVE.L pbParam(A0),GetParam ; move the time into a temporary buffer <sm12>
|
|
; LEA EgretPBSize(SP),SP ; clean up the stack <sm12>
|
|
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretWriteTime
|
|
;
|
|
; Inputs: "Time" contains 32-bit quantity to write to clock
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0
|
|
;
|
|
; Function: writes a new time value to the realtime clock
|
|
;________________________________________________________________________________________
|
|
|
|
EgretWriteTime
|
|
CLR.L -(SP) ; pbCompletion = nil
|
|
LEA pbParam-pbCompletion(SP),SP
|
|
MOVE.L Time,(SP) ; pbParam = current time
|
|
|
|
MOVE.W #(pseudopkt<<8)+\ ; pbCmdType = pseudo type
|
|
+(WrTime<<0),-(SP) ; pbCmd = write time
|
|
MOVEA.L SP,A0 ; point to the parameter block
|
|
_EgretDispatch ; call Egret to read the time
|
|
LEA EgretPBSize(SP),SP ; clean up the stack
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretXParam
|
|
;
|
|
; Inputs: A0 - pointer to user's buffer
|
|
; D0 - [number of bytes to transfer][starting PRAM address]
|
|
; D1 - bit 12: 1=read, 0=write
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0
|
|
;
|
|
; Function: reads/writes byte[s] from Egret-based parameter RAM
|
|
;________________________________________________________________________________________
|
|
|
|
EgretXParam CLR.L -(SP) ; pbCompletion = nil
|
|
SUBQ.W #pbCompletion-pbBufPtr,SP
|
|
MOVE.L A0,(SP) ; pbBufPtr = pointer to user's buffer
|
|
SWAP D0
|
|
MOVE.W D0,-(SP) ; pbByteCnt = number of PRAM bytes to read/write
|
|
SUBQ.W #pbByteCnt-pbParam,SP
|
|
SWAP D0
|
|
ADDI.W #$0100,D0 ;¥ convert PRAM address to 6805 address (temporary?)
|
|
MOVE.W D0,(SP) ; pbParam = starting PRAM address
|
|
MOVE.W #(pseudopkt<<8)+\ ; pbCmdType = pseudo type
|
|
(Rd6805Addr<<0),-(SP) ; pbCmd = read PRAM
|
|
BTST #12,D1 ; is this a write command?
|
|
BNE.S @NotWrite ; -> no
|
|
MOVE.W #(pseudopkt<<8)+\ ; pbCmdType = pseudo type
|
|
(Wr6805Addr<<0),(SP) ; pbCmd = write PRAM
|
|
@NotWrite MOVEA.L SP,A0 ; point to the parameter block
|
|
_EgretDispatch ; call Egret to read/write PRAM
|
|
LEA EgretPBSize(SP),SP ; clean up the stack
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretWrXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to write
|
|
; D2 - byte to write to PRAM
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A5,A6
|
|
;
|
|
; Function: writes a byte to Egret-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
EgretWrXByte
|
|
SWAP D1
|
|
MOVE.B D2,D1 ; save the byte to send
|
|
SWAP D1
|
|
BSET #15,D1 ; mark this as a write
|
|
; fall into EgretRdXByte
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretRdXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to read
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: D1 - byte read from PRAM
|
|
;
|
|
; Trashes: D0,D2,A5,A6
|
|
;
|
|
; Function: reads a byte from Egret-based PRAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
EgretRdXByte
|
|
BCLR #vSysSesbit,vBufB(A2) ; pending transaction?
|
|
BEQ.S @NotPending ; -> no
|
|
BSR5 EgretDelay ; wait a bit
|
|
BCLR #vViaFullBit,vBufB(A2) ; de-assert VIA full
|
|
BSR5 EgretDelay ; wait some more
|
|
BRA.S @NotPending
|
|
|
|
@Abort BSR6 EgretReadByte ; too many bytes being returned so
|
|
BEQ.S @Abort ; keep throwing them out until there are no more,
|
|
BRA.S @Reset ; then start over
|
|
|
|
@DisposePacket
|
|
BSR5 EgretDelay ; wait for the interrupt to occur
|
|
TST.B vSR(A2) ; then read the ATTN byte to clear the interrupt
|
|
|
|
@Reset BCLR #vSysSesbit,vBufB(A2) ; turn off SYS session
|
|
BCLR #vViaFullBit,vBufB(A2) ; and de-assert VIA full
|
|
BCLR #SRdir,vACR(A2) ; define shift direction = FROM Egret
|
|
BSR5 EgretDelay ; wait
|
|
BRA.S @CheckXCVR
|
|
|
|
@ThrowAway BSET #vSysSesbit,vBufB(A2) ; set SYS session
|
|
BSR6 EgretReadByte ; read a byte to throw away
|
|
@CheckXCVR BTST #vXcvrSesBit,vBufB(A2) ; is XCVR session asserted (active low)?
|
|
BEQ.S @ThrowAway ; -> yes, throw away the byte
|
|
BSR5 EgretDelay ; wait
|
|
|
|
@NotPending ORI.B #ACRMode++(1<<SRdir),vACR(A2) ; set VIA shift register to shift out with external clock
|
|
BTST #vXcvrSesBit,vBufB(A2) ; is XCVR session asserted (active low)?
|
|
BEQ.S @DisposePacket ; -> yes, dispose of the packet
|
|
|
|
BSET #vSysSesbit,vBufB(A2) ; assert sys session signal
|
|
MOVEQ #pseudopkt,D0 ; packet type
|
|
BSR6 EgretSendByte
|
|
BMI.S @Reset ; -> Egret's hung, so try to reset
|
|
|
|
BTST #vXcvrSesBit,vBufB(A2) ; is XCVR session asserted (active low)?
|
|
BEQ.S @Reset ; -> yes, throw out the packet
|
|
|
|
MOVEQ #rdPram,D0 ; assume we're doing a read
|
|
TST.W D1 ; are we?
|
|
BPL.S @SendCmd
|
|
MOVEQ #wrPram,D0 ; no, we're writing
|
|
@SendCmd BSR6 EgretSendByte ; send the command byte
|
|
MOVEQ #0,D0
|
|
BSR6 EgretSendByte ; send the MSB of the PRAM address
|
|
MOVE.B D1,D0
|
|
BSR6 EgretSendByte ; send the LSB of the PRAM address
|
|
TST.W D1 ; are we writing?
|
|
BPL.S @SendDone
|
|
MOVE.L D1,D0 ; yes, get the data byte
|
|
SWAP D0
|
|
BSR6 EgretSendByte ; and send it
|
|
|
|
@SendDone BCLR #SRdir,vACR(A2) ; define shift direction = FROM Egret
|
|
BCLR #vSysSesbit,vBufB(A2) ; turn off SYS session
|
|
|
|
@WaitForIRQ BTST #ifSR,vIFR(A2) ; wait for shift to complete
|
|
BEQ.S @WaitForIRQ
|
|
BTST #vXcvrSesBit,vBufB(A2) ; is XCVR session asserted (active low)?
|
|
BNE.S @WaitForIRQ ; -> no, keep polling until it is
|
|
|
|
TST.B vSR(A2) ; read the ATTN byte to clear the interrupt
|
|
BSR5 EgretDelay ; wait
|
|
BSET #vSysSesbit,vBufB(A2) ; assert sys session again
|
|
|
|
BSR6 EgretReadByte ; read the response packet type
|
|
BNE.S @Reset ; -> XCVR session turned off early, so try again
|
|
|
|
BSR6 EgretReadByte ; read the response packet flag
|
|
BNE.S @Reset ; -> XCVR session turned off early, so try again
|
|
|
|
TST.W D1 ; is this a read or write?
|
|
BPL.S @FinishRead ; -> read
|
|
|
|
BSR6 EgretReadByte ; read the "command was" byte
|
|
BEQ.S @Abort ; -> XCVR session is still on, so try again
|
|
CMPI.B #wrPram,D0 ; is this the correct packet?
|
|
BNE.S @Reset ; -> no, try again
|
|
|
|
BCLR #vSysSesbit,vBufB(A2) ; turn off SYS session
|
|
BCLR #vViaFullBit,vBufB(A2) ; and de-assert VIA full
|
|
BSR5 EgretDelay ; delay to let Egret see the lines change state
|
|
JMP (A3)
|
|
|
|
|
|
@FinishRead BSR6 EgretReadByte ; read the "command was" byte
|
|
BNE.S @Reset ; -> XCVR session turned off early, so try again
|
|
SUBQ.B #rdPram,D0 ; is this the correct packet?
|
|
BNE.S @Reset ; -> no, try again
|
|
|
|
BSR6 EgretReadByte ; read the PRAM data byte
|
|
BCLR #vSysSesbit,vBufB(A2) ; turn off SYS session
|
|
|
|
@Wait4XCVR BTST #vXcvrSesBit,vBufB(A2) ; wait for XCVR session to turn off
|
|
BEQ.S @Wait4XCVR
|
|
BCLR #vViaFullBit,vBufB(A2) ; de-assert VIA full
|
|
|
|
BSR5 EgretDelay ; delay to let Egret see the lines change state
|
|
MOVE.B D0,D1 ; return the byte in D1
|
|
JMP (A3)
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretSendByte [Egret utility routine]
|
|
;
|
|
; Inputs: D0 - byte to send to Egret
|
|
; A2 - VIA1 base address
|
|
; A6 - return address (BSR6)
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D2,A5
|
|
;
|
|
; Function: sends a byte to Egret
|
|
;________________________________________________________________________________________
|
|
|
|
EgretSendByte
|
|
MOVEA.L A6,A5 ; save the return address
|
|
MOVE.B D0,vSR(A2) ; load byte into shift register
|
|
BSET #vViaFullBit,vBufB(A2) ; indicate shift register is full
|
|
MOVE.W #32767,D0 ; set a 42msec timeout
|
|
@Wait4Shift BTST #ifSR,vIFR(A2) ; wait for shift to complete
|
|
DBNE D0,@Wait4Shift
|
|
BEQ.S EgretSendTO ; -> timed out
|
|
TST.B vSR(A2) ; clear the interrupt
|
|
BCLR #vViaFullBit,vBufB(A2) ; then negate VIA full bit
|
|
|
|
; delay a bit so Egret can keep up...
|
|
|
|
EgretDelay MOVEQ #104,D2 ; delay 125µs minimum (we actually delay 134µsec)
|
|
@wait TST.B (A2) ; sync to VIA clock (about 1.28µsec/access)
|
|
DBRA D2,@wait ; delay at least 100 us (very rough)
|
|
EgretSendTO TST.W D0 ; set CCR: BMI if send timeout, BPL if OK
|
|
RTS5
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: EgretReadByte [Egret utility routine]
|
|
;
|
|
; Inputs: A2 - VIA1 base address
|
|
; A6 - return address (BSR6)
|
|
;
|
|
; Outputs: D0 - byte read
|
|
;
|
|
; Trashes: D2,A5
|
|
;
|
|
; Function: reads a byte from Egret
|
|
;________________________________________________________________________________________
|
|
|
|
EgretReadByte
|
|
BCLR #vViaFullBit,vBufB(A2) ; de-assert VIA full
|
|
@Wait4Shift BTST #ifSR,vIFR(A2) ; wait for shift to complete
|
|
BEQ.S @Wait4Shift
|
|
BSET #vViaFullBit,vBufB(A2) ; acknowlege byte
|
|
MOVE.B vSR(A2),D0 ; read the byte
|
|
BSR5 EgretDelay ; give Egret time to see it
|
|
BTST #vXcvrSesBit,vBufB(A2) ; return with the state of XCVR session
|
|
RTS6
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: CudaRdXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to read
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: D1 - byte read from PRAM
|
|
;
|
|
; Trashes: D0,D2,A5,A6
|
|
;
|
|
; Function: reads a byte from Egret-based PRAM without using RAM
|
|
;
|
|
; This routine can only be executed after a SyncAck cycle. The SyncAck cycle,
|
|
; which is executed immediately after InitVia will disable all asynchronous
|
|
; message sources within Cuda. This allows the following routine to be written
|
|
; without regard to detecting collisions between command packets and response
|
|
; packets. AT NO TIME SHOULD AN ASYNCHRONOUS MESSAGE SOURCE BE ENABLED PRIOR
|
|
; TO USE OF THIS ROUTINE. ASYNCHRONOUS MESSAGE SOURCES SHOULD ONLY BE ENABLED
|
|
; AFTER THE CUDA MANAGER HAS BEEN INSTALLED.
|
|
;________________________________________________________________________________________
|
|
|
|
CudaRdXByte cmp.l #$8000,sp ; is there any stack yet?
|
|
ble.s @5 ; no, use register based
|
|
cmp.w #0,EgretBase
|
|
bne.s @5
|
|
movec VBR,d2 ; get the contents of VBR
|
|
tst.l d2 ; check for zero
|
|
bne.s @5
|
|
|
|
; This code allows the low level routines to be used after the Cuda manager is installed
|
|
; by constructing a parameter block on the stack and issuing a call to CudaDispatch. This
|
|
; is absolutely necessary because the low level routines do not detect collisions which can
|
|
; occur once the Cuda manager is installed. Further, it is almost impossible to determine
|
|
; how to to synchronize Cuda after an asynchronous re-boot has occurred.
|
|
|
|
movem.l a0-a6/d0/d2-d7,-(sp)
|
|
suba.l #EgretPBSize,sp ; make room on the stack for Cuda parameter block
|
|
move.b #pseudoPkt,pbCmdType(sp) ; set packet type
|
|
move.b #RdPram,pbCmd(sp) ; set packet command
|
|
move.w d1,pbParam(sp) ; set pram address
|
|
move.w #1,pbByteCnt(sp) ; 1 byte to send
|
|
move.l sp,pbBufPtr(sp)
|
|
add.l #pbParam+2,pbBufPtr(sp) ; point to our data buffer
|
|
move.b #0,pbFlags(sp)
|
|
move.w #0,pbResult(sp)
|
|
move.l #0,pbCompletion(sp) ; execute synchronously
|
|
move.l sp,a0 ; parameter block in a0
|
|
_EgretDispatch ; (this is actually the Cuda A-Trap)
|
|
move.w pbParam+2(sp),d1 ; get pram data
|
|
adda.l #EgretPBSize,sp ; remove parameter block from stack
|
|
movem.l (sp)+,a0-a6/d0/d2-d7
|
|
jmp (a3)
|
|
|
|
;
|
|
; The following code is executed only when the Cuda manager is not installed.
|
|
; AT NO TIME SHOULD AN ASYNCHRONOUS MESSAGE SOURCE BE ENABLED PRIOR TO USE OF THIS ROUTINE.
|
|
; ASYNCHRONOUS MESSAGE SOURCES SHOULD ONLY BE ENABLED AFTER THE CUDA MANAGER HAS BEEN INSTALLED.
|
|
;
|
|
@5
|
|
ori.b #ACRmode,vacr(a2) ; else initialize via mode
|
|
bset.b #SRdir,vACR(a2) ; set output mode
|
|
move.b #pseudopkt,vSR(a2) ; send packet type & clear any pending shift irq
|
|
bclr.b #TIP,vBufB(a2) ; start the transaction
|
|
@10
|
|
btst.b #ifSR,vIFR(a2) ; Wait for shift register IRQ to indicate byte was read
|
|
beq.s @10
|
|
|
|
;________________________________________________________________________________________________
|
|
; When outputing from the VIA to Cuda, the interrupt will occur prior to the rising edge
|
|
; of CB1. Delay long enough for the edge to occur before acknowledging the shift
|
|
; register interrupt. (R. Montagne 1/11/93)
|
|
;________________________________________________________________________________________________
|
|
moveq #10,d2 ; delay for 10µS min. <LW3><VIA rbm>
|
|
@m7dly ; <LW3><VIA rbm>
|
|
tst.b vBufB(a1) ; hardware access is 1.2µS <LW3><VIA rbm>
|
|
sub.b #1,d2 ; can only trash low byte <LW3><VIA rbm>
|
|
bne.s @m7dly ; wait long enough for CB1 to rise (10µS delay) <LW3><VIA rbm>
|
|
tst.b vSR(a2) ; Clear the shift register IRQ
|
|
|
|
moveq #rdpram,d2 ; Send the command byte
|
|
BSR6 CudaWriteByte
|
|
move.l d1,d2
|
|
lsr.l #8,d2 ; Send PRAM address MSB
|
|
BSR6 CudaWriteByte
|
|
move.l d1,d2 ; Send PRAM address LSB
|
|
BSR6 CudaWriteByte
|
|
|
|
bclr.b #SRdir,vACR(a2) ; Define direction FROM Cuda
|
|
btst.b #SRdir,vACR(a2)
|
|
@69up
|
|
bne.s @69up
|
|
|
|
ori.b #((1<<vCudaTIP)|\
|
|
(1<<vCudaByteAck)),vBufB(a2) ; terminate current command transaction
|
|
@Wait4Attn
|
|
btst.b #ifSR,vIFR(a2) ; Wait for attention IRQ
|
|
beq.s @Wait4Attn
|
|
tst.b vSR(a2) ; Read attn byte from VIA (clears IRQ)
|
|
|
|
bclr.b #vCudaTIP,vBufB(a2) ; assert TIP to begin response transaction
|
|
|
|
BSR6 CudaReadByte ; Read a byte - response packet type
|
|
BSR6 CudaReadByte ; next byte - response packet flag
|
|
BSR6 CudaReadByte ; next byte - "cmd was" byte
|
|
BSR6 CudaReadByte ; finally! PRAM data in d2
|
|
|
|
ori.b #((1<<vCudaTIP)|\
|
|
(1<<vCudaByteAck)),vBufB(a2) ; terminate current response transaction
|
|
@Wait4CudaDone
|
|
btst.b #vCudaTReq,vBufB(a2) ; Wait for TReq to negate
|
|
beq.s @Wait4CudaDone
|
|
@Wait4IdleIRQ
|
|
btst.b #ifSR,vIFR(a2) ; Wait for idle state IRQ
|
|
beq.s @Wait4IdleIRQ
|
|
tst.b vSR(a2) ; Read idle state byte (clears IRQ)
|
|
move.l d2,d1 ; return data in d1
|
|
jmp (a3) ; return to caller
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: CudaWrXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to write
|
|
; D2 - byte to write to PRAM
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: D0,D1,D2,A5,A6
|
|
;
|
|
; Function: writes a byte to Egret-based PRAM without using RAM
|
|
;
|
|
; This routine can only be executed after a SyncAck cycle. The SyncAck cycle,
|
|
; which is executed immediately after InitVia will disable all asynchronous
|
|
; message sources within Cuda. This allows the following routine to be written
|
|
; without regard to detecting collisions between command packets and response
|
|
; packets. AT NO TIME SHOULD AN ASYNCHRONOUS MESSAGE SOURCE BE ENABLED PRIOR
|
|
; TO USE OF THIS ROUTINE. ASYNCHRONOUS MESSAGE SOURCES SHOULD ONLY BE ENABLED
|
|
; AFTER THE CUDA MANAGER HAS BEEN INSTALLED.
|
|
;________________________________________________________________________________________
|
|
|
|
CudaWrXByte cmp.l #$8000,sp ; is there any stack yet? <SM5>(rbm)<2> begin {
|
|
ble.s @5 ; no, use register based
|
|
cmp.w #0,EgretBase
|
|
bne.s @5
|
|
movec vbr,d5 ; get a copy of VBR
|
|
tst.l d5 ; is it zero?
|
|
bne.s @5
|
|
|
|
; This code allows the low level routines to be used after the Cuda manager is installed
|
|
; by constructing a parameter block on the stack and issuing a call to CudaDispatch. This
|
|
; is absolutely necessary because the low level routines do not detect collisions which can
|
|
; occur once the Cuda manager is installed. Further, it is almost impossible to determine
|
|
; how to to synchronize Cuda after an asynchronous re-boot has occurred.
|
|
|
|
movem.l a0-a6/d0/d3-d7,-(sp)
|
|
suba.l #EgretPBSize,sp ; make room on the stack for Cuda parameter block
|
|
move.b #pseudoPkt,pbCmdType(sp) ; set packet type
|
|
move.b #WrPram,pbCmd(sp) ; set packet command
|
|
move.w d1,pbParam(sp) ; set pram address
|
|
move.w d2,pbParam+2(sp) ; set pram data
|
|
move.w #1,pbByteCnt(sp) ; 1 byte to send
|
|
move.l sp,pbBufPtr(sp)
|
|
add.l #pbParam+2,pbBufPtr(sp) ; point to our data buffer
|
|
move.b #0,pbFlags(sp)
|
|
move.w #0,pbResult(sp)
|
|
move.l #0,pbCompletion(sp) ; execute synchronously
|
|
move.l sp,a0 ; parameter block in a0
|
|
_EgretDispatch ; (this is actually the Cuda A-Trap)
|
|
adda.l #EgretPBSize,sp ; remove parameter block from stack
|
|
movem.l (sp)+,a0-a6/d0/d3-d7
|
|
jmp (a3)
|
|
|
|
; The following code is executed only when the Cuda manager is not installed.
|
|
; AT NO TIME SHOULD AN ASYNCHRONOUS MESSAGE SOURCE BE ENABLED PRIOR TO USE OF THIS ROUTINE.
|
|
; ASYNCHRONOUS MESSAGE SOURCES SHOULD ONLY BE ENABLED AFTER THE CUDA MANAGER HAS BEEN INSTALLED.
|
|
|
|
@5 move.l d2,d5 ; Save the write data
|
|
|
|
ori.b #ACRmode,vacr(a2) ; else initialize via mode
|
|
bset.b #SRdir,vACR(a2) ; set output mode
|
|
move.b #pseudopkt,vSR(a2) ; send packet type & clear any pending shift irq
|
|
bclr.b #TIP,vBufB(a2) ; start the transaction
|
|
@10
|
|
btst.b #ifSR,vIFR(a2) ; Wait for shift register IRQ to indicate byte was read
|
|
beq.s @10
|
|
;________________________________________________________________________________________________
|
|
; When outputing from the VIA to Cuda, the interrupt will occur prior to the rising edge
|
|
; of CB1. Delay long enough for the edge to occur before acknowledging the shift
|
|
; register interrupt. (R. Montagne 1/11/93)
|
|
;________________________________________________________________________________________________
|
|
move.b #10,d2 ; mode 7 interrupt occurs at falling edge CB1 <LW3><VIA rbm>
|
|
@m7dly ; <LW3><VIA rbm>
|
|
tst.b vBufB(a1) ; hardware access is 1.2µS <LW3><VIA rbm>
|
|
sub.b #1,d2 ; can only trash low byte <LW3><VIA rbm>
|
|
bne.s @m7dly ; wait long enough for CB1 to rise (10µS delay) <LW3><VIA rbm>
|
|
tst.b vSR(a2) ; Clear the shift register IRQ
|
|
|
|
moveq #wrpram,d2 ; Send the command byte
|
|
BSR6 CudaWriteByte
|
|
move.l d1,d2
|
|
lsr.l #8,d2 ; Send PRAM address MSB
|
|
BSR6 CudaWriteByte
|
|
move.l d1,d2 ; Send PRAM address LSB
|
|
BSR6 CudaWriteByte
|
|
move.l d5,d2 ; Send the write data
|
|
BSR6 CudaWriteByte
|
|
|
|
bclr.b #SRdir,vACR(a2) ; Define direction FROM Cuda
|
|
|
|
ori.b #((1<<vCudaTIP)|\
|
|
(1<<vCudaByteAck)),vBufB(a2) ; terminate current transaction
|
|
|
|
@40 btst.b #ifSR,vIFR(a2) ; Wait for attention byte shift register IRQ
|
|
beq.s @40
|
|
|
|
btst.b #vCudaTReq,vBufB(a2) ; Is TReq asserted? (active low)
|
|
bne.s @40 ; continue polling if TReq is not asserted
|
|
|
|
tst.b vSR(a2) ; Read attn byte from VIA (clears IRQ)
|
|
bclr.b #vCudaTIP,vBufB(a2) ; assert TIP to begin response transaction
|
|
@20
|
|
btst.b #ifSR,vIFR(a2) ; Wait for shift register IRQ to indicate byte was read
|
|
beq.s @20
|
|
tst.b vSR(a2) ; Clear the shift register IRQ
|
|
|
|
ori.b #((1<<vCudaTIP)|\
|
|
(1<<vCudaByteAck)),vBufB(a2) ; terminate the response transaction
|
|
|
|
@41 btst.b #ifSR,vIFR(a2) ; Wait for idle state IRQ
|
|
beq.s @41
|
|
tst.b vSR(a2) ; Read idle state byte (clears IRQ)
|
|
jmp (a3)
|
|
|
|
;---------------------------------------------------------------------------
|
|
|
|
EXPORT CudaWriteByte
|
|
CudaWriteByte
|
|
move.b d2,vSR(a2) ; Put data in VIA data reg
|
|
eori.b #1<<vCudaByteAck,vBufB(a2) ; indicate data moved into VIA
|
|
@10
|
|
btst.b #ifSR,vIFR(a2) ; Wait for shift register IRQ to indicate byte was read
|
|
beq.s @10
|
|
;________________________________________________________________________________________________
|
|
; When outputing from the VIA to Cuda, the interrupt will occur prior to the rising edge
|
|
; of CB1. Delay long enough for the edge to occur before acknowledging the shift
|
|
; register interrupt. (R. Montagne 1/11/93)
|
|
;________________________________________________________________________________________________
|
|
move.b #10,d2 ; mode 7 interrupt occurs at falling edge CB1 <LW3><VIA rbm>
|
|
@m7dly ; <LW3><VIA rbm>
|
|
tst.b vBufB(a1) ; hardware access is 1.2µS <LW3><VIA rbm>
|
|
sub.b #1,d2 ; can only trash low byte <LW3><VIA rbm>
|
|
bne.s @m7dly ; wait long enough for CB1 to rise (10µS delay) <LW3><VIA rbm>
|
|
tst.b vSR(a2) ; Clear the shift register IRQ
|
|
RTS6
|
|
|
|
;---------------------------------------------------------------------------
|
|
|
|
EXPORT CudaReadByte
|
|
CudaReadByte
|
|
btst.b #ifSR,vIFR(a2) ; Wait for IRQ
|
|
beq.s CudaReadByte
|
|
move.b vSR(a2),d2 ; Read data from VIA (clears IRQ)
|
|
and.w #$00FF,d2 ; isolate the byte
|
|
eori.b #1<<vCudaByteAck,vBufB(a2) ; handshake the byte
|
|
btst.b #vCudaTReq,vBufB(a2) ; Return with state of TReq Session <SM5>(rbm)<2> end }
|
|
RTS6
|
|
|
|
|
|
ENDWITH ; {EgretPB}
|
|
ENDIF ; {hasEgret}
|
|
|
|
|
|
|
|
IF hasProtectedPRAM THEN
|
|
; ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥ PRAM in VRAM, clock is someplace else ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥
|
|
WITH ProductInfo, VideoInfo
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMReadTime
|
|
;
|
|
; Inputs: none
|
|
;
|
|
; Outputs: D0 - result code
|
|
; "GetParam" contains the 32-bit time value read from the clock
|
|
;
|
|
; Trashes: TBD
|
|
;
|
|
; Function: Would read the time from the clock if we knew what kind it was. This routine
|
|
; is provided as an aid to the person who has to code for this case.
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMReadTime
|
|
MOVEQ #dsCoreErr,D0 ; just die so we know something didn't happen
|
|
_SysError
|
|
MOVEQ #clkRdErr,D0 ; and leave this as a clue to why
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMWriteTime
|
|
;
|
|
; Inputs: "Time" contains 32-bit quantity to write to clock
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: TBD
|
|
;
|
|
; Function: Would write the time to the clock if we knew what kind it was. This routine
|
|
; is provided as an aid to the person who has to code for this case.
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMWriteTime
|
|
MOVEQ #dsCoreErr,D0 ; just die so we know something didn't happen
|
|
_SysError
|
|
MOVEQ #clkWrErr,D0 ; and leave this as a clue to why
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMXParam
|
|
;
|
|
; Inputs: A0 - pointer to user's buffer
|
|
; D0 - [number of bytes to transfer][starting PRAM address]
|
|
; D1 - bit 12: 1=read, 0=write
|
|
;
|
|
; Outputs: D0 - result code
|
|
;
|
|
; Trashes: A0,A1,A2,D1,D2,D3
|
|
;
|
|
; Function: reads/writes byte[s] from parameter RAM that hides in unused video RAM
|
|
; (assumes that traps and the MMU are set up)
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMXParam
|
|
MOVEA.L A0,A2 ; save the pointer to the user's buffer,
|
|
MOVE.L D0,D3 ; and the starting PRAM address and count
|
|
|
|
MOVEQ #true32b,D0 ; switch into 32-bit addressing mode
|
|
_SwapMMUMode
|
|
|
|
MOVEA.L UnivInfoPtr,A1 ; point to this machine's product info,
|
|
ADDA.L VideoInfoPtr(A1),A1 ; which leads to video info,
|
|
MOVEA.L VRAMPhysAddr(A1),A1 ; which leads to the base address of video,
|
|
LEA -256(A1),A1 ; which leads to PRAM in unused video memory
|
|
ADD.W D3,A1 ; index into start of desired PRAM
|
|
SWAP D3 ; get the number of bytes to transfer
|
|
BTST #12,D1 ; is it a write?
|
|
BNE.S @StartCopy ; -> no
|
|
EXG A1,A2 ; yes, exchange source and destination pointers
|
|
BRA.S @StartCopy
|
|
|
|
@CopyPRAM MOVE.B (A1)+,(A2)+ ; move the bytes
|
|
@StartCopy DBRA D3,@CopyPRAM
|
|
|
|
_SwapMMUMode ; restore the addressing mode
|
|
MOVEQ #0,D0 ; never an error
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMXPramIO
|
|
;
|
|
; Inputs: A0 -- pointer to table of base addresses
|
|
; A1 -- pointer to ProductInfo record for this machine
|
|
; A3 -- pointer to PRAM I/O buffer
|
|
; D1 -- flags indicating which external features are valid
|
|
; D3 -- [r/w (bit 31)][number of PRAM bytes to read/write][starting PRAM address]
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: A2,A3,D3
|
|
;
|
|
; Function: reads/writes byte[s] from parameter RAM that hides in unused video RAM
|
|
; before traps (or the MMU) are set up
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMXPramIO
|
|
MOVEA.L A1,A2 ; point to this machine's product info,
|
|
ADDA.L VideoInfoPtr(A2),A2 ; which leads to video info,
|
|
MOVEA.L VRAMPhysAddr(A2),A2 ; which leads to the base address of video,
|
|
LEA -256(A2),A2 ; which leads to PRAM in unused video memory
|
|
ADD.W D3,A2 ; index into start of desired PRAM
|
|
SWAP D3 ; get the number of bytes to transfer
|
|
BCLR #15,D3 ; is it a write?
|
|
BEQ.S @StartCopy ; -> no
|
|
EXG A2,A3 ; yes, exchange source and destination pointers
|
|
BRA.S @StartCopy
|
|
|
|
@CopyPRAM MOVE.B (A2)+,(A3)+ ; move the bytes
|
|
@StartCopy DBRA D3,@CopyPRAM
|
|
RTS
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMRdXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to read
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: D1 - byte read from PRAM
|
|
;
|
|
; Trashes: A0,D0
|
|
;
|
|
; Function: reads a byte from parameter RAM that hides in unused video RAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMRdXByte
|
|
MOVEA.L A1,A0 ; point to this machine's product info,
|
|
ADDA.L VideoInfoPtr(A0),A0 ; which leads to video info,
|
|
MOVEA.L VRAMPhysAddr(A0),A0 ; which leads to the base address of video,
|
|
LEA -256(A0),A0 ; which leads to PRAM in unused video memory
|
|
MOVEQ #0,D0
|
|
MOVE.B D1,D0 ; zero-extend the byte number to make an index
|
|
MOVE.B 0(A0,D0),D1 ; and get the byte
|
|
JMP (A3)
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: NoPRAMWrXByte
|
|
;
|
|
; Inputs: D1 - address of PRAM byte to write
|
|
; D2 - byte to write to PRAM
|
|
; A1 - pointer to ProductInfo record for this machine
|
|
; A2 - VIA1 base address
|
|
; A3 - return address
|
|
;
|
|
; Outputs: none
|
|
;
|
|
; Trashes: A0,D0
|
|
;
|
|
; Function: writes a byte to parameter RAM that hides in unused video RAM without using RAM
|
|
;________________________________________________________________________________________
|
|
|
|
NoPRAMWrXByte
|
|
MOVEA.L A1,A0 ; point to this machine's product info,
|
|
ADDA.L VideoInfoPtr(A0),A0 ; which leads to video info,
|
|
MOVEA.L VRAMPhysAddr(A0),A0 ; which leads to the base address of video,
|
|
LEA -256(A0),A0 ; which leads to PRAM in unused video memory
|
|
MOVEQ #0,D0
|
|
MOVE.B D1,D0 ; zero-extend the byte number to make an index
|
|
MOVE.B D2,0(A0,D0) ; and write the byte
|
|
JMP (A3)
|
|
|
|
|
|
ENDWITH ; {ProductInfo, VideoInfo}
|
|
ENDIF ; {hasProtectedPRAM}
|
|
|
|
|
|
; ¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥¥
|
|
|
|
|
|
;________________________________________________________________________________________
|
|
;
|
|
; Routine: ClkNoMem
|
|
;
|
|
; Inputs: A1 - VIA 1 base address
|
|
; A5 - return address (BSR5)
|
|
; D1 - clock chip command byte (bit 7: 0=write, 1=read)
|
|
; D2 - data byte to write
|
|
;
|
|
; Outputs: D2 - data byte read
|
|
;
|
|
; Trashes: A0,A1,A5,A6,D1,D3
|
|
;
|
|
; Function: support for the jClkNoMem vector (do we still need this?)
|
|
;________________________________________________________________________________________
|
|
|
|
EXPORT ClkNoMem
|
|
ClkNoMem RTS5
|
|
;________________________________________________________________________________________ <H9>thru next <H9>
|
|
; These patches change the behavior of StandardXPramIO so that during a multi-byte
|
|
; write to PRAM, interrupts are disabled and re-enabled around the write of each byte,
|
|
; and the RTC is write-enabled and then write-protected again during the time that
|
|
; interrupts are off. This fixes a problem where the Reliability Manager's interrupt
|
|
; task, which calls this routine to write to PRAM every 5 minutes, interrupts during
|
|
; this loop, resulting in PRAM being left write-protected; when control returns to the
|
|
; foreground, remaining bytes to be written to PRAM don't get written.
|
|
;
|
|
; Monte Benaresh - 7/2/93
|
|
;________________________________________________________________________________________
|
|
|
|
StdXPRAM2Patch ; patch for StandardXPramIO
|
|
MOVEA.L A1,A6 ; point to this machine's product info
|
|
ADDA.L ProductInfo.ClockPRAMPtr(A6),A6 ; and get the address of its clock/PRAM table
|
|
MOVE.L 4*cpWrProtOn(A6),D0 ; get the offset to the write-protect routine
|
|
BEQ.S @Done ; -> it's not supported for this implementation
|
|
ADDA.L D0,A6 ; calculate the routine's address
|
|
movea.l (sp)+,A5 ; simulate a BSR5 which returns to our caller
|
|
JMP (A6) ; and call it to turn on PRAM write-protect
|
|
@Done
|
|
rts ; <H9>
|
|
|
|
END
|