mirror of
https://github.com/elliotnunn/mac-rom.git
synced 2025-01-18 00:31:20 +00:00
1463 lines
56 KiB
Plaintext
1463 lines
56 KiB
Plaintext
|
;__________________________________________________________________________________________________
|
|||
|
;
|
|||
|
; File: IOPMgr.a
|
|||
|
;
|
|||
|
; Contains: This module provides the message passing interface between the 680XX and the
|
|||
|
; drivers running on the IOPs. It also provides the routines to download code
|
|||
|
; and to initialize the IOPs.
|
|||
|
;
|
|||
|
; NOTE: The IOP hardware is called a "Peripheral Interface Controller" or "PIC",
|
|||
|
; which can cause some confusion with other names already in the vocabulary
|
|||
|
; of Macintosh developers ("PICT" for example), so from the software side,
|
|||
|
; we will call them "Input / Output Processors" or "IOP" to avoid (or create)
|
|||
|
; confusion.
|
|||
|
;
|
|||
|
; Written by: Gary G. Davidian
|
|||
|
;
|
|||
|
; Copyright <20> 1987-1993 by Apple Computer, Inc. All rights reserved.
|
|||
|
;
|
|||
|
; Change History (most recent first):
|
|||
|
;
|
|||
|
; <SM9> 11/10/93 KW fix localtalk for iop based machines on STP card use forSTP601v1
|
|||
|
; <SM8> 11/9/93 KW BTU on the STP card doesn't handle bytereads very well so
|
|||
|
; changed IOP stuff. Changes are only made when forSTP601=TRUE
|
|||
|
; <SM7> 8/20/93 chp References to SCCIOPBypassInt have been resolved to the more
|
|||
|
; general and historically accepted SccDecode entry point in
|
|||
|
; InterruptHandlers.a. It wasn't necessary to have two names for
|
|||
|
; the same thing.
|
|||
|
; <SM6> 11/19/92 RB When looking for iop code, look in ROM first.
|
|||
|
; <SM5> 9/10/92 kc Fix booting problem on Eclipse. (The last change was trashing a
|
|||
|
; register in IOPInterrupt)
|
|||
|
; <SM4> 6/11/92 PN Roll in Cmd-Shift-Esc patch from PatchIIci.a
|
|||
|
; <SM3> 5/21/92 kc Append "Trap" to the names of Enqueue and Dequeue to avoid name
|
|||
|
; conflict with the glue.
|
|||
|
; <9> 4/27/92 JSM Get rid of conditionals: isUniversal, hasIOPSCC, and hasIOPSWIM
|
|||
|
; are always true, left in the padForOverpatch conditional for
|
|||
|
; now.
|
|||
|
; <8> 12/30/91 RB Made a beq instruction use the long format in order to link. The
|
|||
|
; SCCIOPByPassInt routine cannot be accessed with a 16 bit
|
|||
|
; reference from ROM.
|
|||
|
; <7> 10/1/91 JSM Don<6F>t use eclipseDebug.
|
|||
|
; <6> 8/30/91 JSM Cleanup header.
|
|||
|
; <5> 6/12/91 LN removed #include 'HardwareEqu.a'
|
|||
|
; <4> 9/1/90 BG Added a check to leave Eclipses in Bypass Mode if we are running
|
|||
|
; from RAM so that ReAnimator doesnt have its SCC communications
|
|||
|
; path kicked out from under itself.
|
|||
|
; <3> 1/11/90 CCH Added include of <20>HardwarePrivateEqu.a<>.
|
|||
|
; <2> 1/11/90 SWC Set the bypass mode bit in SCCIOPFlag if we can't put the SCC
|
|||
|
; IOP into enhanced mode so that the serial and AppleTalk drivers
|
|||
|
; will at least configure correctly. Changed the initialization of
|
|||
|
; the SCC IOP SccIOCtlReg from $44 to $23 (per Bob Hollyer).
|
|||
|
; <2.2> 12/1/89 GGD Cleaned up after BAD DOG (SWC), don't include Private.a twice,
|
|||
|
; It was already included by Inc.Sum.a.
|
|||
|
; <2.1> 12/1/89 SWC Oops! Forgot to INCLUDE Private.a since that's where SCCIOPFlag
|
|||
|
; is defined.
|
|||
|
; <2.0> 12/1/89 SWC Changed the polarity of the mode bit (0) in SCCIOPFlag so that a
|
|||
|
; zero means we use IOP mode and a 1 means we use bypass so that
|
|||
|
; if PRAM is reset, Zone 5 will use IOP mode as its default state.
|
|||
|
; <1.9> 11/29/89 SWC Copied the PRAM byte controlling SCC IOP state to new low mem
|
|||
|
; variable SCCIOPFlag ($BFE) so that we can change the PRAM value
|
|||
|
; without hosing current IOP clients.
|
|||
|
; <1.8> 11/2/89 GGD NEEDED FOR ZONE-5 Changed the rules pertaining to the usage of
|
|||
|
; the IOP Ram Address Pointer register, The old convention was
|
|||
|
; that interrupts could trash it, the new convention is that
|
|||
|
; interrupts must save/restore it. Modified all routines to
|
|||
|
; preserve it, and improved the cross processor data transfer
|
|||
|
; performance by not having to mask interrupts and save/restore
|
|||
|
; the addr reg within the xfer loop. Added Vectoring of the
|
|||
|
; interrupt handler, and change internal calls of IOPMgr trap
|
|||
|
; routines to jump through the OS Dispatch table, to make future
|
|||
|
; patching easier.
|
|||
|
; <1.7> 7/25/89 GGD Fixed interrupt masking bug in Deferred Task queueing which
|
|||
|
; caused a circular queue if the SCC IOP interrupted the SWIM IOP
|
|||
|
; interrupt processing at the wrong time. NEEDED FOR F19, Code
|
|||
|
; size doesn't change.
|
|||
|
; <1.6> 7/14/89 GGD Added conditionals around the overpatch padding so that it can
|
|||
|
; be easily located. Turned off the Debugging VBL task. Improved
|
|||
|
; the handling of interrupts from unknown IOPs to check for the
|
|||
|
; SCC IOP, and pass the interrupt on to SCCIOPBypassInt if it was
|
|||
|
; from the SCC.
|
|||
|
; <1.5> 6/30/89 GGD Modified SCCIOPByPass to not hang waiting for a reply if the
|
|||
|
; Send failed. Added padding in preparation for the F19 overpatch.
|
|||
|
; <1.4> 5/29/89 GGD Added new routine SCCIOPHwInit which resets the SCC IOP, puts it
|
|||
|
; into bypass mode, and initializes the IOControl register holdoff
|
|||
|
; and wait delays.
|
|||
|
; <1.3> 5/20/89 GGD Removed IOP register equates, and moved them into HardwareEqu,
|
|||
|
; converted to use new names for registers and bits. Made the VBL
|
|||
|
; debugging task support code conditionally assembly so that it
|
|||
|
; can be easily turned off in the future. Re-wrote initialization
|
|||
|
; code to make it universal.
|
|||
|
; <1.2> 2/8/89 SGS Added support for SCC IOP
|
|||
|
; <1.1> 11/10/88 CCH Fixed Header.
|
|||
|
; <1.0> 11/9/88 CCH Adding to EASE.
|
|||
|
; <1.1> 11/6/88 GGD Modified interrupt handler to allow for non-deferrable message
|
|||
|
; handling (needed for MacsBug/ADB support). Fixed a few
|
|||
|
; un-noticed bugs. Changed interrupt handler to service all
|
|||
|
; interrupt sources before returning because on IoMac the SWIM IOP
|
|||
|
; interrupt line is an edge-triggered VIA input and not a level
|
|||
|
; triggered interrupt. Changed install IOP code to use
|
|||
|
; _RGetResource to find the 'iopc' resource so that a system file
|
|||
|
; can override it.
|
|||
|
; <1.0> 10/7/88 rwh New to EASE today.
|
|||
|
; 6/6/88 GGD Re-Written to allow queued xmt requests, and more flexible data
|
|||
|
; structures to allow IOPs to be added dynamicaly.
|
|||
|
; 7/31/87 GGD Created today.
|
|||
|
;__________________________________________________________________________________________________
|
|||
|
|
|||
|
TITLE 'IOPmgr - Input / Output Processor Manager'
|
|||
|
|
|||
|
BLANKS ON
|
|||
|
STRING ASIS
|
|||
|
PRINT OFF
|
|||
|
LOAD 'StandardEqu.d'
|
|||
|
INCLUDE 'HardwarePrivateEqu.a'
|
|||
|
INCLUDE 'UniversalEqu.a'
|
|||
|
PRINT ON
|
|||
|
INCLUDE 'IOPequ.a'
|
|||
|
MACHINE MC68020
|
|||
|
PRINT NOMDIR
|
|||
|
|
|||
|
macro
|
|||
|
assert &boolExpr
|
|||
|
if not(&Eval(&boolExpr)) then
|
|||
|
aerror &concat('Assertion Failed - ',&boolExpr)
|
|||
|
endif
|
|||
|
endm
|
|||
|
|
|||
|
JIOPMoveData EQU ($88*4)+OSTable ; OS trap table entry for _IOPMoveData
|
|||
|
|
|||
|
Debugging equ 0 ; use debugging VBL task <1.3><1.6>
|
|||
|
|
|||
|
IOPPRAM equ $00000089 ; IOP PRAM index <1.2>
|
|||
|
Active equ $FF ; dummy value for active
|
|||
|
ReplyLen equ $01 ; bypass message reply length
|
|||
|
MsgLen equ $03 ; bypass message length
|
|||
|
KernMsg equ $01 ; kernel message box number
|
|||
|
ByPassCmd equ $04 ; bypass command
|
|||
|
|
|||
|
|
|||
|
MaxIOPRamAddr equ $7FFF ; IOP has 32KB RAM
|
|||
|
IOPMsgPage equ $02
|
|||
|
IOPXmtMsgBase equ $0200
|
|||
|
IOPRcvMsgBase equ $0300
|
|||
|
PatchReqAddr equ $021F ; Shared Loc to sync patching
|
|||
|
IOPAliveAddr equ $031F ; IOP sets to $FF when idle
|
|||
|
|
|||
|
AliveVBLrate equ 127 ; 127 ticks = 2.1166666 seconds
|
|||
|
|
|||
|
; Message State encodings
|
|||
|
MsgIdle equ 0 ; message buffer idle
|
|||
|
NewMsgSent equ 1 ; new message just sent
|
|||
|
MsgReceived equ 2 ; message received, and being processed
|
|||
|
MsgCompleted equ 3 ; message processing complete, reply available
|
|||
|
|
|||
|
|
|||
|
; Internal IOP Manager data structures (IOPmgrVars is the root pointer)
|
|||
|
|
|||
|
IOPMgrGlobals record {IOPInfoPtrs},increment ; miscellaneous globals used by the IOP manager
|
|||
|
if Debugging then ; <1.3>
|
|||
|
ds.w 1 ; force nice long word alignments
|
|||
|
VTask ds.b vblPhase+2 ; Virtical Retrace Task info
|
|||
|
endif ; <1.3>
|
|||
|
DTask ds.b dtQElSize ; Deferred task Manager task info
|
|||
|
ds.b 1 ; filler
|
|||
|
DTaskQueued ds.b 1 ; $FF if task queued or running, $00 when done
|
|||
|
CompleteQHdr ds.b qHeadSize ; queue of completed requests
|
|||
|
IntHandlerPtr ds.l 1 ; [long] ptr to IOP interrupt handler
|
|||
|
IOPInfoPtrs ds.l NumberOfIOPs ; [long] ptrs to IOP info for each IOP
|
|||
|
|
|||
|
IOPInfos ds.b 0 ; start of IOPInfo records for installed IOPs
|
|||
|
|
|||
|
GlobalsSize equ *-IOPMgrGlobals ; size of IOPMgrGlobals
|
|||
|
endr
|
|||
|
|
|||
|
|
|||
|
IOPmgr PROC EXPORT
|
|||
|
EXPORT InitIOPMgr
|
|||
|
EXPORT IOPInfoAccess
|
|||
|
EXPORT IOPMsgRequest
|
|||
|
EXPORT IOPMoveData
|
|||
|
EXPORT IOPInterrupt
|
|||
|
EXPORT MoveReqHandler
|
|||
|
EXPORT SCCIOPByPass ;<1.2>
|
|||
|
EXPORT SCCIOPHwInit ;<1.4>
|
|||
|
IMPORT SccDecode
|
|||
|
IMPORT EnqueueTrap
|
|||
|
IMPORT DequeueTrap
|
|||
|
|
|||
|
TITLE 'IOPmgr - IOP Info Access'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPInfoAccess
|
|||
|
; Inputs: A0 - pointer to IOPAccessInfo paramater block
|
|||
|
; Outputs: D0 - Result Code (NoErr/paramErr)
|
|||
|
; Destroys: A0, A1, D0, D1, D2
|
|||
|
; Calls: none
|
|||
|
; Called by: OsTrap Dispatch Table
|
|||
|
;
|
|||
|
; Function: Performs various access operations on an IOP.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPAccessInfo,IOPInfo,IOPRequestInfo,IOPMoveInfo
|
|||
|
IOPInfoAccess ; a0-a1/d1-d2 saved by OsTrap dispatch
|
|||
|
@SavedRegs reg a2-a4/d3-d4
|
|||
|
movem.l @SavedRegs,-(sp) ; save registers
|
|||
|
moveq.l #NoErr,d0 ; assume success
|
|||
|
moveq.l #0,d1 ; zero extend
|
|||
|
|
|||
|
move.b iaAccessKind(a0),d1 ; get the access kind
|
|||
|
cmpi.b #iaRemoveIOP,d1 ; range check it
|
|||
|
bhi.s @paramError ; if out of range
|
|||
|
move.w @JumpTable(d1.w*2),d2 ; get the routine offset
|
|||
|
|
|||
|
move.b iaIOPNumber(a0),d1 ; get the IOP number
|
|||
|
cmpi.b #MaxIopNum,d1 ; range check it
|
|||
|
bhi.s @paramError ; if out of range
|
|||
|
lea ([IOPmgrVars],d1.w*4,\ ; a1 := pointer into IOPInfoPtrs table
|
|||
|
IOPMgrGlobals.IOPInfoPtrs),a1
|
|||
|
|
|||
|
jmp @JumpTable(d2.w) ; process the request
|
|||
|
|
|||
|
@JumpTable assert iaInstallIOP=((*-@JumpTable)/2)
|
|||
|
dc.w @InstallIOP-@JumpTable
|
|||
|
|
|||
|
assert iaGetIOPInfo=((*-@JumpTable)/2)
|
|||
|
dc.w @GetIOPInfo-@JumpTable
|
|||
|
|
|||
|
assert iaRemoveIOP=((*-@JumpTable)/2)
|
|||
|
dc.w @RemoveIOP-@JumpTable
|
|||
|
|
|||
|
@paramError moveq.l #paramErr,d0 ; indicate failure
|
|||
|
bra.s @Done ; restore registers and return
|
|||
|
|
|||
|
|
|||
|
@GetIOPInfo move.l (a1),iaIOPInfoPtr(a0) ; return the pointer to IOP Info
|
|||
|
@Done movem.l (sp)+,@SavedRegs ; restore registers
|
|||
|
rts ; and return
|
|||
|
|
|||
|
|
|||
|
@RemoveIOP move.l (a1),iaIOPInfoPtr(a0) ; return the old pointer to IOP Info
|
|||
|
clr.l (a1) ; and remove it from the table
|
|||
|
bra.s @Done ; return with success
|
|||
|
|
|||
|
|
|||
|
@InstallIOP tst.l (a1) ; see if it is already installed
|
|||
|
bne.s @paramError ; if so, it's an error
|
|||
|
movea.l iaIOPInfoPtr(a0),a4 ; save IOPInfo pointer
|
|||
|
move.l d1,d3 ; save IOP number
|
|||
|
subq.l #4,sp ; allocate function result
|
|||
|
move.l #'iopc',-(sp) ; ResType = 'iopc' (IOP code)
|
|||
|
move.w d3,-(sp) ; theID = IOP number
|
|||
|
move.w #MapTrue,RomMapInsert ; look in ROM first <SM6> rb
|
|||
|
_GetResource ; top of stack <- IOP code resource handle <SM6> rb
|
|||
|
move.l (sp)+,d0 ; d0 <- resource handle
|
|||
|
beq.s @paramError ; error if neither found
|
|||
|
movea.l d0,a0 ; a0 <- resource handle
|
|||
|
movea.l (a0),a0 ; a0 <- resource pointer
|
|||
|
|
|||
|
; Note that this routine will destroy the old contents of the IopAddrReg, but
|
|||
|
; since we are also destroying all of it's RAM contents, it isn't worth the
|
|||
|
; trouble to save it.
|
|||
|
move.w sr,-(sp) ; save interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; disable interrupts
|
|||
|
assert IopAddrRegPtr=0
|
|||
|
movea.l (a4)+,a2 ; a2 <- IOP Ram Address Reg pointer
|
|||
|
assert IopDataRegPtr=(IopAddrRegPtr+4)
|
|||
|
movea.l (a4)+,a3 ; a3 <- IOP Ram Data Reg pointer
|
|||
|
assert IopCtlRegPtr=(IopDataRegPtr+4)
|
|||
|
movea.l (a4),a1 ; a1 <- IOP Status/Ctl Reg pointer
|
|||
|
move.b #resetIopRun,(a1) ; init the Status/Ctl reg, hold IOP reset
|
|||
|
|
|||
|
; Fill all of IOP memory with $FF
|
|||
|
|
|||
|
move.w #MaxIOPRamAddr,d0 ; d0 <- fill addr / loop counter
|
|||
|
moveq.l #-1,d1 ; d1 <- fill data ($FF)
|
|||
|
@Fill_FF_Loop
|
|||
|
move.w d0,(a2) ; load the address register
|
|||
|
move.b d1,(a3) ; write the data to the data reg
|
|||
|
dbra d0,@Fill_FF_Loop ; fill all of IOP memory with $FF
|
|||
|
|
|||
|
; Complement all of IOP memory. $FF -> $00
|
|||
|
|
|||
|
move.w #MaxIOPRamAddr,d0 ; d0 <- Complement addr / loop counter
|
|||
|
@ComplementLoop move.w d0,(a2) ; load the address register
|
|||
|
move.b (a3),d1 ; read the data byte
|
|||
|
not.b d1 ; complement the data
|
|||
|
move.w d0,(a2) ; reload the address register
|
|||
|
move.b d1,(a3) ; write the data to the data reg
|
|||
|
dbne d0,@ComplementLoop ; Complement all of IOP memory
|
|||
|
bne.s @InitError ; if didn't read back $FF (HW error)
|
|||
|
|
|||
|
; Check all of IOP memory for $00
|
|||
|
|
|||
|
move.w #MaxIOPRamAddr,d0 ; d0 <- Check addr / loop counter
|
|||
|
@Check_00_Loop move.w d0,(a2) ; load the address register
|
|||
|
move.b (a3),d1 ; read the data byte
|
|||
|
dbne d0,@Check_00_Loop ; Check all of IOP memory for $00
|
|||
|
bne.s @InitError ; if didn't read back $00 (HW error)
|
|||
|
|
|||
|
; Download and Verify the IOP code
|
|||
|
|
|||
|
movea.l a0,a1 ; a1 <- copy of code start
|
|||
|
@SegLoop move.b (a0)+,d0 ; get the size of the segment
|
|||
|
beq.s @LoadDone ; size of zero terminates segment list
|
|||
|
move.w (a0)+,(a2) ; setup the code load address
|
|||
|
|
|||
|
@LoadLoop move.b (a0)+,(a3) ; download a byte
|
|||
|
subq.b #1,d0 ; decrement the loop count
|
|||
|
bne.s @LoadLoop ; load all of the bytes
|
|||
|
|
|||
|
move.b (a1)+,d0 ; get size of the segment
|
|||
|
move.w (a1)+,(a2) ; setup the code check address
|
|||
|
|
|||
|
@CheckCodeLoop move.b (a3),d1 ; read a byte from the IOP
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
cmp.b (a1)+,d1 ; compare it to what was loaded
|
|||
|
bne.s @InitError ; if download verify error (HW error)
|
|||
|
subq.b #1,d0 ; decrement the loop count
|
|||
|
bne.s @CheckCodeLoop ; check all of the bytes
|
|||
|
|
|||
|
bra.s @SegLoop ; load the next segment
|
|||
|
@LoadDone
|
|||
|
|
|||
|
; Start IOP execution
|
|||
|
|
|||
|
move.w #IOPAliveAddr,(a2) ; Address the ALIVE flag
|
|||
|
clr.b (a3) ; clear ALIVE
|
|||
|
assert IopCtlRegPtr=(IopDataRegPtr+4)
|
|||
|
movea.l (a4),a1 ; a1 <- IOP Status/Ctl Reg pointer
|
|||
|
move.b #setIopRun,(a1) ; release IOP reset
|
|||
|
moveq.l #-1,d0 ; timeout loop counter
|
|||
|
@AliveWaitLoop
|
|||
|
move.w #IOPAliveAddr,(a2) ; Address the ALIVE flag
|
|||
|
cmpi.b #$FF,(a3) ; check for alive
|
|||
|
dbeq d0,@AliveWaitLoop ; loop until alive
|
|||
|
beq.s @Alive ; error if time out (HW error)
|
|||
|
@InitError move.w (sp)+,sr ; restore interrupt priority
|
|||
|
bra.w @paramError ; return with error
|
|||
|
@Alive
|
|||
|
|
|||
|
; Update the Max Xmt / Rcv messages supported
|
|||
|
|
|||
|
move.w #IOPXmtMsgBase,(a2) ; address Xmt MAX
|
|||
|
addq.l #MaxXmt-IopCtlRegPtr,a4 ; point to MaxXmt in IOPInfo
|
|||
|
move.b (a3),(a4)+ ; get the Xmt MAX
|
|||
|
|
|||
|
move.w #IOPRcvMsgBase,(a2) ; address Rcv MAX
|
|||
|
assert MaxRcv=(MaxXmt+1)
|
|||
|
move.b (a3),(a4) ; get the Rcv MAX
|
|||
|
|
|||
|
suba.w #MaxRcv,a4 ; point to start of IOPInfo record
|
|||
|
move.l a4,([IOPmgrVars],d3.w*4,\ ; install pointer into IOPInfoPtrs table
|
|||
|
IOPMgrGlobals.IOPInfoPtrs)
|
|||
|
|
|||
|
lea MoveReqInfo(a4),a0 ; a0 := pointer to IOPRequestInfo
|
|||
|
lea irIOPNumber(a0),a1 ; a1 := initialization pointer
|
|||
|
move.b d3,(a1)+ ; initialize the irIOPNumber field
|
|||
|
assert irRequestKind=(irIOPNumber+1)
|
|||
|
move.b #irWaitRcvMessage,(a1)+ ; initialize the irRequestKind field
|
|||
|
|
|||
|
assert irMsgNumber=(irRequestKind+1)
|
|||
|
assert irMessageLen=(irMsgNumber+1)
|
|||
|
assert irReplyLen=(irMessageLen+1)
|
|||
|
assert irReqActive=(irReplyLen+1)
|
|||
|
move.l #(1<<24)+\ ; irMsgNumber := 1
|
|||
|
(imMoveInfoSize<<16)+\ ; irMessageLen := imMoveInfoSize
|
|||
|
(imMoveInfoSize<<8)+\ ; irReplyLen := imMoveInfoSize
|
|||
|
(0),(a1)+ ; irReqActive := 0
|
|||
|
|
|||
|
lea MoveReqBuffer(a4),a2 ; a2 := pointer to IOPMoveInfo
|
|||
|
assert irMessagePtr=(irReqActive+1)
|
|||
|
move.l a2,(a1)+ ; irMessagePtr := MoveReqBuffer
|
|||
|
assert irReplyPtr=(irMessagePtr+4)
|
|||
|
move.l a2,(a1)+ ; irReplyPtr := MoveReqBuffer
|
|||
|
|
|||
|
lea MoveReqHandler,a2 ; a2 := completion routine address
|
|||
|
assert irHandler=(irReplyPtr+4)
|
|||
|
move.l a2,(a1)+ ; irHandler := MoveReqHandler
|
|||
|
_IOPMsgRequest ; install the message handler
|
|||
|
|
|||
|
move.w (sp)+,sr ; restore interrupt priority
|
|||
|
bra.w @Done ; return with success
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - IOP Message Request'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPMsgRequest
|
|||
|
; Inputs: A0 - pointer to IOPRequestInfo paramater block
|
|||
|
; Outputs: D0 - Result Code (NoErr/paramErr)
|
|||
|
; irReqActive - set to $FF, until request completed
|
|||
|
; Destroys: A0, A1, D0, D1, D2
|
|||
|
; Calls: CopyToIOP, Enqueue
|
|||
|
; Called by: OsTrap Dispatch Table
|
|||
|
;
|
|||
|
; Function:
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPRequestInfo
|
|||
|
IOPMsgRequest
|
|||
|
@SavedRegs reg a2/a3/d3/d4 ; a0-a1/d1-d2 saved by OsTrap dispatch
|
|||
|
movem.l @SavedRegs,-(sp) ; save registers
|
|||
|
|
|||
|
moveq.l #paramErr,d0 ; assume Parameter Error status
|
|||
|
moveq.l #0,d2 ; prepare to zero extend
|
|||
|
|
|||
|
lea irIOPNumber(a0),a1 ; a1 <- pointer to parameters
|
|||
|
move.b (a1)+,d2 ; d2 <- zero extended IOP number
|
|||
|
cmpi.w #MaxIopNum,d2 ; range check the iop number
|
|||
|
bhi.s @toParamError ; if out of range
|
|||
|
|
|||
|
movea.l IOPmgrVars,a2 ; get message table base
|
|||
|
movea.l IOPMgrGlobals.IOPInfoPtrs(a2,d2.w*4),a2 ; a2 <- iop info ptr
|
|||
|
move.l a2,d3 ; test for zero
|
|||
|
beq.s @toParamError ; IOP isn't initialized
|
|||
|
move.b IOPInfo.MaxXmt(a2),d3 ; d3 := max message number
|
|||
|
|
|||
|
assert irRequestKind=(irIOPNumber+1)
|
|||
|
move.b (a1)+,d2 ; d2 <- zero extended request Kind
|
|||
|
assert irSendXmtMessage=0
|
|||
|
beq.s @CheckReqKind ; if for Xmt Msg, max is correct
|
|||
|
move.b IOPInfo.MaxRcv(a2),d3 ; d3 := max message number
|
|||
|
@CheckReqKind
|
|||
|
cmpi.w #irRemoveRcvWaiter,d2 ; range check the request Kind
|
|||
|
bhi.s @toParamError ; if out of range
|
|||
|
move.w @JumpTable(d2.w*2),d1 ; d1 := offset to routine
|
|||
|
|
|||
|
assert irMsgNumber=(irRequestKind+1)
|
|||
|
move.b (a1)+,d2 ; d2 <- zero extended Msg number
|
|||
|
beq.s @paramError ; Message zero is reserved
|
|||
|
cmp.b d3,d2 ; compare to max
|
|||
|
bhi.s @paramError ; if out of range
|
|||
|
move.w d2,d3 ; d3.w := iop state address
|
|||
|
assert IOPMsgEntry.IOPMsgEntrySize=(1<<4)
|
|||
|
lsl.w #4,d2 ; d2 := index into MsgTable
|
|||
|
|
|||
|
moveq.l #MaxIopMsgLen,d4 ; prepare to check message lengths
|
|||
|
assert irMessageLen=(irMsgNumber+1)
|
|||
|
cmp.b (a1)+,d4 ; check the message length
|
|||
|
blo.s @paramError ; if out of range
|
|||
|
assert irReplyLen=(irMessageLen+1)
|
|||
|
cmp.b (a1)+,d4 ; check the reply length
|
|||
|
blo.s @paramError ; if out of range
|
|||
|
|
|||
|
lea IOPInfo.MsgTable+IOPMsgEntry.RcvMsgInfoPtr(a2,d2.w),a3
|
|||
|
jmp @JumpTable(d1.w) ; process the request
|
|||
|
@JumpTable assert irSendXmtMessage=((*-@JumpTable)/2)
|
|||
|
dc.w @SendXmtMessage-@JumpTable
|
|||
|
|
|||
|
assert irSendRcvReply=((*-@JumpTable)/2)
|
|||
|
dc.w @SendRcvReply-@JumpTable
|
|||
|
|
|||
|
assert irWaitRcvMessage=((*-@JumpTable)/2)
|
|||
|
dc.w @WaitRcvMessage-@JumpTable
|
|||
|
|
|||
|
assert irRemoveRcvWaiter=((*-@JumpTable)/2)
|
|||
|
dc.w @RemoveRcvWaiter-@JumpTable
|
|||
|
|
|||
|
@toParamError
|
|||
|
bra.s @paramError ; allow short branches
|
|||
|
|
|||
|
@SendXmtMessage
|
|||
|
assert irReqActive=(irReplyLen+1)
|
|||
|
st (a1) ; mark it as active now
|
|||
|
lea IOPMsgEntry.XmtMsgQHdr-IOPMsgEntry.RcvMsgInfoPtr(a3),a1
|
|||
|
move.w sr,d2 ; d2 := saved interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; no interrupts while checking queue
|
|||
|
cmpa.l qHead(a1),a0 ; see if already the head of the queue
|
|||
|
beq.s @SkipEnqueue ; if so, don't enqueue it
|
|||
|
jsr EnqueueTrap ; enqueue the request
|
|||
|
cmpa.l qHead(a1),a0 ; see if at the head of the queue
|
|||
|
bne.s @notAtHead ; exit with request queued
|
|||
|
@SkipEnqueue
|
|||
|
move.w d2,sr ; restore interrupt priority
|
|||
|
moveq.l #0,d0 ; zero extend message length
|
|||
|
move.b irMessageLen(a0),d0 ; get the message buffer length
|
|||
|
movea.l irMessagePtr(a0),a1 ; get the message buffer pointer
|
|||
|
move.b #NewMsgSent,-(sp) ; save new message state
|
|||
|
ori.w #IOPXmtMsgBase,d3 ; d3 := iop message state address
|
|||
|
@SendAndInterrupt
|
|||
|
move.w d3,-(sp) ; save state address for later
|
|||
|
assert MaxIOPMsgLen=(1<<5)
|
|||
|
lsl.b #5,d3 ; d3 := iop message address
|
|||
|
move.l IOPInfo.IopCtlRegPtr(a2),-(sp) ; save ctl/status reg ptr for later
|
|||
|
movea.l IOPInfo.IopDataRegPtr(a2),a3 ; (a3) := IOPRamDataReg
|
|||
|
movea.l IOPInfo.IopAddrRegPtr(a2),a2 ; (a2) := IOPRamAddrReg
|
|||
|
move.w (a2),-(sp) ; save the IOPRamAddrReg
|
|||
|
move.w d3,(a2) ; setup iop ram address
|
|||
|
bsr.w CopyToIop ; copy the message to the IOP
|
|||
|
move.w (sp)+,d0 ; get the saved IOPRamAddrReg
|
|||
|
movea.l (sp)+,a1 ; (a1) := IopCtlReg
|
|||
|
move.w (sp)+,(a2) ; setup iop ram address of message state
|
|||
|
move.b (sp)+,(a3) ; set new state
|
|||
|
move.w d0,(a2) ; restore the IOPRamAddrReg
|
|||
|
move.b #setIopGenInt,(a1) ; interrupt the IOP
|
|||
|
@success moveq.l #noErr,d0 ; report success
|
|||
|
@paramError movem.l (sp)+,@SavedRegs
|
|||
|
rts ; all done
|
|||
|
@notAtHead move.w d2,sr ; restore interrupt priority
|
|||
|
bra.s @success ; return with success
|
|||
|
|
|||
|
@SendRcvReply
|
|||
|
cmpa.l (a3),a0 ; see if we are the waiter
|
|||
|
bne.s @paramError ; return with error if not
|
|||
|
assert irReqActive=(irReplyLen+1)
|
|||
|
tst.b (a1) ; see if the request is already active
|
|||
|
bne.s @paramError ; return with error if active
|
|||
|
st (a1) ; mark it as active now
|
|||
|
moveq.l #0,d0 ; zero extend reply length
|
|||
|
move.b irReplyLen(a0),d0 ; get the reply buffer length
|
|||
|
movea.l irReplyPtr(a0),a1 ; get the reply buffer pointer
|
|||
|
move.b #MsgCompleted,-(sp) ; save new message state
|
|||
|
ori.w #IOPRcvMsgBase,d3 ; d3 := iop message state address
|
|||
|
bra.s @SendAndInterrupt ; send the reply and interrupt the IOP
|
|||
|
|
|||
|
|
|||
|
@WaitRcvMessage
|
|||
|
tst.l (a3) ; see if waiter already exists
|
|||
|
bne.s @paramError ; return with error if so
|
|||
|
assert irReqActive=(irReplyLen+1)
|
|||
|
st (a1) ; mark the request as active
|
|||
|
move.l a0,(a3) ; setup waiter
|
|||
|
bra.s @success ; return with success
|
|||
|
|
|||
|
|
|||
|
@RemoveRcvWaiter
|
|||
|
cmpa.l (a3),a0 ; see if we are the waiter
|
|||
|
bne.s @paramError ; return with error if not
|
|||
|
assert irReqActive=(irReplyLen+1)
|
|||
|
clr.b (a1) ; mark the request as complete
|
|||
|
clr.l (a3) ; setup waiter
|
|||
|
bra.s @success ; return with success
|
|||
|
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - IOP Move Data'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPMoveData
|
|||
|
; Inputs: A0 - pointer to IOPMoveInfo paramater block
|
|||
|
; Outputs: D0 - Result Code (NoErr/paramErr)
|
|||
|
; Destroys: A0, A1, D0, D1, D2
|
|||
|
; Calls: CopyToIOP, CopyFromIOP, CompareWithIop
|
|||
|
; Called by: OsTrap Dispatch Table
|
|||
|
;
|
|||
|
; Function: Moves, or compares data between the IOP and the Host
|
|||
|
; memories, using the parameters inthe IOPMoveInfo parameter
|
|||
|
; block. Also has a special mode to apply patches to IOP
|
|||
|
; memory as an atomic operation.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPMoveInfo
|
|||
|
IOPMoveData
|
|||
|
@SavedRegs reg a2/a3 ; a0-a1/d1-d2 saved by OsTrap dispatch
|
|||
|
movem.l @SavedRegs,-(sp) ; save registers
|
|||
|
moveq.l #paramErr,d0 ; assume Parameter Error status
|
|||
|
moveq.l #0,d2 ; prepare to zero extend
|
|||
|
|
|||
|
assert imCopyKind=0
|
|||
|
move.b (a0)+,d2 ; d2 <- zero extended Copy Kind
|
|||
|
cmpi.w #imPatchIop,d2 ; range check the copy kind
|
|||
|
bhi.s @paramError ; if out of range
|
|||
|
move.w @JumpTable(d2.w*2),d1 ; d1 := offset to copy routine
|
|||
|
|
|||
|
assert imIOPNumber=(imCopyKind+1)
|
|||
|
move.b (a0)+,d2 ; d2 <- zero extended IOP number
|
|||
|
cmpi.w #MaxIopNum,d2 ; range check the iop number
|
|||
|
bhi.s @paramError ; if out of range
|
|||
|
|
|||
|
movea.l IOPmgrVars,a1 ; get message table base
|
|||
|
movea.l IOPMgrGlobals.IOPInfoPtrs(a1,d2.w*4),a2 ; get iop info base address
|
|||
|
move.l a2,d2 ; test for zero
|
|||
|
beq.s @paramError ; IOP isn't initialized
|
|||
|
|
|||
|
movea.l IOPInfo.IopDataRegPtr(a2),a3 ; (a3) := IOPRamDataReg
|
|||
|
movea.l IOPInfo.IopAddrRegPtr(a2),a2 ; (a2) := IOPRamAddrReg
|
|||
|
|
|||
|
moveq.l #0,d0 ; zero extend to long
|
|||
|
assert imByteCount=(imIOPNumber+1)
|
|||
|
move.w (a0)+,d0 ; d0 := byte count
|
|||
|
|
|||
|
assert imHostAddr=(imByteCount+2)
|
|||
|
movea.l (a0)+,a1 ; a1 := host buffer address
|
|||
|
move.w (a2),-(sp) ; save the IOPRamAddrReg
|
|||
|
assert imIopAddr=(imHostAddr+4)
|
|||
|
move.w (a0)+,(a2) ; setup iop ram address
|
|||
|
assert imCompRel=(imIopAddr+2) ; a0 pointer to imCompRel for CompareWithIop
|
|||
|
jsr @JumpTable(d1.w) ; do the copy
|
|||
|
move.w (sp)+,(a2) ; restore the IOPRamAddrReg
|
|||
|
moveq.l #noErr,d0 ; return No Error status
|
|||
|
|
|||
|
@paramError
|
|||
|
@return movem.l (sp)+,@SavedRegs ; restore registers
|
|||
|
rts
|
|||
|
|
|||
|
@JumpTable assert imIopToHost=((*-@JumpTable)/2)
|
|||
|
dc.w CopyFromIop-@JumpTable
|
|||
|
|
|||
|
assert imHostToIop=((*-@JumpTable)/2)
|
|||
|
dc.w CopyToIop-@JumpTable
|
|||
|
|
|||
|
assert imCompare=((*-@JumpTable)/2)
|
|||
|
dc.w CompareWithIop-@JumpTable
|
|||
|
|
|||
|
assert imPatchIop=((*-@JumpTable)/2)
|
|||
|
dc.w @Patch-@JumpTable
|
|||
|
|
|||
|
|
|||
|
@Patch move.w sr,d2 ; d2 := saved interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; disable interrupts
|
|||
|
move.w #PatchReqAddr,d1 ; address of Patch Request Byte
|
|||
|
move.w d1,(a2) ; setup the IOP address
|
|||
|
moveq.l #NewMsgSent,d0 ; patch request state byte
|
|||
|
move.b d0,(a3) ; issue the patch request
|
|||
|
|
|||
|
@PatchSync move.w d1,(a2) ; setup the IOP address
|
|||
|
cmp.b (a3),d0 ; see if patch request was accepted
|
|||
|
beq.s @PatchSync ; wait until it is accepted
|
|||
|
|
|||
|
@PatchSegLoop
|
|||
|
move.b (a1)+,d0 ; get the size of the segment
|
|||
|
beq.s @PatchDone ; size of zero terminates segment list
|
|||
|
move.w (a1)+,(a2) ; setup the patch load address
|
|||
|
|
|||
|
@PatchLoadLoop
|
|||
|
move.b (a1)+,(a3) ; patch a byte
|
|||
|
subq.b #1,d0 ; decrement the loop count
|
|||
|
bne.s @PatchLoadLoop ; load all of the bytes
|
|||
|
bra.s @PatchSegLoop ; load the next segment
|
|||
|
|
|||
|
@PatchDone move.w d1,(a2) ; setup the IOP address
|
|||
|
move.b #MsgIdle,(a3) ; tell the IOP that patching is done
|
|||
|
move.w d2,sr ; restore interrupt priority
|
|||
|
rts ; patching is complete
|
|||
|
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - Copy From IOP'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: CopyFromIop
|
|||
|
; Inputs: A1 - pointer to Host RAM buffer (dest addr)
|
|||
|
; A2 - pointer to IOPRamAddrReg
|
|||
|
; (A2) - IOP data address (source addr)
|
|||
|
; A3 - pointer to IOPRamDataReg
|
|||
|
; D0 - (word) transfer byte count (zero extended to long)
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A1, D0, D1
|
|||
|
; Calls: none
|
|||
|
; Called by: IOPInterrupt, IOPMoveData
|
|||
|
;
|
|||
|
; Function: Moves data from the IOP to the Host memory.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
CopyFromIop cmpi.w #3,d0 ; check for very short copy
|
|||
|
bls.s @veryShort ; skip alignment if very short
|
|||
|
move.l a1,d1 ; get the Host Address
|
|||
|
andi.w #$0003,d1 ; check for long word alignment
|
|||
|
beq.s @Aligned ; if no alignment needed
|
|||
|
subq.w #4,d1 ; get byte count bias
|
|||
|
add.w d1,d0 ; adjust the byte count
|
|||
|
IF forSTP601v1 THEN
|
|||
|
jmp @Aligned(d1.w*4) ; ZZZdo the alignment
|
|||
|
ELSE
|
|||
|
jmp @Aligned(d1.w*2) ; do the alignment
|
|||
|
ENDIF
|
|||
|
move.b (a3),(a1)+ ; move a byte
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
move.b (a3),(a1)+ ; move a byte
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
move.b (a3),(a1)+ ; move a byte
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ELSE
|
|||
|
move.b (a3),(a1)+ ; move a byte
|
|||
|
move.b (a3),(a1)+ ; move a byte
|
|||
|
ENDIF
|
|||
|
@Aligned ror.l #2,d0 ; save tail byte count in high 2 bits
|
|||
|
moveq.l #7,d1 ; mask for starting index
|
|||
|
and.w d0,d1 ; number of long words to move first
|
|||
|
neg.w d1 ; negate to index backwards
|
|||
|
lsr.w #3,d0 ; number of 32 byte blocks to move
|
|||
|
jmp @CopyStart(d1.w*2) ; jump into the loop
|
|||
|
|
|||
|
@CopyLoop move.l (a3),(a1)+ ; move a 32 byte block of data....
|
|||
|
move.l (a3),(a1)+ ; ... 4 bytes at a time
|
|||
|
move.l (a3),(a1)+
|
|||
|
move.l (a3),(a1)+
|
|||
|
move.l (a3),(a1)+
|
|||
|
move.l (a3),(a1)+
|
|||
|
move.l (a3),(a1)+
|
|||
|
move.l (a3),(a1)+
|
|||
|
@CopyStart dbra d0,@CopyLoop ; loop through all of the blocks
|
|||
|
|
|||
|
add.l d0,d0 ; load c and n with remaining 2 bits
|
|||
|
bcs.s @Copy2or3bytes
|
|||
|
@Copy0or1byte
|
|||
|
bpl.s @exit ; if no bytes left
|
|||
|
move.b (a3),(a1) ; copy the last byte
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
@exit rts ; all done
|
|||
|
|
|||
|
@veryShort roxr.l #2,d0 ; load c and n with remaining 2 bits
|
|||
|
bcc.s @Copy0or1byte
|
|||
|
@Copy2or3bytes
|
|||
|
bpl.s @Copy2bytes ; if only 2 bytes left
|
|||
|
move.b (a3),(a1)+ ; copy 1 byte, 2 still remaining
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
@Copy2bytes move.w (a3),(a1) ; copy the last 2 bytes
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #2, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
rts ; all done
|
|||
|
TITLE 'IOPmgr - Copy To IOP'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: CopyToIop
|
|||
|
; Inputs: A1 - pointer to Host RAM buffer (source addr)
|
|||
|
; A2 - pointer to IOPRamAddrReg
|
|||
|
; (A2) - IOP data address (dest addr)
|
|||
|
; A3 - pointer to IOPRamDataReg
|
|||
|
; D0 - (word) transfer byte count (zero extended to long)
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A1, D0, D1
|
|||
|
; Calls: none
|
|||
|
; Called by: IOPMsgAccess, IOPMoveData
|
|||
|
;
|
|||
|
; Function: Moves data from the Host to the IOP memory.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
CopyToIop cmpi.w #3,d0 ; check for very short copy
|
|||
|
bls.s @veryShort ; skip alignment if very short
|
|||
|
move.l a1,d1 ; get the Host Address
|
|||
|
andi.w #$0003,d1 ; check for long word alignment
|
|||
|
beq.s @Aligned ; if no alignment needed
|
|||
|
subq.w #4,d1 ; get byte count bias
|
|||
|
add.w d1,d0 ; adjust the byte count
|
|||
|
jmp @Aligned(d1.w*2) ; do the alignment
|
|||
|
move.b (a1)+,(a3) ; move a byte
|
|||
|
move.b (a1)+,(a3) ; move a byte
|
|||
|
move.b (a1)+,(a3) ; move a byte
|
|||
|
@Aligned ror.l #2,d0 ; save tail byte count in high 2 bits
|
|||
|
moveq.l #7,d1 ; mask for starting index
|
|||
|
and.w d0,d1 ; number of long words to move first
|
|||
|
neg.w d1 ; negate to index backwards
|
|||
|
lsr.w #3,d0 ; number of 32 byte blocks to move
|
|||
|
jmp @CopyStart(d1.w*2) ; jump into the loop
|
|||
|
|
|||
|
@CopyLoop move.l (a1)+,(a3) ; move a 32 byte block of data....
|
|||
|
move.l (a1)+,(a3) ; ... 4 bytes at a time
|
|||
|
move.l (a1)+,(a3)
|
|||
|
move.l (a1)+,(a3)
|
|||
|
move.l (a1)+,(a3)
|
|||
|
move.l (a1)+,(a3)
|
|||
|
move.l (a1)+,(a3)
|
|||
|
move.l (a1)+,(a3)
|
|||
|
@CopyStart dbra d0,@CopyLoop ; loop through all of the blocks
|
|||
|
|
|||
|
add.l d0,d0 ; load c and n with remaining 2 bits
|
|||
|
bcs.s @Copy2or3bytes
|
|||
|
@Copy0or1byte
|
|||
|
bpl.s @exit ; if no bytes left
|
|||
|
move.b (a1),(a3) ; copy the last byte
|
|||
|
@exit rts ; all done
|
|||
|
|
|||
|
@veryShort roxr.l #2,d0 ; load c and n with remaining 2 bits
|
|||
|
bcc.s @Copy0or1byte
|
|||
|
@Copy2or3bytes
|
|||
|
bpl.s @Copy2bytes ; if only 2 bytes left
|
|||
|
move.b (a1)+,(a3) ; copy 1 byte, 2 still remaining
|
|||
|
@Copy2bytes move.w (a1),(a3) ; copy the last 2 bytes
|
|||
|
rts ; all done
|
|||
|
TITLE 'IOPmgr - Compare With IOP'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: CompareWithIop
|
|||
|
; Inputs: A0 - pointer to imCompRel byte (result addr)
|
|||
|
; A1 - pointer to Host RAM buffer (dest addr)
|
|||
|
; A2 - pointer to IOPRamAddrReg
|
|||
|
; (A2) - IOP data address (source addr)
|
|||
|
; A3 - pointer to IOPRamDataReg
|
|||
|
; D0 - (word) transfer byte count (zero extended to long)
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A1, D0, D1, D2
|
|||
|
; Calls: none
|
|||
|
; Called by: IOPMoveData
|
|||
|
;
|
|||
|
; Function: Compares data from the IOP to the Host memory.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
CompareWithIop
|
|||
|
cmpi.w #3,d0 ; check for very short compare
|
|||
|
bls.s @Aligned ; skip alignment if very short
|
|||
|
move.l a1,d1 ; get the Host Address
|
|||
|
andi.w #$0003,d1 ; check for long word alignment
|
|||
|
beq.s @Aligned ; if no alignment needed
|
|||
|
subq.w #4,d1 ; get byte count bias
|
|||
|
add.w d1,d0 ; adjust the byte count
|
|||
|
not.w d1 ; setup count for DBNE
|
|||
|
@AlignLoop move.b (a3),d2 ; get byte of data from IOP
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
cmp.b (a1)+,d2 ; compare to HOST data
|
|||
|
dbne d1,@AlignLoop ; loop until longword aligned
|
|||
|
bne.s @NotEqual ; exit if didn't compare
|
|||
|
|
|||
|
@Aligned moveq.l #3,d1 ; mask for byte remainder
|
|||
|
and.w d0,d1 ; number of tail bytes to compare
|
|||
|
lsr.w #2,d0 ; number of long words to compare
|
|||
|
beq.s @TailStart ; if no longs, start with tail
|
|||
|
subq.w #1,d0 ; adjust loop count for DBNE
|
|||
|
@CmpLoop move.l (a3),d2 ; get long word of data from IOP
|
|||
|
cmp.l (a1)+,d2 ; compare to HOST data
|
|||
|
dbne d0,@CmpLoop ; loop through all long words
|
|||
|
bra.s @TailStart ; then compare the remaining bytes
|
|||
|
|
|||
|
@TailLoop move.b (a3),d2 ; get byte of data from IOP
|
|||
|
IF forSTP601v1 THEN
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
ENDIF
|
|||
|
cmp.b (a1)+,d2 ; compare to HOST data
|
|||
|
@TailStart dbne d1,@TailLoop ; loop through all remaining bytes
|
|||
|
|
|||
|
@NotEqual sne.b (a0) ; return -1 if IOP < HOST
|
|||
|
bls.s @Exit ; return 0 if IOP = HOST
|
|||
|
neg.b (a0) ; return +1 if IOP > HOST
|
|||
|
@Exit rts
|
|||
|
TITLE 'IOPmgr - IOP Interrupt'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPInterrupt
|
|||
|
; Inputs: D0{31<33>24} - flags indicating non-deferrable XMT messages
|
|||
|
; D0{23<32>16} - flags indicating non-deferrable RCV messages
|
|||
|
; D0{15<31>8} - must be zero
|
|||
|
; D0{7<>0} - IOP number of IOP requesting the interrupt
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A0, A1, A2, A3, D0, D1, D2, D3
|
|||
|
; Calls: DTInstall, IOP message Handler
|
|||
|
; Called by: System interrupt handler
|
|||
|
;
|
|||
|
; Function: Services interrupts from the IOPs. Acknowledges the messages,
|
|||
|
; and calls the message handlers directly, or schedules a
|
|||
|
; deferred task manager task to call them, if they were
|
|||
|
; deferrable handlers. If the interrupt was a Bypass Interrupt
|
|||
|
; Request, it will reset the interrupt and JMP to the external
|
|||
|
; label IOPBypassInterrupt passing the IOP number that generated
|
|||
|
; the interrupt in register D0.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPMgrGlobals,IOPInfo
|
|||
|
|
|||
|
IOPInterrupt ; a0-a3/d0-d3 saved by int handler
|
|||
|
movea.l IOPmgrVars,a0 ; point to globals
|
|||
|
jmp ([IntHandlerPtr,a0]) ; jump through the patch vector
|
|||
|
vIOPInterrupt
|
|||
|
move.l IOPInfoPtrs(a0,d0.w*4),d1
|
|||
|
beq.s @noIop ; ignore if IOP doesn't exist
|
|||
|
movea.l d1,a0 ; a0 := pointer to IOPInfo
|
|||
|
assert IopAddrRegPtr=0
|
|||
|
movea.l (a0)+,a2 ; a2 := pointer to IOP Ram addr reg
|
|||
|
assert IopDataRegPtr=(IopAddrRegPtr+4)
|
|||
|
movea.l (a0)+,a3 ; a3 := pointer to IOP Ram data reg
|
|||
|
move.w sr,d2 ; save old interrupt level
|
|||
|
move.b d0,d2 ; save the IOP number, overwrite saved ccr
|
|||
|
move.l a0,-(sp) ; save reg for msg handler to restore
|
|||
|
@checkAgain
|
|||
|
assert IopCtlRegPtr=(IopDataRegPtr+4)
|
|||
|
movea.l (a0)+,a1 ; a1 := pointer to IOP control reg
|
|||
|
move.b (a1),d1 ; get the control register
|
|||
|
lsr.b #iopInt0Active+1,d1 ; test INT0 (xmt MessageCompleted)
|
|||
|
bcs.s @xmtComp ; handle INT0
|
|||
|
lsr.b #iopInt1Active-iopInt0Active,d1 ; test INT1 (rcv NewMessageSent)
|
|||
|
bcs.s @rcvNew ; handle INT1
|
|||
|
lsr.b #iopBypassIntReq-iopInt1Active,d1 ; test BypassIntReq
|
|||
|
bcs.s @bypassInt ; all done if not bypass interrupt
|
|||
|
@IntHandled move.w d2,sr ; restore interrupt level
|
|||
|
addq.l #4,sp ; pop saved a0
|
|||
|
lsr.l #8,d0 ; test the flags
|
|||
|
bne.w @RunNonDeferrable ; if non-deferrables, go check for and run them
|
|||
|
rts ; return from the interrupt <1.6>
|
|||
|
|
|||
|
@noIop cmpi.w #SccIopNum,d0 ; see if it was from the SCC <1.6>
|
|||
|
beq.l SccDecode ; if so, pass it on to the Bypass handler <1.6> <8> rb
|
|||
|
rts ; return from the interrupt <1.6>
|
|||
|
|
|||
|
|
|||
|
@bypassInt
|
|||
|
assert BypassHandler=(IopCtlRegPtr+4)
|
|||
|
move.l (a0),d1 ; get address of bypass int handler
|
|||
|
beq.s @IntHandled ; no bypass handler, not much we can do
|
|||
|
move.w d2,sr ; restore interrupt level
|
|||
|
move.l d0,(sp) ; save the flags/iop number
|
|||
|
movea.l d1,a0 ; setup handler address
|
|||
|
jsr (a0) ; call the ByPass interrupt handler
|
|||
|
move.l (sp)+,d0 ; restore flags/IOP number
|
|||
|
bra.s IOPInterrupt ; check again for other interrupts
|
|||
|
|
|||
|
|
|||
|
@rcvNew move.b #clrIopInt1,(a1) ; clear the interrupt
|
|||
|
move.w #(MsgReceived<<8)+\ ; new state
|
|||
|
(NewMsgSent<<0),d1 ; state to search for
|
|||
|
move.w #IOPRcvMsgBase+1,d3 ; IOP address to start search at
|
|||
|
lea @rcvFound,a1 ; a1 := rcv found routine
|
|||
|
assert BypassHandler=(IopCtlRegPtr+4)
|
|||
|
move.b MaxRcv-BypassHandler(a0),d0 ; get message count
|
|||
|
adda.w #MsgTable-BypassHandler,a0 ; point before the first message
|
|||
|
move.w (a2),-(sp) ; save the IOPRamAddrReg
|
|||
|
bra.s @SearchStart ; start searching the message states
|
|||
|
|
|||
|
@xmtComp move.b #clrIopInt0,(a1) ; clear the interrupt
|
|||
|
moveq.l #(MsgIdle<<8)+\ ; new state
|
|||
|
(MsgCompleted<<0),d1 ; state to search for
|
|||
|
move.w #IOPXmtMsgBase+1,d3 ; IOP address to start search at
|
|||
|
lea @xmtFound,a1 ; a1 := xmt done routine
|
|||
|
assert BypassHandler=(IopCtlRegPtr+4)
|
|||
|
move.b MaxXmt-BypassHandler(a0),d0 ; get message count
|
|||
|
adda.w #MsgTable-BypassHandler,a0 ; point before the first message
|
|||
|
move.w (a2),-(sp) ; save the IOPRamAddrReg
|
|||
|
bra.s @SearchStart ; start searching the message states
|
|||
|
|
|||
|
@searchLoop adda.w #IOPMsgEntry.IOPMsgEntrySize,a0 ; point to next message info
|
|||
|
cmp.b (a3),d1 ; check for state match
|
|||
|
IF forSTP601v1 THEN
|
|||
|
move ccr, -(sp) ;ZZZ
|
|||
|
subq.w #3, (a2) ;ZZZ
|
|||
|
move (sp)+, ccr ;ZZZ
|
|||
|
ENDIF
|
|||
|
dbeq d0,@searchLoop ; loop until match or end of list
|
|||
|
bne.s @searchDone ; if all message states checked
|
|||
|
|
|||
|
move.w (a2),d3 ; get the address pointer
|
|||
|
@MsgRegs reg a0/a1/d0/d1/d2/d3 ; registers to save when servicing message
|
|||
|
movem.l @MsgRegs,-(sp) ; save the registers
|
|||
|
subq.w #1,d3 ; point to the message state that was found
|
|||
|
move.w d3,(a2) ; re-load ram address pointer
|
|||
|
lsr.w #8,d1 ; get the new message state
|
|||
|
move.b d1,(a3) ; set the new message state
|
|||
|
move.w d2,sr ; give higher priority ints a chance
|
|||
|
jsr (a1) ; call the xmt/rcv found routine
|
|||
|
movem.l (sp)+,@MsgRegs ; restore the registers
|
|||
|
@SearchStart
|
|||
|
ori.w #HiIntMask,sr ; disable all interrupts
|
|||
|
move.w d3,(a2) ; restore the address pointer
|
|||
|
dbra d0,@searchLoop ; loop until end of list
|
|||
|
|
|||
|
@searchDone move.w (sp)+,(a2) ; restore the IOPRamAddrReg
|
|||
|
clr.w d0 ; setup for zero extended IOP number
|
|||
|
move.b d2,d0 ; flags in high word, IOP number in low word
|
|||
|
movea.l IOPmgrVars,a1
|
|||
|
tst.b DTaskQueued(a1)
|
|||
|
bne.s @toCkAgain ; if already queued, just return
|
|||
|
tst.l CompleteQHdr+qHead(a1)
|
|||
|
beq.s @toCkAgain ; if nothing queued, just return
|
|||
|
st DTaskQueued(a1)
|
|||
|
lea DTask(a1),a0
|
|||
|
@DTaskRegs reg a2/d0/d2 ; regs to save when queueing a deferred task
|
|||
|
movem.l @DTaskRegs,-(sp) ; save regs
|
|||
|
jsr ([jDTInstall]) ; queue the deferred task
|
|||
|
movem.l (sp)+,@DTaskRegs ; restore regs
|
|||
|
@toCkAgain move.w d2,sr ; give higher priority ints a chance <1.7>
|
|||
|
movea.l (sp),a0 ; setup a0 for entry at @checkAgain
|
|||
|
bra.w @checkAgain ; return from the interrupt <1.7>
|
|||
|
|
|||
|
@XmtFound lea IOPMsgEntry.XmtMsgQHdr+qHead(a0),a1 ; point to request info
|
|||
|
move.l (a1),d1 ; get request info
|
|||
|
beq.s @empty ; if no handler installed
|
|||
|
move.l a1,-(sp) ; save message entry pointer
|
|||
|
movea.l d1,a0 ; setup request info address
|
|||
|
subq.l #qHead,a1 ; point to queue header
|
|||
|
jsr DequeueTrap ; remove it from the msg queue
|
|||
|
movea.l IOPRequestInfo.irReplyPtr(a0),a1 ; get the buffer address
|
|||
|
moveq.l #0,d0 ; zero extend the reply byte count
|
|||
|
move.b IOPRequestInfo.irReplyLen(a0),d0 ; get the byte count
|
|||
|
bsr.s @CopyAndQueue ; read the reply and queue it
|
|||
|
movea.l (sp)+,a1 ; point to message queue head
|
|||
|
move.l (a1),d1 ; get head of queue
|
|||
|
beq.s @empty ; if empty, do nothing
|
|||
|
movea.l d1,a0 ; setup to send the queued message
|
|||
|
jmp ([JIOPMsgRequest]) ; send the next message and return
|
|||
|
@empty rts
|
|||
|
|
|||
|
@RcvFound move.l IOPMsgEntry.RcvMsgInfoPtr(a0),d1 ; get request info
|
|||
|
beq.s @empty ; if no handler installed
|
|||
|
movea.l d1,a0 ; setup request info address
|
|||
|
movea.l IOPRequestInfo.irMessagePtr(a0),a1 ; get the buffer address
|
|||
|
moveq.l #0,d0 ; zero extend the message byte count
|
|||
|
move.b IOPRequestInfo.irMessageLen(a0),d0 ; get the byte count
|
|||
|
@CopyAndQueue
|
|||
|
beq.s @CopyDone ; if no message data to copy
|
|||
|
move.w d3,d1 ; get copy of message state address
|
|||
|
assert MaxIOPMsgLen=(1<<5)
|
|||
|
lsl.b #5,d1 ; convert to message data address
|
|||
|
move.w d1,(a2) ; point to the message
|
|||
|
bsr.w CopyFromIop ; read the message
|
|||
|
@CopyDone movea.l IOPmgrVars,a1 ; point to globals
|
|||
|
adda.w #CompleteQHdr,a1 ; point to the completion queue
|
|||
|
tst.l IOPRequestInfo.irHandler(a0) ; check the handler address
|
|||
|
beq.s @NoHandler ; if no handler, don't enqueue it
|
|||
|
jmp EnqueueTrap ; enqueue the completed request, and return
|
|||
|
|
|||
|
@NoHandler clr.b IOPRequestInfo.irReqActive(a0) ; indicate that it is done
|
|||
|
rts ; return
|
|||
|
|
|||
|
|
|||
|
@RunNonDeferrable
|
|||
|
move.l d0,d3 ; save the flags
|
|||
|
lsr.l #8,d3 ; get flags into low 2 bytes
|
|||
|
movea.l IOPmgrVars,a1 ; get the globals
|
|||
|
lea CompleteQHdr(a1),a1 ; point to the completion queue
|
|||
|
lea qHead-qLink(a1),a0 ; point to ptr to first element
|
|||
|
ori.w #HiIntMask,sr ; disable all ints while touching queue
|
|||
|
@RunLoop move.l qLink(a0),d0 ; get ptr to next element
|
|||
|
beq.s @RunDone ; if end of queue, we're done
|
|||
|
movea.l d0,a0 ; setup queue element pointer
|
|||
|
cmp.b IOPRequestInfo.irIOPNumber(a0),d2 ; is it for this IOP
|
|||
|
bne.s @RunLoop ; if not, go on to next one
|
|||
|
move.b IOPRequestInfo.irMsgNumber(a0),d0 ; get the message number
|
|||
|
assert IOPRequestInfo.irSendXmtMessage=0
|
|||
|
tst.b IOPRequestInfo.irRequestKind(a0) ; test for xmt/rcv
|
|||
|
bne.s @ChkMsgNum ; if receive, bit number is correct
|
|||
|
addq.b #8,d0 ; point to next byte for xmt
|
|||
|
@ChkMsgNum btst.l d0,d3 ; see if this msg is non-deferrable
|
|||
|
beq.s @RunLoop ; if not, go on to next one
|
|||
|
jsr DequeueTrap ; remove it from the queue
|
|||
|
move.l a0,-(sp) ; remember the address
|
|||
|
bsr.s @RunLoop ; check the next one (using recursion)
|
|||
|
|
|||
|
movea.l (sp)+,a0 ; get the queue element address
|
|||
|
clr.b IOPRequestInfo.irReqActive(a0) ; indicate that it is done
|
|||
|
move.l IOPRequestInfo.irHandler(a0),d0 ; get the handler address
|
|||
|
beq.s @RunRTS ; if no handler, don't try to call it
|
|||
|
movea.l d0,a1 ; get handler address
|
|||
|
jmp (a1) ; run the handler (may trash a0-a3/d0-d3)
|
|||
|
|
|||
|
@RunDone move.w d2,sr ; restore interrupt level
|
|||
|
@RunRTS rts ; search done, now let them run
|
|||
|
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - IOP Deferred Task'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPDefTask
|
|||
|
; Inputs: none
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A0, A1, D0
|
|||
|
; Calls: IOP message Handler
|
|||
|
; Called by: Deferred task manager
|
|||
|
;
|
|||
|
; Function: Calls the message handlers for the deferrable messages
|
|||
|
; that were marked and acknowledged by the IOP interrupt
|
|||
|
; handler.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
CallCompHandler
|
|||
|
move.w (sp)+,sr ; restore interrupts
|
|||
|
movea.l d0,a0 ; setup element to dequeue
|
|||
|
jsr DequeueTrap ; dequeue the element
|
|||
|
clr.b IOPRequestInfo.irReqActive(a0) ; indicate that it is done
|
|||
|
move.l IOPRequestInfo.irHandler(a0),d0 ; get the handler address
|
|||
|
beq.s IOPDefTask ; if no handler, don't try to call it
|
|||
|
|
|||
|
move.l a1,-(sp) ; save queue pointer across call to handler
|
|||
|
movea.l d0,a1 ; get handler address
|
|||
|
jsr (a1) ; call the handler (may trash a0-a3/d0-d3)
|
|||
|
movea.l (sp)+,a1 ; restore queue pointer
|
|||
|
|
|||
|
IOPDefTask ; Enters HERE ; a0-a3/d0-d3 saved by int handler
|
|||
|
; a1 := pointer to CompleteQHdr
|
|||
|
move.w sr,-(sp) ; save interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; disable interrupts
|
|||
|
move.l qHead(a1),d0 ; read the head of the queue
|
|||
|
bne.s CallCompHandler ; if not empty, dequeue and call it
|
|||
|
|
|||
|
assert IOPMgrGlobals.DTaskQueued=(IOPMgrGlobals.CompleteQHdr-1)
|
|||
|
clr.b -(a1) ; indicate that the task is done
|
|||
|
move.w (sp)+,sr ; restore interrupts
|
|||
|
rts ; all done
|
|||
|
if Debugging then ; <1.3>
|
|||
|
TITLE 'IOPmgr - IOP VBL Task'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: IOPVblTask
|
|||
|
; Inputs: none
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A0, A1, A2, D0
|
|||
|
; Calls: none
|
|||
|
; Called by: Vertical Retrace Manager
|
|||
|
;
|
|||
|
; Function: Tests and clears the Alive flag on each IOP, and crashes
|
|||
|
; if any IOP was not Alive
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPMgrGlobals,IOPInfo
|
|||
|
IOPVblTask ; a0-a3/d0-d3 saved by int handler
|
|||
|
movea.l IOPmgrVars,a0 ; get globals
|
|||
|
move.w #AliveVBLrate,VTask+vblCount(a0) ; re-initialize the count
|
|||
|
move.w sr,-(sp) ; save interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; disable interrupts
|
|||
|
moveq.l #NumberOfIOPs-1,d0 ; loop counter
|
|||
|
@PollLoop movea.l (a0)+,a1 ; a1 := pointer to IOPInfo
|
|||
|
move.l a1,d1 ; test for zero
|
|||
|
beq.s @Next ; skip if IOP doesn't exist
|
|||
|
assert IopAddrRegPtr=0
|
|||
|
movea.l (a1)+,a2 ; a2 := IopAddrRegPtr
|
|||
|
assert IopDataRegPtr=(IopAddrRegPtr+4)
|
|||
|
movea.l (a1)+,a1 ; a1 := IopDataRegPtr
|
|||
|
move.w (a2),-(sp) ; save the IOPRamAddrReg
|
|||
|
move.w #IOPAliveAddr,(a2) ; setup the IOP Ram Address
|
|||
|
cmpi.b #$FF,(a1) ; IOP sets it to $FF when Idle
|
|||
|
beq.s @Alive ; if alive and well
|
|||
|
bsr.s IOPisDead ; crash if dead
|
|||
|
@Alive move.w #IOPAliveAddr,(a2) ; setup the IOP Ram Address
|
|||
|
clr.b (a1) ; we set it to $00 after testing it
|
|||
|
move.w (sp)+,(a2) ; restore the IOPRamAddrReg
|
|||
|
@Next dbra d0,@PollLoop ; poll the next IOP
|
|||
|
move.w (sp)+,sr ; restore interrupt priority
|
|||
|
rts ; return
|
|||
|
|
|||
|
|
|||
|
IOPisDead move.w #dsSysErr,d0 ; Sorry, a system error occurred
|
|||
|
_SysError ; crash
|
|||
|
bra.s IOPisDead ; IOP failure, crash and burn
|
|||
|
|
|||
|
endwith
|
|||
|
endif ; <1.3>
|
|||
|
TITLE 'IOPmgr - Move Request Handler'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: MoveReqHandler
|
|||
|
; Inputs: A0 - pointer to IOPRequestInfo
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A0-A2/D0-D2
|
|||
|
; Calls: IOPMsgRequest, IOPMoveData
|
|||
|
; Called by: IOP interrupt handler
|
|||
|
;
|
|||
|
; Function: Handles Data Movement requests issued by drivers on the IOPs.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPRequestInfo,IOPMoveInfo
|
|||
|
MoveReqHandler ; a0-a3/d0-d3 saved by Interrupt Handler
|
|||
|
|
|||
|
; The message that the IOP sent looks very much like a IOPMoveInfo parameter
|
|||
|
; block, the message from the IOP was read directly into the IOPMoveInfo block.
|
|||
|
|
|||
|
movea.l a0,a2 ; save pointer to IOPRequestInfo for later
|
|||
|
movea.l irMessagePtr(a0),a0 ; message buffer is IOPMoveInfo request block
|
|||
|
|
|||
|
; The IOP number field is the only field that was not in the message from
|
|||
|
; the IOP, so just copy it from the IOPAccessInfo, and then issue the Move.
|
|||
|
|
|||
|
move.b irIOPNumber(a2),imIOPNumber(a0) ; fill in the IOP number
|
|||
|
jsr ([JIOPMoveData]) ; Perform the Move Request
|
|||
|
|
|||
|
; Send the result from IOPMoveData back as the reply.
|
|||
|
; Send the Message Completed interrupt to the IOP, and return.
|
|||
|
|
|||
|
movea.l a2,a0 ; a0 <- IOPRequestInfo
|
|||
|
move.b #irSendRcvReply,irRequestKind(a0)
|
|||
|
jmp ([JIOPMsgRequest]) ; tell IOP that Message Completed and return
|
|||
|
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - Initialize IOP Manager'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: jsr InitIOPMgr
|
|||
|
; Inputs: none
|
|||
|
; Outputs: none
|
|||
|
; Destroys: none
|
|||
|
; Calls: none
|
|||
|
; Called by: Start Manager
|
|||
|
;
|
|||
|
; Function: Initializes the IOPs, downloads their code, and starts them
|
|||
|
; running. Allocates and initializes the IOPMgr global data
|
|||
|
; structures.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPMgrGlobals,IOPAccessInfo,IOPInfo
|
|||
|
InitIOPMgr
|
|||
|
@SavedRegs reg a0-a3/d0
|
|||
|
movem.l @SavedRegs,-(sp) ; save registers
|
|||
|
|
|||
|
move.w sr,-(sp) ; save interrupt priority
|
|||
|
ori.w #HiIntMask,sr ; disable interrupts
|
|||
|
|
|||
|
moveq.l #0,d0 ; amount of space for IOPInfos
|
|||
|
|
|||
|
TestFor SCCIOPExists ; see if we have an SCC IOP
|
|||
|
beq.s @noSCCIOP ; if not, don't allocate space for it
|
|||
|
|
|||
|
addi.w #IOPInfoSize,d0 ; allocate space for it
|
|||
|
@noSCCIOP
|
|||
|
|
|||
|
TestFor SWIMIOPExists ; see if we have an SWIM IOP
|
|||
|
beq.s @noSWIMIOP ; if not, don't allocate space for it
|
|||
|
|
|||
|
addi.w #IOPInfoSize,d0 ; allocate space for it
|
|||
|
@noSWIMIOP
|
|||
|
|
|||
|
tst.w d0 ; are there any IOPs
|
|||
|
beq.s @noIOPs ; if not, exit
|
|||
|
|
|||
|
addi.w #GlobalsSize,d0 ; d0 <- data structure size
|
|||
|
_NewPtr ,SYS,CLEAR ; allocate and clear the structure
|
|||
|
|
|||
|
lea -IOPMgrGlobals(a0),a2 ; Globals are negative to the pointer
|
|||
|
move.l a2,IOPmgrVars ; initialize the Pointer
|
|||
|
if Debugging then ; <1.3>
|
|||
|
|
|||
|
; Install the VBL task to look for dead IOPs
|
|||
|
|
|||
|
lea VTask(a2),a0 ; a0 <- VBL task
|
|||
|
move.w #vType,qType(a0) ; initialize qType field
|
|||
|
move.w #AliveVBLrate,vblCount(a0) ; initialize the count
|
|||
|
lea IOPVblTask,a1 ; get the task address
|
|||
|
move.l a1,vblAddr(a0) ; initialize vblAddr field
|
|||
|
_VInstall ; install the VBL task
|
|||
|
endif ; <1.3>
|
|||
|
|
|||
|
; Setup the deferred task manager data structure
|
|||
|
|
|||
|
move.w #dtQType,DTask+qType(a2) ; Initialize DTask.qType
|
|||
|
lea IOPDefTask,a0 ; a0 <- addr of IOPDefTask
|
|||
|
move.l a0,DTask+dtAddr(a2) ; Initialize DTask.dtAddr
|
|||
|
lea CompleteQHdr(a2),a0 ; a0 <- completion queue header
|
|||
|
move.l a0,DTask+dtParm(a2) ; DTask.dtParm (loaded into A1 for DTask)
|
|||
|
|
|||
|
; Setup the Interrupt Handler Pointer
|
|||
|
|
|||
|
lea vIOPInterrupt,a0 ; point to the handler
|
|||
|
move.l a0,IntHandlerPtr(a2) ; initialize the data structure
|
|||
|
|
|||
|
; Install and initialize the IOPs
|
|||
|
|
|||
|
lea IOPInfos(a2),a2 ; point to start of IOPInfo records
|
|||
|
|
|||
|
TestFor SCCIOPExists ; see if we have an SCC IOP
|
|||
|
beq.s @SCCIOPdone ; if not, don't install it
|
|||
|
|
|||
|
moveq.l #(iaInstallIOP<<8)|\
|
|||
|
SccIopNum,d0 ; get the IOP number and access kind
|
|||
|
movea.l SCCrd,a1 ; get the base address
|
|||
|
lea SccDecode,a3 ; get the bypass interrupt handler address
|
|||
|
bsr.s @InstallIOP ; install the SCC IOP
|
|||
|
@SCCIOPdone
|
|||
|
|
|||
|
TestFor SWIMIOPExists ; see if we have an SWIM IOP
|
|||
|
beq.s @SWIMIOPdone ; if not, don't install it
|
|||
|
|
|||
|
moveq.l #(iaInstallIOP<<8)|\
|
|||
|
SwimIopNum,d0 ; get the IOP number and access kind
|
|||
|
movea.l IWM,a1 ; get the base address
|
|||
|
suba.l a3,a3 ; no bypass interrupt handler
|
|||
|
bsr.s @InstallIOP ; install the SWIM IOP
|
|||
|
@SWIMIOPdone
|
|||
|
|
|||
|
@Done move.w (sp)+,sr ; restore interrupts
|
|||
|
movem.l (sp)+,@SavedRegs ; restore registers
|
|||
|
rts ; all done
|
|||
|
|
|||
|
@noIOPs lea @traps,a1 ; point to the list
|
|||
|
move.w (a1)+,d0 ; get the _Unimplemented trap
|
|||
|
_GetTrapAddress ,newTool ; get address of unimplemented trap
|
|||
|
@trapLoop move.w (a1)+,d0 ; get next trap to un-implement
|
|||
|
beq.s @Done ; traps disabled, exit
|
|||
|
_SetTrapAddress ,newOS ; un-implement the trap
|
|||
|
bra.s @trapLoop ; loop through the table
|
|||
|
|
|||
|
@traps _Unimplemented
|
|||
|
_IOPInfoAccess
|
|||
|
_IOPMsgRequest
|
|||
|
_IOPMoveData
|
|||
|
dc.w 0 ; end of list
|
|||
|
|
|||
|
@InstallIOP movea.l a2,a0 ; a0 <- IOPInfoRecord
|
|||
|
assert iaIOPInfoPtr=(iaAccessInfoSize-4)
|
|||
|
move.l a0,-(sp) ; setup IOP info pointer
|
|||
|
clr.w -(sp) ; clear reserved field
|
|||
|
assert iaIOPNumber=(iaIOPInfoPtr-3)
|
|||
|
assert iaAccessKind=(iaIOPInfoPtr-4)
|
|||
|
move.w d0,-(sp) ; setup the IOP number and access kind
|
|||
|
|
|||
|
assert IopAddrRegPtr=0
|
|||
|
lea iopRamAddr(a1),a1 ; point to the address register
|
|||
|
move.l a1,(a0)+ ; setup IopAddrRegPtr
|
|||
|
|
|||
|
assert IopDataRegPtr=(IopAddrRegPtr+4)
|
|||
|
addq.w #iopRamData-iopRamAddr,a1 ; point to the data register
|
|||
|
move.l a1,(a0)+ ; setup IopDataRegPtr
|
|||
|
|
|||
|
assert IopCtlRegPtr=(IopDataRegPtr+4)
|
|||
|
subq.w #iopRamData-iopStatCtl,a1 ; point to the status/control register
|
|||
|
move.l a1,(a0)+ ; setup IopCtlRegPtr
|
|||
|
|
|||
|
assert BypassHandler=(IopCtlRegPtr+4)
|
|||
|
move.l a3,(a0)+ ; setup BypassHandler
|
|||
|
|
|||
|
movea.l sp,a0 ; a0 <- IOPAccessInfo record
|
|||
|
_IOPInfoAccess ; install the IOP
|
|||
|
addq.l #iaAccessInfoSize,sp ; de-allocate the IOPAccessInfo record
|
|||
|
adda.w #IOPInfoSize,a2 ; point to next IOPInfo record
|
|||
|
rts ; go on to the next IOP
|
|||
|
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - Set SCC IOP ByPass mode'
|
|||
|
|
|||
|
;_______________________________________________________________________
|
|||
|
;
|
|||
|
; Routine: jsr SCCIOPByPass <1.2>
|
|||
|
; Inputs: none
|
|||
|
; Outputs: none
|
|||
|
; Destroys: none
|
|||
|
; Calls: none
|
|||
|
; Called by: InitIO
|
|||
|
;
|
|||
|
; Function: Checks to see if the SCC IOP should be placed in bypass
|
|||
|
; and set it accordingly.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
with IOPMgrGlobals,IOPAccessInfo,IOPInfo,IOPRequestInfo
|
|||
|
|
|||
|
SCCIOPByPass
|
|||
|
|
|||
|
;
|
|||
|
; Check PRAM to see if we should put the SCC IOP into enhanced communications mode.
|
|||
|
;
|
|||
|
; Bit 0 = 1 -> leave in bypass <2>
|
|||
|
; = 0 -> put in enhanced communications mode <2>
|
|||
|
;
|
|||
|
|
|||
|
@SavedRegs reg a0-a2/d0
|
|||
|
movem.l @SavedRegs,-(sp) ; save registers
|
|||
|
move.l #((01<<16)|IOPPRAM), d0 ; index and number of bytes to get
|
|||
|
clr.b -(sp) ; make room for data
|
|||
|
movea.l sp, a0 ; a0 points to data buffer
|
|||
|
_ReadXPRAM
|
|||
|
|
|||
|
MOVE.B (SP),SCCIOPFlag ; save the PRAM byte in low mem <1.9>
|
|||
|
|
|||
|
moveq.l #$01,d0 ; check the bypass bit
|
|||
|
and.b (sp)+, d0 ; get the PRAM value
|
|||
|
bne.s @Done ; branch if SET, stay in bypass mode <2.0>
|
|||
|
|
|||
|
clr.l -(sp) ; reply buffer
|
|||
|
clr.l -(sp) ; [long] -> Completion Routine Address (Nil)
|
|||
|
pea 4(sp) ; [long] -> Reply Buffer Address
|
|||
|
pea @ByPassMsg ; [long] -> Message Buffer Address
|
|||
|
move.l #((Active<<0)+\ ; [byte] -> Request active
|
|||
|
(ReplyLen<<8)+\ ; [byte] -> Reply Length
|
|||
|
(MsgLen<<16)+\ ; [byte] -> Message Length
|
|||
|
(KernMsg<<24)), -(sp) ; [byte] -> Message Number
|
|||
|
move.l #((irSendXmtMessage<<0)+\ ; [byte] -> kind of request to perform
|
|||
|
(SccIopNum<<8)), -(sp) ; [byte] -> IOP Number for SCC
|
|||
|
; [word] -> queue type
|
|||
|
clr.l -(sp) ; [long] -> link to next queue element
|
|||
|
movea.l sp, a0
|
|||
|
_IOPMsgRequest
|
|||
|
bne.s @FlagBypass ; if send failed, don't wait <1.5><2>
|
|||
|
|
|||
|
@Sync_Loop tst.b irReqActive(sp) ; make this a sync call
|
|||
|
bne.s @Sync_Loop
|
|||
|
|
|||
|
@waitDone adda.w #(irReqInfoSize+4), sp ; adjust stack <1.5>
|
|||
|
|
|||
|
@Done movem.l (sp)+,@SavedRegs ; restore registers
|
|||
|
rts
|
|||
|
|
|||
|
@FlagBypass BSET #0,SCCIOPFlag ;set bit 0=1 to flag bypass mode <2>
|
|||
|
BRA.S @waitDone ; <2>
|
|||
|
|
|||
|
@ByPassMsg DC.B ByPassCmd, 0, $ff ; Turn bypass on or off
|
|||
|
align 2
|
|||
|
endwith
|
|||
|
TITLE 'IOPmgr - Initialize SCC IOP Hardware'
|
|||
|
|
|||
|
;_______________________________________________________________________ <1.4>
|
|||
|
;
|
|||
|
; Routine: jsr SCCIOPHwInit
|
|||
|
; Inputs: none
|
|||
|
; Outputs: none
|
|||
|
; Destroys: A0, A1, D0
|
|||
|
; Calls: none
|
|||
|
; Called by: Start Manager
|
|||
|
;
|
|||
|
; Function: Initializes the SCC IOP, so that the SCC can be used in ByPass
|
|||
|
; mode and has the correct timing to that when TimeSCCDB is
|
|||
|
; computed it will be correct.
|
|||
|
;
|
|||
|
;_______________________________________________________________________
|
|||
|
|
|||
|
|
|||
|
SCCIOPHwInit
|
|||
|
movea.l SCCRd,a1 ; a1 <- IOP Base
|
|||
|
move.b #resetIopRun,iopStatCtl(a1) ; init the Status/Ctl reg, hold IOP reset
|
|||
|
|
|||
|
; Download the IOP code
|
|||
|
|
|||
|
lea SCCIOPInitCode,a0 ; a0 <- start of IOP code (in this ROM)
|
|||
|
move.w (a0)+,d0 ; get the size code (-1 for DBRA)
|
|||
|
move.w (a0)+,iopRamAddr(a1); setup the code load address
|
|||
|
|
|||
|
@LoadLoop move.b (a0)+,iopRamData(a1); download a byte
|
|||
|
dbra d0,@LoadLoop ; load all of the bytes
|
|||
|
|
|||
|
; Start IOP execution
|
|||
|
|
|||
|
move.b #setIopRun,iopStatCtl(a1) ; release IOP reset (let it rip!)
|
|||
|
lsr.w #6,d0 ; $FFFF -> $03FF -> loop 1024 times
|
|||
|
@wait dbra d0,@wait ; delay a bit while IOP initializes
|
|||
|
rts ; all done
|
|||
|
|
|||
|
; 6502 code to throw SCC IOP into bypass mode
|
|||
|
SCCIOPInitCode
|
|||
|
dc.w (@end-@Start)-1 ; size of the code (in bytes), -1 for DBRA
|
|||
|
dc.w $8000-(@end-@Start) ; (word) load address
|
|||
|
@start ; code starts here
|
|||
|
dc.b $A9,$81 ; 7FEE: lda #1*DEBUG+0*SCCISM+1*BYPASS
|
|||
|
dc.b $8D,$30,$F0 ; 7FF0: sta SCCControlReg
|
|||
|
dc.b $A9,$23 ; 7FF3: lda #SccIOCtlReg <2>
|
|||
|
dc.b $8D,$31,$F0 ; 7FF5: sta IOControlReg
|
|||
|
dc.b $80,$FE ; 7FF8: bra $7FF8
|
|||
|
|
|||
|
dc.b $EE,$7F ; 7FFA 7FEE Non-Maskable Interrupt vector
|
|||
|
dc.b $EE,$7F ; 7FFC 7FEE Processor reset vector
|
|||
|
dc.b $EE,$7F ; 7FFE 7FEE Interrupt Request vector
|
|||
|
@end ; code ends here
|
|||
|
|
|||
|
|
|||
|
if PadForOverpatch then ; <1.6>
|
|||
|
align 4 ; <1.5>
|
|||
|
string asis ; <1.5>
|
|||
|
dcb.l ($770-(*-IOPmgr))/4,'Gary' ; padding for overpatch <1.5>
|
|||
|
endif ; <1.6>
|
|||
|
|
|||
|
endproc
|
|||
|
END
|
|||
|
|