mac-rom/OS/SCSIMgr/SCSIMgrNew.a

2328 lines
96 KiB
Plaintext
Raw Normal View History

;----------------------------------------------------------------------------------------------
;
; File: SCSIMgrNew.a
;
; Written by: Jerry Katzung
;
; Copyright: <09> 1989-1992 by Apple Computer, Inc. All rights reserved.
;
; Change History (most recent first):
;
; <SM6> 11/3/92 SWC Changed SCSIEqu.a->SCSI.a.
; <SM5> 6/2/92 kc Revert to <SM3>
; <SM4> 6/1/92 CS Roll more cyclone stuff in.
; <SM3> 5/17/92 kc Include PowerPrivEqu.a. The powerCntl equ was moved inside the
; pmCommandRec record.
; <8> 5/1/92 JSM Get rid of conditionals: don<6F>t use not forROM, this file is only
; used in ROM builds, don<6F>t use hasGDU, isUniversal, or onMacXX
; style conditionals, left padForOverPatch conditionals for now.
; <7> 1/15/92 KC Fixed "short branch to the next instruction changed to NOP"
; warning.
; <6> 9/16/91 JSM Cleanup header.
; <5> 9/21/90 BG Removed <4>. 040s are behaving more reliably now.
; <4> 7/20/90 BG Added EclipseNOPs for flakey 040s.
; <3> 5/16/90 MSH Added hasPowerControls to hcmac conditional.
; <2> 1/11/90 CCH Added include of <20>HardwarePrivateEqu.a<>.
; <1.5> 9/11/89 jwk NEEDED FOR F19 - Support for Disc during DMA, Mac SE and Mac
; Plus support
; <1.4> 7/24/89 jwk NEEDED FOR AURORA - Unlink w/ ioQElSize to fix alt. bus call bug
; <1.3> 7/17/89 jwk NEEDED FOR AURORA - Fixed "Command" routine to use scsiCmdOffs
; ptr
; <1.2> 7/15/89 jwk NEEDED FOR AURORA - Added code review changes, F19 support
; routines
; <1.1> 6/29/89 jwk NEEDED FOR AURORA - Fixed Tape Drive and Slow Write timing bugs
; <1.0> 6/13/89 jwk Reorganizing SCSI sources to be Universal-ROM friendly
;
;----------------------------------------------------------------------------------------------
;
; Old comments are saved for historical purposes in the SCSIMgrOld.a file. <v1.2>
;
;----------------------------------------------------------------------------------------------
BLANKS ON
STRING ASIS
PRINT OFF
IF (&TYPE('dbSymbols') = 'UNDEFINED') THEN
dbSymbols EQU 0 ; for debugging purposes <v1.5>
ENDIF
IF (&TYPE('phaseLogging') = 'UNDEFINED') THEN
phaseLogging EQU 0
ENDIF
LOAD 'StandardEqu.d'
INCLUDE 'HardwarePrivateEqu.a'
INCLUDE 'SCSI.a'
INCLUDE 'SCSIPriv.a'
INCLUDE 'PowerPrivEqu.a'
PRINT ON
SCSINew PROC EXPORT
WITH scsiPB,scsiPrivPB, scsiGlobalRecord, dcInstr, machSpecRecord
EXPORT ClearBus
EXPORT DataDMA ;<v1.2>
EXPORT DataIO, Command, StatusPhase, ResetBus, MsgOut, MsgIn
EXPORT msgCmdComplete, msgSaveDataPtr, msgRestorePtrs, msgDisconnect
EXPORT msgIdentifyIn, msgMsgRejIn, msgInvalidIn, msgLCCF
EXPORT msgInvalidOut, msgIdentifyOut, msgNoOp, msgMsgRejOut, msgKillIO
EXPORT msgBusDevRst ;<v1.5>
EXPORT Preflight, EnDequeue
EXPORT Message, SCSIDT, Find, Setup, TimeTask
EXPORT ClearState, SetTimer, LogError ;<v1.2>
EXPORT NewSCSITrap
EXPORT DoRequestIO, DoKillIO, DoBusInfo, AltBusCall
WITH scsiPB,scsiPrivPB, scsiGlobalRecord, dcInstr, PmgrRec
;-------------------------------------------------------------
;
; The new and improved SCSI Manager.
;
; A4 is set up by StdEntry and contains a pointer to the SCSI Manager global
; area in the system heap.
;
;
; Register convention:
;
; Old SCSI Mgr New SCSI Mgr
;
; a0 - scratch scratch
; a1 - scratch scratch
; a2 - buffer pointer buffer pointer
; a3 - SCSI read base pointer SCSI read base pointer
; a4 - SCSI Mgr globals pointer SCSI Mgr globals pointer
; a5 - n/a active request pointer
;
; d0 - scratch (result) scratch
; d1 - scratch (bytes left) scratch
; d2 - scratch (count) scratch
; d3 - scratch (6) scratch
; d4 - scratch (data transfer flags) data transfer flags
; d5 - scratch data-chaining instruction ptr
; d6 - scratch/patches' SCSI write offset SCSI write offset for patches
; d7 - zero register zero register
;
;
; NewSCSITrap
;
; Called by: OS trap dispatcher.
;
; Calls: Preflight, ResetBus, Message
;
; On entry: a0 - pointer to SCSI request parameter block
; d0 - routine selector
;
; On exit: d0 - result code (OSErr)
;
; Function: This routine is responsible for implementing the various new SCSI Mgr routines.
; The new SCSI Mgr routines are dispatched OS traps, with a selector passed in d0.
; It checks the parameter block, enqueues the request, and starts processing it if
; it is the only request in the queue. Returns an OSErr to describe what happened.
;
NewSCSITrap:
movem.l scsiRegs,-(sp) ; save registers
movea.l SCSIGlobals,a4 ; get pointer to SCSI Manager globals <v1.7>
NewSCSICommon ; <v1.2>
move.l a0,a5 ; set up pointer to the request <v1.2>
move.l base5380(a4),a3 ; SCSI chip read base address
moveq.l #0,zeroReg ; set up the "zero register"
@checkBusID
tst.w scsiBus(a5) ; is it for the motherboard ?
beq.s @checkCall ; keep going, it's for us
@sendItOff
movea.l jvAltBusCall(a4),a0 ; get addr of alternate SCSI bus call
jsr (a0) ; pass the call off to the other SCSI bus
bra.s @exit ; let's get out of here
@checkCall
cmpi.w #scsiNewSCSIMax,d0 ; valid selector ?
bls.s @goodSelector ; if so, call the appropriate routine
@SCSIOops
move.w #scsiBadPBErr,d0 ; tell driver that call is not implemented <v1.2>
bra.s @exit ; return the error message
@goodSelector
lea.l newSCSIDisp(a4),a0 ; point to base of NewSCSI dispatch table
movea.l 0(a0,d0.w*4),a0 ; get address of trap code <v1.2>
jsr (a0) ; call the appropriate trap code
@exit
movem.l (sp)+,scsiRegs ; restore registers (d0 is not restored)
rts
;
; Send a synchronous control call to the other SCSI bus.
; Map any errors to "scsiBadPB".
;
AltBusCall:
lea.l -ioQElSize(sp),sp ; reserve storage for control PB <v1.2>
movea.l sp,a0 ; point to Control PB
move.w scsiBus(a5),ioRefNum(a0) ; SCSI bus ID is the driver refnum
move.w d0,csCode(a0) ; pass NewSCSI selector to driver
move.l a5,csParam+4(a0) ; pass pointer to scsiPB in Control PB
_Control ; send it off synchronously
lea.l ioQElSize(sp),sp ; clean up the stack <v1.4>
beq.s @exit ; if no error, then we can leave
move.w #scsiBadPBErr,d0 ; map all errors into "bad PB" errors <v1.2>
@exit
rts
;
; Perform a SCSIRequestIO call
;
DoRequestIO:
movea.l jvPreflight(a4),a0 ; get address of routine
jsr (a0) ; preflight the parameter block
bne.s @exit ; get out
@goodPB
moveq.l #scNewRequest,d0 ; we've got a new request to process
move.l a5,d1 ; ptr to parameter block
movea.l jvMessage(a4),a0 ; addr of deferred task messaging routine
jsr (a0) ; send the message
moveq.l #noErr,d0 ; queueing was successful
move.l scsiMgrFlags(a5),d1 ; get SCSI flags
btst.l #scsiBAsync,d1 ; is the async bit set ?
bne.s @exit ; if async, then exit
@wait
tst.w scsiResult(a5) ; wait for a change to negative or zero
bgt.s @wait ; if greater than zero, keep waiting
@exit
rts
;
; Perform a SCSIKillIO call
; Could be graceful here; instead, reset the SCSI bus, killing all aboard.
;
DoKillio:
movea.l jvResetBus(a4),a0 ; addr of "reset SCSI bus" routine
jsr (a0) ; reset the bus, and kill all I/O requests
moveq.l #noErr,d0 ; it worked (imagine that)
rts ; we're done
;
; Perform a SCSIBusInfo call
;
DoBusInfo:
moveq.l #noErr,d0 ; assume no error <v1.2>
move.w scsiVersion(a5),d1 ; get the selector <v1.2>
bne.s @check1 ; if nonzero, check if it is a 1 <v1.2>
move.l zeroReg,d1 ; clear upper bits <v1.2>
move.b state2(a4),d1 ; return our SCSI Mgr revision level <v1.2>
move.l d1,scsiQLink(a5) ; return the result <v1.2>
rts ; we're done <v1.2>
@check1
subq.l #1,d1 ; what's our byte-to-byte timeout ?
bne.s @check2 ; check if it is a 2
move.l busTO(a4),scsiQLink(a5) ; return the result <v1.2>
rts ; we're done <v1.2>
@check2
subq.l #1,d1 ; do we have a SCSI DMA channel ?
bne.s @SCSIOops ; if not, we have an unimplemented selector
move.l G_Reserved0(a4),d1 ; get machine info (including SCSI DMA) <v1.1><v1.2>
andi.l #1<<sSCSIDMAExists,d1 ; mask for DMA bit (clears other bits) <v1.1><v1.2>
beq.s @DMArslt ; return zero if no DMA <v1.1><v1.2>
moveq.l #1,d1 ; return one if DMA <v1.1><v1.2>
@DMArslt move.l d1,scsiQLink(a5) ; return the result <v1.1><v1.2>
rts ; we're out of here <v1.2>
@SCSIOops
move.w #scsiBadPBErr,d0 ; tell driver that call is not implemented <v1.2>
rts ; return the result
;
; ResetBus - Reset the SCSI bus and kill all the requests in the queue.
;
; Calls: ClearState
;
ResetBus:
move.w sr,-(sp)
ori.w #HiIntMask,sr ; no more interrupts
move.b #iRST,sICR+WrOffs(a3) ; assert SCSI Reset line
lea.l scsiQHead(a4),a5 ; load pointer to first request in the queue
lea.l dummy(a4),a1 ; point to dummy parameter block
move.w #scsiBusResetErr,d0 ; report that SCSI bus was reset <v1.2>
movea.l jvClearState(a4),a0 ; addr of clear request routine
@loop
movea.l scsiQLink(a5),a5 ; point to the next request <v1.2>
cmp.l a5,zeroReg ; are we done ?
beq.s @delay1 ; if so, delay for good measure <v1.2>
cmpa.l a1,a5 ; are we pointing at the dummy PB ?
beq.s @killOldPB ; if so, kill it differently
jsr (a0) ; kill off the new SCSI Mgr PB
bra.s @loop ; keep going
@killOldPB
moveq.l #scsiOldSCSIErr,d1 ; pretend old SCSI Mgr is killing this request
exg.l d0,d1 ; switch error messages
jsr (a0) ; kill the request
exg.l d0,d1 ; restore the error message
bra.s @loop ; keep going
@delay1
move.w TimeDBRA,d0 ; DBRAs/ms. <v1.2>
lsr.w #2,d0 ; 1 ms./4 = 250 us. <C859> <v1.2>
@delay
dbra d0,@delay ; twiddle our thumbs...
move.b zeroReg,sICR+WrOffs(a3) ; de-assert *RST
movea.l jvClearIRQ(a4),a0 ; routine to clear interrupt request
jsr (a0) ; clear 5380's IRQ
move.w (sp)+,sr ; restore interrupts
move.w TimeDBRA,d0 ; DBRAs/ms.
mulu.w #500,d0 ; delay 500 msec for recovery <C859>
move.l d0,d1
swap d1 ; get high word for dbra
@2 dbra d0,@2 ; low word
dbra d1,@2 ; high word
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ? <2.8>
beq.s @DMAdone ; if not, skip it <2.8>
move.l #iINTREN,sDCTRL(a3) ; turn off hhsk <C914> <v1.2>
@DMAdone ; <2.8>
move.b zeroReg,sMR+WrOffs(a3) ; make certain of SCSI mode
rts
;
; EnDequeue
;
; Called by: ClearState, Message, SCSIGet
;
; On entry: a1 - ptr to "scsiPB" to enqueue/dequeue
; d0.l - selector:
; enqueueFlag (bit 31): 1 = enqueue, 0 = dequeue
; values:
; enqNormal = Enqueue at end of queue
; enqFront = Enqueue at front of queue
; deqNormal = Dequeue 1 request (linked commands are elevated)
; deqLinked = Dequeue linked cmds completely (to cleanly kill linked cmds)
;
; Function: Removes a SCSI request from the SCSI request queue
;
EnDequeue:
move.w sr,-(sp) ; preserve status
ori.w #HiIntMask,sr ; disable interrupts for exclusion
btst.l #enqueueFlag,d0 ; is this an enqueue request ?
bne.s @enqueue ; if so, enqueue the PB
;
; dequeue the request
;
@dequeue
movem.l a2-a3,-(sp) ; preserve work registers
move.l scsiQHead(a4),a2 ; start searching at the head
move.l a2,a3
@deqLoop
cmp.l a3,a1 ; is this the one?
beq.s @delete ; if so, go delete it
@nextElement
move.l a3,a2 ; update previous pointer
move.l QLink(a2),a3 ; follow the link
cmp.l scsiQTail(a4),a2 ; have we reached the tail?
bne.s @deqLoop
@noElement
bra.s @deqDone1 ; we don't return an error <v1.2>
@delete
cmp.l a2,a3 ; deleting the first element?
bne.s @delGeneral ; if not, perform general case delete
@first ; special case for first element in the queue
move.l QLink(a2),scsiQHead(a4) ; point head ptr to next element
bne.s @deqDone ; was it the only element?
clr.l scsiQTail(a4) ; if so, clear the tail too
bra.s @deqDone ; we're done
@delGeneral ; delete for the general case
move.l QLink(a3),QLink(a2) ; unlink it from the chain
cmp.l scsiQTail(a4),a1 ; was it the tail?
bne.s @deqDone ; if not we're done
move.l a2,scsiQTail(a4) ; update tail pointer
@deqDone
clr.l scsiQLink(a1) ; clear the link in the element
@deqDone1
movem.l (sp)+,a2-a3 ; restore work registers
bra.s @exit ; we're finished
;
; enqueue the request
;
@enqueue
movem.l d1/a2-a3,-(sp) ; save registers
movea.l a1,a3 ; point to request
@linkLoop
move.l scsiLinkCmd(a3),d1 ; is there a linked command ?
beq.s @noLink ; if not, exit
move.l d1,scsiQLink(a3) ; point this PB at the linked PB
move.l d1,a3 ; point to the linked PB
bra.s @linkLoop ; keep going
@noLink
move.l scsiQHead(a4),d1 ; anything in the queue ? (Used later)
bne.s @nonempty ; if so, handle differently
@empty
move.l a1,scsiQHead(a4) ; empty queue - point head to new element
move.l a3,scsiQTail(a4) ; point tail to new element
bra.s @enqDone ; we're out of here
@nonempty
cmpi.w #enqNormal,d0 ; enqueue at the end ?
bne.s @front ; no - enqueue at the front
@normal
move.l scsiQTail(a4),a2 ; get ptr to old QTail
move.l a1,scsiQLink(a2) ; update its link
move.l a3,scsiQTail(a4) ; update QTail
bra.s @enqDone ; we're done
@front
move.l d1,scsiQLink(a3) ; point last linked PB at old head-of-queue (set above)
move.l a1,scsiQHead(a4) ; point queue header at new head-of-queue
@enqDone
movem.l (sp)+,d1/a2-a3 ; restore registers
@exit
move.w (sp)+,sr ; restore status and go home
moveq.l #noErr,d0 ; return noErr <v1.2>
rts
;
; LogError - Record a SCSI error in the parameter block <v1.2>
; <start>
; On entry: a4 - ptr to the SCSI Mgr globals
; a5 - ptr to the active PB
; d0.w - result code for the parameter block
;
LogError:
cmp.l a5,zeroReg ; do we have an active request ?
beq.s @exit ; if not, nothing to record
movea.l scsiPrivate(a5),a0 ; get private storage
cmp.l a0,zeroReg ; so we have private storage ?
beq.s @exit ; if not, nothing to record
tst.w scsiTempErr(a0) ; have we stored an error message ?
bne.s @exit ; if so, don't clobber it
move.w d0,scsiTempErr(a0) ; save the error message
@exit ; <end>
rts ; we're done <v1.2>
;
; ClearState - Mark the SCSI Mgr as free
;
; Called by: SCSIRequestIO, SCSIKillIO, SCSIIntHnd, MsgIn, Data
;
; Calls: EnDequeue
;
; On entry: a3 - ptr to SCSI base read address
; a4 - ptr to the SCSI Mgr globals
; a5 - ptr to the PB to be removed
; d0.w - result code for the parameter block
;
; On exit: d0.w - result
;
; Function:
; This routine is the standard exit routine for a completed request, as well as a failed
; request.
; It removes the request from the SCSI request queue, sets the result field,
; and then calls the completion routine (if any).
; Additionally, the "timeout" global is reset by walking the SCSI request queue.
;
ClearState:
movem.l intrRegs,-(sp) ; save registers
;
; If the error in d0 is "oldSCSIMgrErr", then ClearState was called
; by the old SCSI Mgr. Dequeue the dummy PB, and check if the SCSI bus is ok.
;
cmpi.w #scsiOldSCSIErr,d0 ; were we called by the old SCSI Mgr ?
bne.s @newSCSIMgr ; if not, we were called by the new SCSI Mgr
@oldSCSIMgr
move.l scsiQHead(a4),d1 ; get head of the queue
beq.s @clearGState ; if no dummy element, just clear G_State
moveq.l #deqNormal,d0 ; dequeue the dummy PB normally
lea.l dummy(a4),a1 ; get address of dummy parameter block
movea.l jvEnDequeue(a4),a0 ; blow away the dummy parameter block
jsr (a0) ; use the standard routine
@clearGState
move.b zeroReg,G_State(a4) ; clear out SCSI Mgr semaphore
tst.b discID(a4) ; any disconnected ID's ?
beq.w @exit ; if not, we can leave Reselection interrupts disabled
move.b G_ID(a4),sSER+wrOffs(a3) ; enable Reselection interrupts
bra.w @exit ; bus can't be cleared under old SCSI Mgr <v2.3>
;
; If we get here, ClearState was called by the new SCSI Mgr, so a5 is valid.
; If a5 is zero, then we got a Reselect from a device we don't know about.
;
@newSCSIMgr
move.l zeroReg,activeReq(a4) ; no currently active request
move.l a5,a1 ; point to SCSI request to dequeue
move.l d0,d2 ; hold onto the result code
moveq.l #deqNormal,d0 ; dequeue this request normally
movea.l jvEnDequeue(a4),a0 ; addr of routine to dequeue request
jsr (a0) ; dequeue the request
@releasePrivates
move.l scsiPrivate(a5),d0 ; point at the SCSI Mgr private storage
beq.s @cleanupGlobals ; no private storage for this request <v1.2>
move.l d0,a0 ; valid pointer <v1.2>
move.w scsiTempErr(a0),d0 ; do we have a pending error ? <v1.2>
beq.s @rel1 ; if not, then use the error in d2 <v1.2>
move.w d0,d2 ; return original error message <v1.2>
@rel1
move.l zeroReg,scsiPrivate(a5) ; release the SCSI Mgr private storage
move.l zeroReg,scsiFlags(a0) ; clear "scsiFlags","State","Flag1", and "Flag2"
move.l zeroReg,scsiTime(a0)
move.l zeroReg,scsiCmdOffs(a0)
move.l zeroReg,scsiTempErr(a0) ; clear temporary error and filler fields <v1.2>
move.l zeroReg,scsiDCSaved(a0)
move.l zeroReg,scsiDCOffs(a0)
move.l zeroReg,scsiMoveOffs(a0)
tst.l nextPrivate(a4) ; check for a free private storage record
bne.s @cleanupGlobals ; there is already a free record
move.l a0,nextPrivate(a4) ; this record is free
@cleanupGlobals
move.w d2,scsiResult(a5) ; save the result code
lea.l discLUN(a4),a1 ; point to disconnected LUN table
move.l zeroReg,d2 ; clear offset into LUN table
move.b scsiReqID(a5),d2 ; get offset into LUN table
move.b scsiReqLUN(a5),d1 ; this LUN is available now
bclr.b d1,0(a1,d2.l) ; requests for this LUN can be run
tst.b 0(a1,d2.l) ; is anybody disconnected at this ID ?
bne.s @completion ; if so, continue
bclr.b d2,discID(a4) ; this entire ID is available
@completion
move.l scsiMgrFlags(a5),d0 ; get the SCSI flags
btst.l #scsiBImmed,d0 ; is it a page-fault request ? <v1.1>
beq.s @chkAsync ; if not, continue
move.l zeroReg,pageFault(a4) ; clear page fault indicator
@chkAsync
btst.l #scsiBAsync,d0 ; is it an asynchronous call ?
beq.s @newTimeOut ; synchronous - "scsiCompletion" is undefined
move.l scsiCompletion(a5),d0 ; is there a valid completion routine ?
beq.s @newTimeOut ; if not, calculate new earliest timeout
movem.l intrRegs,-(sp) ; save all registers <v1.1>
btst.b #sDeferUserFn,G_Reserved0+3(a4) ; is the VM-friendly call implemented ? <start>
bne.s @DUF ; if so, defer completion routine cleanly
@noDUF
move.l d0,a1 ; get completion routine address
move.l a5,a0 ; point a0 at parameter block
jsr (a1) ; simply jump to completion routine
bra.s @afterCompletion ; we've called the completion routine
@DUF
move.l d0,a0 ; pointer to completion routine
move.l a5,d0 ; pointer to parameter block
nop ; trap not defined ???
; _DeferUserFn ; post the completion (can't recover from an error)
@afterCompletion ; <v1.1>
movem.l (sp)+,intrRegs ; restore all registers <end>
@newTimeOut
moveq.l #-1,d0 ; start out with an "infinite" timeout
lea.l scsiQHead(a4),a1 ; get pointer to first parameter block
@loop
movea.l scsiQLink(a1),a1 ; get the next parameter block
cmp.l a1,zeroReg ; are we done ?
beq.s @setTimeOut ; if so, save timeout and make sure the bus is free
move.l scsiPrivate(a1),d1 ; is this request in progress ?
beq.s @loop ; if not, check next parameter block
move.l d1,a0 ; point to private storage for this request
cmp.l scsiTime(a0),d0 ; is this timeout earlier ?
blo.s @loop ; if not, check the next parameter block
move.l scsiTime(a0),d0 ; if earlier, save it
bra.s @loop ; check next parameter block
@setTimeOut
move.l d0,timeout(a4) ; save the new timeout
@exit
movem.l (sp)+,intrRegs ; restore registers
rts
;
; ClearBus - Clean up the bus, if necessary.
;
; Called by: SCSIRequestIO, SCSIKillIO, SCSIIntHnd, MsgIn, Data
;
; Calls: ResetBus
;
; On entry: a3 - ptr to SCSI base read address
; a4 - ptr to the SCSI Mgr globals
; a5 - ptr to the PB to be removed (NIL = reselected by a timed-out ID)
; d0.w - result code for the parameter block
;
; On exit: d0.w - result
;
; Function:
; This routine attempts to clean up the bus in three stages: first, an attempt is made
; to send an "Abort" message to the device; second, a "Bus Device Reset" message is
; sent to the device; finally, the SCSI bus is reset.
; (If the parameter block pointer is nil, then a device is on the bus, but the device's
; request has timed out. This is handled by aborting the rest of the transfer, since
; the original request has already failed.)
;
;
ClearBus:
movem.l intrRegs,-(sp) ; save registers
@chkBus
;
; At this point, the bus is probably hosed. It is possible that a SCSI Bus Reset
; is in progress, but the following code will ignore this.
;
moveq.l #aBSY+aSEL,d0 ; clear all but BSY and SEL (ignore RST) <v1.2>
and.b sCSR(a3),d0 ; mask with a snapshot of the bus <v1.2>
beq.s @exit ; if no BSY or SEL, bus is clear
;
; Reset the SCSI bus. We were running, so our device is hanging on the bus.
; If we are in the middle of a Reset, no other devices should be on the bus.
;
@assertReset
movea.l jvResetBus(a4),a0 ; get address of bus reset routine
jsr (a0) ; reset the SCSI bus and kill all requests
@exit
movem.l (sp)+,intrRegs ; restore registers
rts
;
; Message
;
; Function: Sends a message to the deferred task.
; (It also installs the deferred task, if necessary)
;
; On entry: d0.b - message selector
; d1 - additional information, if any
;
; Current messages include:
;
; scResel ID mask in d1.b
; scEOP
; scPhaseMm
; scNewRequest ptr to scsiPB in d1.l
; scLossBsy
; scBusReset
; scParity
;
;
;
Message:
movem.l intrRegs,-(sp) ; save registers
bset.b d0,dTask+dtParm(a4) ; tell SCSIDT which interrupt occurred
@reselect
cmpi.b #scResel,d0 ; is it a Reselect message ?
bne.s @newRequest
move.b d1,dTask+dtParm+3(a4) ; pass the reselect ID to dTask in low byte of dtParm
bra.s @enqueueSCSIDT
@newRequest
cmpi.b #scNewRequest,d0 ; is there a new request to process ?
bne.s @enqueueSCSIDT
movea.l jvEnDequeue(a4),a0 ; address of enqueue/dequeue routine
move.l d1,a1 ; get the pointer to the request
move.l scsiMgrFlags(a1),d1 ; get the flags from the request
btst.l #scsiBImmed,d1 ; is this a page-fault request ? <v1.1>
bne.s @enqInsert ; if so, place it at the front
@enqNormal
moveq.l #enqNormal,d0 ; enqueue this request at the end of the queue
jsr (a0) ; go do it
bra.s @enqExit ; we're done
@enqInsert
move.l a1,pageFault(a4) ; we're handling a page fault - keep ptr to PB
moveq.l #enqFront,d0 ; place this request at the front of the queue
jsr (a0) ; go do it
movea.l jvSCSIDT(a4),a0 ; address of SCSI Mgr deferred task
jsr (a0) ; handle VM page-fault by calling SCSIDT directly
@enqExit
@enqueueSCSIDT
bset.b #dtEnqueued,state1(a4) ; going to enqueue the deferred task
bne.s @exit ; if already set, then don't enqueue dTask
lea.l dTask(a4),a0 ; point to deferred task record
_DTInstall ; install it in the deferred task queue
@exit
movem.l (sp)+,intrRegs ; restore registers
rts
;
; Preflight
;
; Called by: SCSIRequestIO
;
; Calls: None.
;
; On entry: a0 - addr of Preflight routine (used for recursive calls) <v1.2>
; a4 - ptr to SCSI Mgr globals
; a5 - ptr to parameter block in question
;
; On exit: d0 - result code
;
; Function:
; This routine checks that the parameter block is consistent.
;
Preflight:
@initResult
moveq.l #noErr,d0 ; assume a good parameter block
@prelimCheck
tst.l scsiQLink(a5) ; is link NIL ?
bne.w @errorExit ; if not, get out
tst.l scsiPrivate(a5) ; is private storage link NIL ?
bne.w @errorExit ; if not, get out
tst.w scsiVersion(a5) ; is the version number greater than zero ?
beq.w @errorExit ; valid version numbers > zero
@checkLink
tst.l scsiLinkCmd(a5) ; are we pointing to a linked command ?
beq.s @checkPB ; if not, continue to check PB <v1.2>
bra.w @errorExit ; not allowed in the initial release <v1.2>
movem.l a0/a5,-(sp) ; save pointer to the PB
movea.l scsiLinkCmd(a5),a5 ; point to linked command PB
jsr (a0) ; call Preflight recursively
movem.l (sp)+,a0/a5 ; restore pointer to the parameter block
@checkPB
tst.l scsiReqTO(a5) ; check for a nonzero request timeout
beq.w @errorExit ; no timeout provided
tst.w scsiSelTO(a5) ; check for a nonzero selection timeout
beq.w @errorExit ; no timeout provided
cmpi.w #mgrVersion,scsiVersion(a5) ; is this a level-1 PB ?
bne.s @1 ; if not, we can't preflight the reserved fields
move.l scsiMgrFlags(a5),d2 ; get a copy of the flags
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ? <2.8><v1.1>
beq.s @noDMA ; if not, different check <2.8>
andi.l #$FFFFFF00,d2 ; if anything else is on, it's bad <v1.2>
bra.s @univDone ; done with the check
@noDMA
andi.l #$FFFFFF10,d2 ; if "physical" or anything else on, error <v1.2>
@univDone
bne.w @errorExit ; a reserved bit was non-zero
@1
moveq.l #LUNMask,d2 ; mask to check for valid ID and LUN
cmp.b scsiReqID(a5),d2 ; check for a valid target ID
blo.w @errorExit ; get out (ID > 7)
cmp.b scsiReqLUN(a5),d2 ; check for a valid logical unit number
blo.w @errorExit ; get out (LUN > 7)
tst.l scsiCmdBuf(a5) ; check that command pointer is non-nil
beq.s @errorExit ; nil command buffer ptr - get out
cmp.l scsiSnsBuf(a5),zeroReg ; is auto-sense being requested ?
beq.s @noSense ; if not, continue
cmpi.b #minSnsLen,scsiSnsLen(a5) ; is sense buffer of minimum length ?
blo.s @errorExit ; it's too short - error
@noSense
cmp.l scsiDataLen(a5),zeroReg ; is a data transfer expected ?
bne.s @dataTransfer ; if non-zero, data transfer is expected
@noDataTransfer
cmp.l scsiDCInstr(a5),zeroReg ; if no data transfer, this must be nil
bne.s @errorExit ; if non-nil, error
moveq.l #noErr,d0 ; no error
bra.s @initialize ; go initialize rest of PB
@dataTransfer
move.l scsiDCInstr(a5),a0 ; get ptr to data-chaining instructions
cmp.l a0,zeroReg ; check that data chaining is requested
beq.s @errorExit ; no data-chaining instr provided - error
@repeatInstr
tst.l dcAddr(a0) ; is it a dcStop instruction ?
beq.s @chkStop ; check if it is a valid dcStop (dcCount=dcOffset=0)
adda.w #dcSize,a0 ; point to next data-chaining instruction
bra.s @repeatInstr ; check next instruction
@chkStop
tst.l dcCount(a0) ; is the dcCount field zero ?
bne.s @errorExit ; if not, error
tst.l dcOffset(a0) ; is the dcOffset field zero ?
bne.s @errorExit ; if not, error
@initialize
;
; NOTE: "scsiCompletion","scsiUsrData", and "scsiCmdLen" are always untouched
;
move.w #scsiEnqueued,scsiResult(a5) ; mark async request in progress
move.b zeroReg,scsiSnsXfer(a5) ; no sense bytes returned, yet
move.l zeroReg,scsiDataXfer(a5) ; no data bytes transferred, yet
move.b #statusInitial,scsiStatus(a5) ; initialize status byte to "invalid" value
tst.l scsiDataLen(a5) ; data transfer expected ?
beq.s @exit ; if not, we're done
;
; In order to implement the "Save Data Pointer" message, it is necessary to
; save a copy of the entire data-chaining instruction block. The data-chaining
; instructions combined with the DCOffs pointer make up the data pointer for
; the request.
;
@datachaining
move.l scsiDCInstr(a5),a0 ; point to data-chaining instructions
moveq.l #dcLoop,d2 ; used for comparison
@repeat
move.l dcAddr(a0),d1 ; get the address or opcode
beq.s @dcStop ; if "dcStop", then we're done
cmp.l d2,d1 ; is it a "dcLoop" instruction ?
bne.s @save ; if not, save the address in "dcStore"
move.l dcCount(a0),d1 ; the loop count needs to be saved
@save
move.l d1,dcStore(a0) ; save the necessary value
adda.w #dcSize,a0 ; point to next data-chaining instruction
bra.s @repeat ; copy another instruction
@dcStop
move.l zeroReg,dcStore(a0) ; clear out dcStore field in dcStop instruction
bra.s @exit ; everything is fine, no error
@errorExit
move.w #scsiBadPBErr,d0 ; a bad SCSI parameter block <v1.2>
@exit
tst.w d0 ; set condition codes
rts
;
; Setup
;
; Called by: SCSIDT
;
; Calls: SetTimer.
;
; On entry: a4 - ptr to SCSI Mgr globals
; a5 - ptr to parameter block in question
;
; On exit: d0 - result code
;
; Function:
; This routine sets up the SCSI Manager globals and the CPU registers to handle
; a SCSI request. It is the "standard entry" routine used by the rest of the
; new SCSI Manager code.
;
Setup:
move.l a5,activeReq(a4) ; record the active request
@calcTimeout
move.l Ticks,d2 ; get the current time
move.l scsiReqTO(a5),d1 ; get the request timeout value
asr.l #4,d1 ; convert to ticks (using msec timebase if possible)
move.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l scsiTime(a0),d0 ; check the deadline for the request
bne.s @calculated ; if non-zero, it's already calculated
move.l d2,d0 ; get the current time
add.l d1,d0 ; calculate the time it should timeout
move.l d0,scsiTime(a0) ; save timeout
bra.s @default ; let's go
@calculated
sub.l d2,d0 ; calculate the differential
bcs.s @default ; current time past timeout, use default
cmp.l d1,d0 ; compare default and calculated timeouts
bls.s @default ; if lower, use the default timeout
move.l d0,d1 ; else, use the calculated timeout
@default
move.l d2,d0 ; get current time
add.l d1,d0 ; add time differential
move.l d0,timeout(a4) ; next timeout time
mulu.w #17,d1 ; convert ticks to milliseconds
move.l d1,d0 ; milliseconds to the timeout
movea.l jvSetTimer(a4),a0 ; set the timer (no error possible)
jsr (a0) ; go to the routine
rts
;
; Find
;
; Called by: SCSIDT
;
; Calls: None.
;
; On entry: d0.l - contains a mask of SCSI ID's to check.
; bit 31 : if set, find any request
; bits 7-0 : bitmap of possible ID(s)
; (Time to start a new request.)
;
; bit 31 : if clear, find the first request (A reselection has occurred.)
; bits 15-8 : bitmap with ID to find
; bits 7-0 : bitmap with LUN to find (Zero, if reselected without an Identify)
;
; On exit: d0 - points to the request that was found. (Set to zero if none was found.)
;
; Function:
; This routine is used in two ways: (1) it searches for the first request that is not
; disconnected, and is not queued behind a disconnected request, or (2) find the first
; disconnected request for a given ID and LUN. It stops when it hits the end of the
; queue, or the dummy parameter block used by an old SCSI Mgr call.
;
; If an Identify message is not sent after a Reselection, the LUN bitmap will zero.
; (It is assumed that the LUN, if any, was passed in the command buffer.)
; In this case, the target does not implement the Identify message, and so the LUN
; is forced to zero. This means that only one LUN at a time may be used at a given
; ID if the Identify message is not implemented. ("scsiReqLUN" must be set to zero.)
;
; ActiveReq, by convention, is always cleared at the end of a SCSI request.
; Find does nothing if ActiveReq is set.
;
Find
movem.l a0-a3/d1-d2,-(sp) ; save registers
move.w sr,-(sp) ; save the status register contents
ori.w #HiIntMask,sr ; disable interrupts temporarily
tst.l activeReq(a4) ; are we currently servicing a request ?
bne.s @notFound ; if so, act as if nothing was found
lea.l discLUN(a4),a1 ; ptr to base of the disconnected LUN table
lea.l scsiQHead(a4),a0 ; get the head pointer
move.l zeroReg,d2 ; clear out d2 to use it as an offset
btst.l #findAnyFlag,d0 ; which type of search are we doing ?
beq.s @setupGiven ; go find first request for given ID mask
move.b discID(a4),d1 ; get mask of disconnected ID's
eor.b d1,d0 ; remove disconnected ID's from the running
bra.s @nextReq ; start searching
@setupGiven
move.b d0,d1 ; get LUN bitmap into d1
bne.s @fixID ; nonzero LUN bitmap is correct
bset.l zeroReg,d0 ; LUN is in scsiCmdBuf (no Identify - force to LUN 0)
@fixID
rol.w #8,d0 ; get ID bitmap into d0.b
;
; Bit 31 set: d0.b contains a bitmap of possible ID's (1 or more)
; d1.b is unused
;
; Bit 31 clear: d0.b contains an ID bitmap
; d1.b contains an LUN bitmap to use:
; ID 3, LUN 6 : d0.b = 00001000 d1.b = 01000000
;
@nextReq
move.l scsiQLink(a0),a0 ; get next request
cmp.l a0,zeroReg ; are we at the end ?
beq.s @notFound ; if so, we didn't find the request
@chkSearchType
btst.l #findAnyFlag,d0 ; what type of search was requested ?
bne.s @findAny ; find first request for any non-disc ID
@findGiven
move.b scsiReqID(a0),d2 ; get the ID of the parameter block
blt.s @dummy ; if ID < 0, this is the dummy param block
btst.l d2,d0 ; is the ID the same as the given ID ?
beq.s @nextReq ; if not, check the next request
move.b scsiReqLUN(a0),d2 ; get the LUN of the parameter block
btst.l d2,d1 ; is the LUN the same ?
beq.s @nextReq ; if not, keep looking
bra.s @found ; we found the request we were looking for
@findAny
move.b scsiReqID(a0),d2 ; get the ID that we wish to use
blt.s @dummy ; if ID < 0, this is the dummy parameter block
btst.l d2,d0 ; is this ID still in the running ?
beq.s @nextReq ; if not, check the next request
@okID
move.b 0(a1,d2.w),d1 ; get the LUN bitmap for the current ID
move.b scsiReqLUN(a0),d2 ; get the LUN to check
btst.l d2,d1 ; check if this LUN is disconnected
bne.s @nextReq ; if it is, it is not a possible candidate
@found ; we found it -- fall through from above
move.l a0,d0 ; return the pointer in d0
bra.s @exit ; we found our request
@notFound
@dummy
move.l zeroReg,d0 ; no requests in front of the dummy parameter block
@exit
move.w (sp)+,sr ; restore status register contents
movem.l (sp)+,a0-a3/d1-d2 ; restore register contents
tst.l d0
rts
;
; SetTimer
;
; Called by: Setup, MsgIn
;
; Calls: _RmvTime, _InsTime, _PrimeTime
;
; On entry: a3 - ptr to SCSI read base address
; a4 - ptr to SCSI Manager globals
; d0 - number of milliseconds to wait (0 means stop the timer)
;
; Function: Provides a timer facility for the SCSI Mgr. It contains the use of the
; Time Manager traps in one routine. (This makes it easier for A/UX to use
; the new SCSI Manager.)
;
SetTimer:
lea.l timer(a4),a0 ; point to timer task
tst.l d0 ; is there a need to re-arm the timer ?
bne.s @arm ; if count <> 0, re-arm the timer
@stop
_RmvTime ; stop the timer (ignore error)
rts ; we're done (we've stopped the timer) <v1.2>
@arm
move.l d0,-(sp) ; hold on to timeout value
_RmvTime ; just for safety's sake (ignore error)
_InsTime ; put it back in the queue (ignore error)
move.l (sp)+,d0 ; restore timeout value
_PrimeTime ; set the timer
rts
;
; TimeTask
;
; Called by: Time Manager
;
; Calls: ClearState
;
; On exit: Timed-out requests are removed from the queue.
;
; Function: This routine handles timeouts on the SCSI bus.
;
; Algorithm: if (queue nonempty) {
; if (curtime > timeout) {
; for (each element) {
; if (tagged)
; ClearState(element, saTimeOutErr);
; else if (scsiTime < curtime)
; tag element;
; }
; }
; rearm timer;
; }
;
TimeTask:
IF dbSymbols THEN
link a6,#0 ; MacsBug stack frame
nop ; filler
ENDIF
movem.l intrRegs,-(sp) ; save registers
movea.l SCSIGlobals,a4 ; get pointer to SCSI Manager globals <v1.7>
TimeTaskCommon ; <v1.2>
move.l base5380(a4),a3 ; point to the SCSI base read address
moveq.l #0,zeroReg ; set up the "zero" register
@timer
move.l scsiQHead(a4),d0 ; is there anything in the queue ?
beq.s @exit ; if empty, leave without resetting the timer
move.l Ticks,d0 ; get the current "time"
cmp.l timeout(a4),d0 ; have we hit the first possible timeout ?
blo.s @rearmTimer ; not yet, it's a spurious watchdog interrupt
move.l activeReq(a4),a5 ; get the active request, if any
cmp.l a5,zeroReg ; is there a currently active request ?
beq.s @watchdogTimeout ; if not, then the watchdog timer went off
@requestTimeout
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
cmp.l scsiTime(a0),d0 ; check timeout time against current time
blo.s @rearmTimer ; if less, then it has not timed out
@timedOut
move.w #scsiTimeOutErr,d0 ; it took too long <v1.2>
movea.l jvClearState(a4),a0 ; clear SCSI Mgr and SCSI bus, if necessary
jsr (a0) ; go to the routine
;movea.ljvClearBus(a4),a0
;jsr (a0)
bra.s @rearmTimer ; set up the next timeout
@watchdogTimeout
move.l d0,d1 ; hold on to the current time
move.w #scsiTimeOutErr,d0 ; walk queue, killing requests
lea.l scsiQHead(a4),a5 ; ptr to start of SCSI request queue
@loop
move.l a5,a1 ; hold onto current parameter block
movea.l scsiQLink(a5),a5 ; point to the next parameter block
cmp.l a5,zeroReg ; are we done ?
beq.s @rearmTimer ; if so, get out
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
cmp.l a0,zeroReg ; is it in progress ?
beq.s @loop ; if not, check next request
cmp.l scsiTime(a0),d1 ; else, has this request timed out ?
blo.s @loop ; if not, check the next request
movea.l jvClearState(a4),a0 ; get address of ClearState routine
jsr (a0) ; kill off this request
move.l a1,a5 ; continue with the next parameter block
bra.s @loop ; continue until all PB's are checked
@rearmTimer
move.l zeroReg,d0 ; clear high word of count
move.w #WDTO,d0 ; set timeout
movea.l jvSetTimer(a4),a0 ; set the timer
jsr (a0) ; go to the routine
@exit
movem.l (sp)+,intrRegs ; restore registers
IF dbSymbols THEN
unlk a6
rts
dc.b 'TIMETASK'
ELSE
rts
ENDIF
;
; SCSIDT
;
; Called by: Deferred Task Dispatcher, SCSIIntHnd, TimeTask, Message
;
; Calls:
;
; On entry: "dtParm" (a1) reflects the type(s) of interrupts.
;
; Function:
; This is the remainder of the SCSI interrupt handler. It also handles the Time Manager
; interrupts that provide the watchdog timer feature. (On the Mac Plus, a second timer
; is used to monitor the SCSI chip's interrupt line.) It is called with all interrupts
; enabled, except possibly during a page-fault.
;
SCSIDT:
if dbSymbols then
link a6,#0
nop ; filler for debugging purposes
endif
@setup
movem.l intrRegs,-(sp) ; save registers
movea.l SCSIGlobals,a4 ; get pointer to SCSI Manager globals <v1.7>
SCSIDTCommon ; <v1.2>
moveq.l #0,zeroReg ; clear out the zero register <v1.2>
move.l base5380(a4),a3 ; SCSI read base address
move.l activeReq(a4),a5 ; point to active request, if any <v1.2>
@clrDTEnqueued
move.l zeroReg,dTask(a4) ; clear out the link for safety
bclr.b #dtEnqueued,state1(a4) ; tell Message that dTask is not enqueued
@doWhile ; start of giant loop
@intrPoll ;
; If interrupts are disabled, and there are no interrupts to process, then
; we will have to poll for interrupts, and call IntHnd to post them.
; (This only happens while servicing a page fault with interrupts disabled.)
;
tst.l dTask+dtParm(a4) ; are there any interrupts to process ?
bne.s @chkBusReset ; if there are, service them first
move.w sr,d0 ; get a copy of the status register
andi.w #HiIntMask,d0 ; get the CPU interrupt level
cmpi.w #scsiIntMask,d0 ; check against the SCSI chip's interrupt level
blo.s @chkBusReset ; if CPU level is lower, interrupts will get through
@waitForIRQ
btst.b #bIREQ,sBSR(a3) ; is an interrupt pending ?
beq.s @waitForIRQ ; if not, keep waiting
movea.l jvIntHnd(a4),a0 ; get address of interrupt handler
jsr (a0) ; call SCSIIntHnd to post interrupt for service
;
; At this point, we have an interrupt to service.
;
@chkBusReset
bclr.b #scBusReset,dTask+dtParm(a4) ; is it a SCSI Bus Reset interrupt ?
beq.s @notBusReset ; if not, check other interrupts
;
; Send "bus was reset" message to all the requests in the queue
;
@busReset
lea.l scsiQHead(a4),a5 ; load pointer to first request in the queue
lea.l dummy(a4),a1 ; point to dummy parameter block
move.w #scsiBusResetErr,d0 ; report that SCSI bus was reset <v1.2>
@brLoop
movea.l jvClearState(a4),a0 ; addr of clear request routine
movea.l scsiQLink(a5),a5 ; point to the next request <v1.2>
cmp.l a5,zeroReg ; are we done ?
beq.w @endWhile ; if so, see if anything needs to be done
cmpa.l a1,a5 ; are we pointing at the dummy PB ?
beq.s @killOldPB ; if so, kill it differently
jsr (a0) ; kill off the new SCSI Mgr PB
bra.s @brLoop ; keep going
@killOldPB
moveq.l #scsiOldSCSIErr,d1 ; pretend old SCSI Mgr is killing this request
exg.l d0,d1 ; switch error messages
jsr (a0) ; kill the request
exg.l d0,d1 ; restore the error message
bra.s @brLoop ; keep going
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
;
; Handle any of the other interrupts. The algorithm is as follows:
;
; If (Reselect or EOP or PhaseMm) {
; do what is necessary to handle the interrupt
; } elseif (newRequest or requestsToProcess or LossOfBSY or Parity) {
; do what is necessary to handle the interrupt,
; start up a new request if possible
; }
; if necessary, handle phases until a Bus Free condition exists
;
@notBusReset
@Reselect
bclr.b #scResel,dTask+dtParm(a4) ; is it a Reselection interrupt ?
beq.s @EOP ; if not, check for EOP
@Resel1
btst.b #bSEL,sCSR(a3) ; wait for *SEL to go away (infinite loop)
bne.s @Resel1
move.b zeroReg,sICR+WrOffs(a3) ; release *BSY
@validResel ; bypass check for "activeReq" because
bra.w @noActiveReq ; "activeReq" is set after an Identify message
@EOP
bclr.b #scEOP,dTask+dtParm(a4) ; is it an End Of Process interrupt ?
beq.w @PhaseMm ; if not, check if Phase Mismatch <v1.2>
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ?
beq.s @nonDMA ; if not, no processing required
bclr.b #waitingForIRQ,state1(a4) ; we've been kicked out of DMA <v1.5>
move.l sDCTRL(a3),d0 ; look at the DMA control register <v1.2>
btst.l #bDMABERR,d0 ; did we get a DMA bus error ? <v1.2>
bne.s @eopDMABERR ; if so, handle it <v1.2>
move.l sDCNT(a3),d0 ; did we transfer all the data ? <v1.2>
beq.s @eopSuccess ; if so, clean up after the DMA <v1.2>
@eopDMABERR
@eopError
if dbSymbols then
_Debugger
endif
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
bra.w @nextPhase ; just exit for now <v1.2>
@eopSuccess
movea.l jvDataDMA(a4),a0 ; addr of DMA data-chaining interpreter <v1.2>
jsr (a0) ; start up the next DMA, if necessary <v1.2>
btst.b #waitingForIRQ,state1(a4) ; did we start a new DMA ? (waiting for intr) <v1.5>
bne.w @endWhile ; if so, bail out for now, if possible <v1.2>
;
; fall through - we moved all the data requested, so it should be a new phase <v1.2>
;
@nonDMA
bra.w @nextPhase ; finished - resume handling phases <v1.2>
@PhaseMm
bclr.b #scPhaseMm,dTask+dtParm(a4) ; is it a Phase Mismatch interrupt ?
beq.w @newRequest ; if not, check for a new request
if dbSymbols then
_Debugger
endif
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ? <v1.5>
beq.s @nonDMA2 ; if not, no processing required <start>
cmp.l a5,zeroReg ; do we have an active request ?
beq.w @pmmCleanup ; if not, just clean up the interrupt
bclr.b #waitingForIRQ,state1(a4) ; are we awaiting an interrupt ?
beq.s @pmmCleanup ; if not, just clean up the interrupt
movea.l scsiPrivate(a5),a0 ; point to private storage for request
move.l scsiPCount(a0),d0 ; length of DMA transfer request
sub.l sDCNT(a3),d0 ; subtract number of bytes left to transfer
;
; On the SCSI DMA chip, if a Disconnect occurs during a long DMA transfer, the
; sDCNT register will be off by one. (It reports that one too many bytes were
; transferred.) To correct this, we need to subtract one from the number of
; bytes to be transferred only if it is a write operation. <v1.5>
;
move.l scsiMgrFlags(a5),d1 ; get the flags for this transaction
btst.l #scsiBWrite,d1 ; test the Write flag
beq.s @pmmUpdate ; if it's a read operation, just continue
subq.l #1,d0 ; adjust the number of bytes transferred
@pmmUpdate
add.l d0,scsiMoveOffs(a0) ; save number of bytes transferred
add.l d0,scsiDataXfer(a5) ; update number of bytes transferred
move.b zeroReg,sICR+WrOffs(a3) ; disable data bus (mainly for writes)
move.l #iINTREN,sDCTRL(a3) ; turn off hhsk <C914> <end>
; finish cleaning up <v1.5> <7>
@nonDMA2
@pmmCleanup
move.b zeroReg,sMR+WrOffs(a3) ; disable phase mismatch interrupts
bra.w @nextPhase ; finished - resume handling phases
;
; These interrupts happen outside of a transaction, so after servicing
; the interrupt, we start up another request, if possible.
;
@newRequest
bclr.b #scNewRequest,dTask+dtParm(a4) ; is it a new request to process ?
beq.s @LossBsy ; if not, check for Loss of BSY
bra.s @startSCSI ; start up a request, if possible
@LossBsy
bclr.b #scLossBsy,dTask+dtParm(a4) ; is it a Loss of BSY interrupt ?
beq.s @Parity ; if not, check for Parity Error
bclr.b #waitingForIRQ,state1(a4) ; we got the interrupt we were waiting for <v1.5>
move.b zeroReg,sMR+WrOffs(a3) ; disable Loss of BSY interrupts
bra.s @startSCSI ; finished - start up a request, if possible
@Parity
bclr.b #scParity,dTask+dtParm(a4) ; is it a Parity Error interrupt ?
IF dbSymbols THEN
beq.s @startSCSI ; if not, start up a request, if possible <7>
_Debugger ; we got a parity interrupt (never enabled)
ENDIF
; Fall through to "@startSCSI"
;
; Start up another request, if possible
;
@startSCSI
move.w sr,-(sp) ; save status register contents
ori.w #HiIntMask,sr ; bump up the interrupt level (critical section)
moveq.l #findAny,d0 ; find the first available request to process
movea.l jvFind(a4),a0 ; get address of Find routine
jsr (a0) ; is there a request to process ?
beq.w @failed ; there is nothing to process, so exit
@foundRequest
movea.l d0,a5 ; point at the request to process
tst.l scsiPrivate(a5) ; does it have private storage ?
bne.s @arbitrate ; if so, then it's an autosense PB - just start it up
move.l nextPrivate(a4),d0 ; is there a private storage record available ?
beq.s @failed ; if not, we can't process, so exit
@havePrivate
movea.l d0,a0 ; point to private storage record
move.l a0,scsiPrivate(a5) ; point to SCSI Mgr private storage record
move.b #scsiFInUse,scsiFlags(a0) ; clear SCSI Mgr flags and mark record as in use
@findNextPrivate
move.l zeroReg,nextPrivate(a4) ; assume no more records are available
lea.l firstPrivate(a4),a1 ; point to head of private storage queue
@loop
movea.l scsiPrivLink(a1),a1 ; look at the next record
cmp.l a1,zeroReg ; are we done ?
beq.s @arbitrate ; nothing free ("nextPrivate" = nil), try to arbitrate
tst.b scsiFlags(a1) ; is this record in use ?
bne.s @loop ; it's in use, so check the next record
@setNextPrivate
move.l a1,nextPrivate(a4) ; this is the next available record
@arbitrate
btst.b #shwCbPwrMgr,G_Reserved0+3(a4) ; see if we have a Power Mgr
beq.s @PwrDone ; if not, exit
;
; This portion of code freezes the "Spin Down" timer for the hard disk, preventing
; the Power Mgr from cutting hard disk power during ANY SCSI transaction. <v1.1>
;
movea.l PmgrBase,a0 ; point to Pmgr locals
move.l zeroReg,LastHd(a0) ; always freeze the spin down timer <v1.1>
@startItUp ; power up the hard drive <v1.1>
with pmCommandRec
link a6,#-14 ; Power Mgr task, and 2-byte buffer <v1.1>
lea.l -14(a6),a0 ; point to the parameter block <v1.1>
move.w #powerCntl,pmCommand(a0) ; power up the SCSI +5V and +12V <v1.1>
move.w #1,pmLength(a0) ; one byte of data in buffer <v1.1>
move.w #hdOn*256,-2(a6) ; hdOn command <v1.1>
pea.l -2(a6) ; address of the send buffer <v1.1>
move.l (sp)+,pmSBuffer(a0) ; save the send buffer pointer <v1.1>
move.l zeroReg,pmRBuffer(a0) ; no data to be received <v1.1>
_PmgrOp ; send it off to the Power Mgr <v1.1>
unlk a6 ; release local storage <v1.1>
endwith
@PwrDone
movea.l jvArb(a4),a0 ; arbitrate for the SCSI bus
jsr (a0) ; go to the routine
beq.s @success ; arbitration succeeded, so we can continue
@arbFailed ; arbitration failed, but we'll get a LossOfBSY intr
move.l scsiPrivate(a5),a0 ; get pointer to private storage record
move.b zeroReg,scsiFlags(a0) ; mark this private storage record available
move.l zeroReg,scsiPrivate(a5) ; since it failed, release the private storage
tst.l nextPrivate(a4) ; check for a free private storage record
bne.s @failed ; there is already a free record
move.l a0,nextPrivate(a4) ; this record is free
@failed
move.w (sp)+,sr ; restore the status register (end critical section)
bra.s @afterPhase ; no phase handling will be necessary - no request
@success
move.w (sp)+,sr ; restore status register (end critical section)
move.l scsiPrivate(a5),a0 ; get pointer to private storage record
move.l scsiCmdBuf(a5),scsiCmdOffs(a0) ; copy of command buf ptr between interrupts
movea.l jvSetup(a4),a0 ; get address of Setup routine
jsr (a0) ; set up the current request (activeReq)
movea.l jvSel(a4),a0 ; select the target device
jsr (a0) ; go to the routine
beq.s @nextPhase ; selection succeeded, so start handling phases
@failedSelection
movea.l jvClearState(a4),a0 ; error in d0 <v1.5>
jsr (a0) ; blow away the SCSI PB <v1.5>
movea.l zeroReg,a5 ; no active request <v1.5>
bra.s @afterPhase ; no phase handling will be necessary - no request
@noActiveReq
@nextPhase
IF phaseLogging THEN
nop ; good place to jump to phase-logging routine
ENDIF
move.l zeroReg,d0 ; clear jump table offset in d0
@wfREQ
move.b sCSR(a3),d0 ; get current state
btst.l #bBSY,d0 ; is *BSY asserted ?
beq.s @afterPhase ; if not, we're done
btst.l #bREQ,d0 ; do we have a REQ (valid phase)
beq.s @wfREQ ; if not, wait
andi.b #aMSG+aCD+aIO,d0 ; d0 = {0,4,8,12,24,28} (16,20 undefined) <v1.2>
lea.l phaseTable(a4),a0 ; point to phase jump table in globals
movea.l 0(a0,d0),a0 ; get phase handler address from phase jump table
lsr.b #2,d0 ; set TCR so phase match bit is correct
move.b d0,sTCR+WrOffs(a3) ; set phase to match
addi.w #scsiDataOut,d0 ; create the constant for phase logging <v1.2>
cmp.l a5,zeroReg ; are we pointing to a request ? <v1.2>
beq.s @skip ; if not, can't log phase <v1.2>
move.w d0,scsiResult(a5) ; reflect the current phase in the PB
@skip
jsr (a0) ; go to phase-handling routine
btst.b #waitingForIRQ,state1(a4) ; is DMA in progress ? (waiting for an intr) <v1.5>
bne.s @afterPhase ; if so, then we should exit
tst.l activeReq(a4) ; is there still a current request ?
bne.s @nextPhase ; if so, handle next phase
@afterPhase
tst.b discID(a4) ; are any ID's disconnected ?
beq.s @endWhile ; if not, no need to enable reselection interrupts
move.b G_ID(a4),sSER+WrOffs(a3) ; enable reselection interrupts
;
; do {
; } while ( (pageFault && noInterrupts) || (dTask.dtParm != 0) ||
; (!waitingForIRQ && nextPrivate && scsiQHead && !Find(any,discID)) ); <v1.5>
;
@endWhile
tst.l pageFault(a4) ; check pointer to page-fault parameter block
beq.s @chkIntrPending ; if no page fault, any other interrupts pending ?
move.w sr,d0 ; get a copy of the CPU's interrupt level
andi.w #HiIntMask,d0 ; get CPU's interrupt level
cmpi.w #scsiIntMask,d0 ; compare SCSI interrupt level to the CPU's
bhs.s @goBack ; no interrupts, so keep looping
@chkIntrPending
tst.l dTask+dtParm(a4) ; are there any interrupt to be serviced ?
bne.s @goBack ; there are interrupts left to service (keep looping)
@chkOtherRequests
btst.b #waitingForIRQ,state1(a4) ; are we "waiting for interrupt" ? <v1.5>
bne.s @exit ; if so, we should exit (we'll get an intr) <v1.5>
tst.l nextPrivate(a4) ; do we have private storage for a new request ?
beq.s @exit ; if not, we can't start up a new request
tst.l scsiQHead(a4) ; any requests left ?
beq.s @exit ; no requests in the queue
moveq.l #findAny,d0 ; find the first available request to process
movea.l jvFind(a4),a0 ; get address of Find routine
jsr (a0) ; is there a request to process ?
beq.s @exit ; there is nothing to process
@goBack
bra.w @doWhile ; keep going (don't exit the "while" loop)
@exit
tst.l activeReq(a4) ; are we processing a request ?
bne.s @enableIntr ; if so, the timer was already set (by "Setup")
movea.l jvSetTimer(a4),a0 ; addr of timer setup routine
move.l zeroReg,d0 ; assume no more timing needed
tst.l scsiQHead(a4) ; are there any more requests ?
beq.s @setTimer ; if there are not, stop the timer
move.w #WDTO,d0 ; watchdog timeout
@setTimer
jsr (a0) ; set the timer
btst.b #shwCbPwrMgr,G_Reserved0+3(a4) ; do we have a Power Mgr chip ?
beq.s @PwrDone2 ; if not, skip
;
; This portion of code "starts" the "Spin Down" timer for the hard disk.
; The timer value is the value of Ticks when the last SCSI I/O completed.
; Whenever the SCSI bus is used, the timer is reset. <v1.1>
; When it goes off, "DoSpinDown" turns off the hard drive power. <v1.1>
;
; If there are any requests in the queue, leave the hard disk power on. <v1.1>
;
@chkPower
tst.l scsiQHead(a4) ; anything left in the queue ? <v1.1>
bne.s @PwrDone2 ; if so, don't start the timer <v1.1>
move.l PmgrBase,a0 ; point to Pmgr locals <v1.4>
move.l Ticks,LastHd(A0) ; indicate last activity for ID 0 <v1.4>
@PwrDone2
@enableIntr
moveq.l #1,d0 ; enable interrupts
move.l jvDisEnable(a4),a0 ; addr of interrupt disable/enable routine
jsr (a0) ; enable all SCSI interrupts
movem.l (sp)+,intrRegs ; restore registers
IF dbSymbols THEN
unlk a6
rts ; finished
dc.b 'SCSIDT '
ELSE
rts
ENDIF
;
; Command
;
; Called by: SCSIDT
;
; Function: Sends "scsiCmdLen" command bytes to the target. <v1.2>
; If BSY disappears, it's assumed we timed out and the bus was reset to recover.
;
Command:
move.l scsiPrivate(a5),a0 ; get ptr to private storage <v1.3>
move.l scsiCmdOffs(a0),a2 ; get the current command buffer ptr <v1.3>
move.l zeroReg,d2 ; clear lower word of count <v1.2>
move.b scsiCmdLen(a5),d2 ; get number of command bytes <v1.2>
subq.l #1,d2 ; setup for a DBRA <v1.2>
@nextCmdByte
@wfREQ
move.b sCSR(a3),d0 ; look at the bus
btst.l #bBSY,d0 ; is BSY asserted ?
beq.s @errExit ; if not, won't send command (a5 may be invalid)<v1.3>
btst.l #bREQ,d0 ; wait for REQ
beq.s @wfREQ
btst.b #bPM,sBSR(a3) ; check for phase match
beq.s @exit ; out of phase, so exit <v1.2>
@inPhase
move.b (a2)+,sODR+WrOffs(a3) ; load byte
move.b #iDB,sICR+WrOffs(a3) ; assert the data bus
move.b #iACK+iDB,sICR+WrOffs(a3) ; set *ACK
@wfnREQ
btst.b #bBSY,sCSR(a3) ; do we still have BSY ? <v1.2>
beq.s @errExit ; if not, get out <v1.3>
btst.b #bREQ,sCSR(a3) ; wait for no REQ
bne.s @wfnREQ
move.b zeroReg,sICR+WrOffs(a3) ; deassert *ACK and *DB
dbra.w d2,@nextCmdByte ; send the next command byte <v1.2>
;
; This is a hack for the Tape Backup 40SC drive. It asserts an extra REQ on
; 10-byte commands, which appears as a request for an 11th byte (that's what
; the SCSI standard says!) We shouldn't leave until this extra REQ goes away.<v1.2>
; This delay will wait at least 256ms for a phase change before continuing. <start>
;
move.l zeroReg,d0 ; clear upper word
move.w TimeSCSIDB,d0 ; get SCSI DBRA's/ms timing constant
lsl.l #8,d0 ; wait at least 256ms for the tape drive
move.l d0,d2 ; set up d2 as high word
swap d2
@cmdHack
btst.b #bBSY,sCSR(a3) ; is BSY still asserted ?
beq.s @exit ; if not, leave
btst.b #bPM,sBSR(a3) ; are we still in phase ?
dbeq.w d0,@cmdHack ; loop until phase mismatch or timeout <v1.2>
dbeq.w d2,@cmdHack ; high word of timeout <end>
@exit
move.l a2,scsiCmdOffs(a0) ; save current offset into the cmd buffer <v1.3>
@errExit ; if we lost BSY, a5 may be invalid <v1.3>
rts
;
; StatusPhase
;
; Called by: SCSIDT
;
; On entry: a4 - ptr to SCSI Mgr globals
; a5 - ptr to active SCSI request parameter block
;
; Function: Reads only 1 status byte from the target.
;
StatusPhase:
@wfREQ
move.b sCSR(a3),d0 ; look at the bus
btst.l #bBSY,d0 ; is BSY asserted ?
beq.s @exit ; no BSY - we won't get status (a5 may be invalid)
btst.l #bREQ,d0 ; wait for REQ
beq.s @wfREQ
btst.b #bPM,sBSR(a3) ; check for phase match
beq.s @exit ; we're out of phase
@inPhase
move.b sCDR(a3),d0 ; get the status byte
move.b #iACK,sICR+WrOffs(a3) ; set *ACK
@wfnREQ
btst.b #bBSY,sCSR(a3) ; do we have BSY ? <v1.2>
beq.s @exit ; if not, exit <v1.2>
btst.b #bREQ,sCSR(a3) ; wait for no REQ
bne.s @wfnREQ
move.b zeroReg,sICR+WrOffs(a3) ; deassert *ACK
@done
move.b d0,scsiStatus(a5) ; store the status byte
movea.l scsiPrivate(a5),a0 ; point at private storage for this request
btst.b #scsiBAutoSns,scsiFlags(a0) ; is this the auto sense param block ?
bne.s @exit ; don't note passage through Status phase
@normal
bset.b #scsiBStatus,scsiFlags(a0) ; note passage through status phase
@exit
rts
;
; MsgIn and MsgOut - handle the "Message In" and "Message Out" phases
;
; Called by: PhaseHnd
;
; Calls: None.
;
; Function: Transfers message bytes to and from the target in programmed I/O mode.
;
; Note: In the new SCSI Manager, the message support is as follows:
;
; Message In: Message Out:
;
; Command Complete Supported
; Extended Messages
; Extended Identify Msg Reject Never sent
; Modify Data Ptr Msg Reject
; Synchronous DTR Msg Reject Never sent
; Wide DTR Msg Reject Never sent
; Reserved Msg Reject Never sent
; Vendor Unique Msg Reject Never sent
; Save Data Pointer Supported
; Restore Pointers Supported
; Disconnect (In) Supported
; Disconnect (Out) Never sent
; Init Detected Err Msg Reject Never sent
; Abort Msg Reject Supported
; Message Reject Supported Supported
; No Operation Msg Reject Supported
; Message Parity Err Msg Reject Never sent
; Linked Cmd Complete Msg Reject (Currently)
; Lnkd Cmd Cmpl w/Flag Msg Reject (Currently)
; Bus Device Reset Msg Reject Supported
; Clear Queue Msg Reject Never sent
; Initiate Recovery Msg Reject Never sent
; Release Recovery Msg Reject Never sent
; Mundane Queue Tag Msg Reject Never sent
; Head of Queue Tag Msg Reject Never sent
; Ordered Queue Tag Msg Reject Never sent
; Abort Tag Msg Reject Never sent
; Ignore Wide Residue Msg Reject
; Identify Supported Supported
; Reserved (0D) Msg Reject Never sent
; Reserved (11-1F) Msg Reject Never sent
; Reserved (25-7F) Msg Reject Never sent
;
;
; This is the Message In phase handler.
;
MsgIn:
@getNextByte
move.l zeroReg,d0 ; clear out offset
@wfREQ
move.b sCSR(a3),d0 ; look at the bus
btst.l #bBSY,d0 ; is BSY asserted ?
beq.s @msgInDone ; BSY is gone -- we won't get a message byte
btst.l #bREQ,d0 ; wait for REQ
beq.s @wfREQ
btst.b #bPM,sBSR(a3) ; check for phase match
beq.s @msgInDone ; lost phase match
@inPhase
move.b sCDR(a3),d0 ; get byte
move.b #iACK,sICR+WrOffs(a3) ; set *ACK
@wfnREQ
btst.b #bREQ,sCSR(a3) ; wait for no REQ <from C846/PA081/03Mar87>
bne.s @wfnREQ
@chkIdentify
btst.l #scsiIdentBit,d0 ; check for "Identify" message (bit 7 set)
beq.s @chkInvalid ; if not "Identify", is it valid ?
move.b d0,d1 ; hold onto the "Identify" message byte
movea.l jvIdentifyIn(a4),a0 ; address of Identify (In) message handler
jsr (a0) ; handle the Identify message
bra.s @msgHandled ; check if there is another message to handle
@chkInvalid
cmpi.b #numMsgVct-1,d0 ; is it in range of the message routing table ?
bhi.s @msgInvalid ; if not, it's invalid
@msgValid
lea.l msgInTbl(a4),a0 ; point to base of message byte routing table
move.b 0(a0,d0.l),d0 ; get offset into msgTable
lea.l msgTable(a4),a0 ; point to message handler jump table
movea.l 0(a0,d0.l),a0 ; get address of message byte handler
jsr (a0) ; go handle the message byte
@msgHandled
btst.b #bPM,sBSR(a3) ; still in message in phase ?
bne.s @getNextByte ; keep going
@msgInDone
rts
@msgInvalid
movea.l jvInvalidIn(a4),a0 ; otherwise, start sending a Message Reject
jsr (a0) ; go to the routine
bra.s @msgHandled ; see if there is anything else to do
msgInvalidIn:
movea.l scsiPrivate(a5),a0 ; point at SCSI Mgr private storage
move.b #scsiMOMR,scsiState(a0) ; set state machine to "Msg Reject" state
move.b #iATN+iACK,sICR+wrOffs(a3) ; assert ATN to start Msg Reject sequence
move.b #iATN,sICR+wrOffs(a3) ; release ACK to complete handshake
rts
msgIdentifyIn:
move.l zeroReg,d0 ; clear the "findAnyFlag" in d0
move.b dTask+dtParm+3(a4),d0 ; get the reselection ID mask from SCSIIntHnd
move.b zeroReg,dTask+dtParm+3(a4) ; clear reselect ID mask (this intr has been serviced)
rol.w #8,d0 ; upper byte of d0.w = ID mask; d0.b = 0
andi.b #LUNMask,d1 ; leave LUN in d1
bset.l d1,d0 ; set the bit in the LUN bitmap for Find
movea.l jvFind(a4),a0 ; get address of Find routine
jsr (a0) ; go find that request
movea.l d0,a5 ; point to the parameter block
bne.s @clearDiscLUN ; we found a real parameter block
@PBTimedOut ; A timed-out target has reconnected.
move.l jvClearBus(a4),a0 ; get address of ClearBus routine
jsr (a0) ; abort this target (a5=0) (go to a BUS FREE)
rts ; get out of here (BUS FREE condition)
@clearDiscLUN
move.l zeroReg,d0 ; clear out d0 (including the "findAnyFlag")
move.b scsiReqID(a5),d0 ; use ID in d0 as offset into "discLUN[]"
move.b scsiReqLUN(a5),d1 ; restore the LUN in d1 as offset into bitmap
lea.l discLUN(a4),a0 ; point at disconnected LUN bitmap
bclr.b d1,0(a0,d0.w) ; mark LUN as no longer disconnected
tst.b 0(a0,d0.w) ; are any more LUN's disconnected for this ID ?
bne.s @continue ; if so, don't clear the disconnected ID bit
bclr.b d0,discID(a4) ; this is no longer a valid reselecting ID
@continue
movea.l jvSetup(a4),a0 ; get address of Setup routine
jsr (a0) ; setup the current request
movea.l jvRestorePtrs(a4),a0 ; implied Restore Pointers message
jmp (a0) ; perform the Restore Pointers message
msgCmdComplete:
movea.l scsiPrivate(a5),a0 ; point at private storage for this request
btst.b #scsiBAutoSns,scsiFlags(a0) ; is this an auto Request Sense PB ?
bne.w @autoSense ; if so, restore original contents
cmp.l scsiSnsBuf(a5),zeroReg ; was automatic sense disabled ?
beq.w @completed ; if disabled, just leave
@checkStat
move.b scsiStatus(a5),d0 ; get the status byte
andi.b #statusByteCode,d0 ; mask off vendor-unique bits in status byte
btst.l #statusRsrvBit,d0 ; is the reserved bit set ?
bne.w @completed ; if so, might not be "Check Condition" - exit
cmpi.b #statusChkCond,d0 ; "Check Condition" status ?
bne.w @completed ; no, so we're done
;
; At this point, the user wants an automatic REQUEST SENSE command, and the
; original command has failed. It is necessary to save all the fields in the
; user's parameter block that will be altered when handling the Request Sense
; command. The fields are saved in a portion of the dummy parameter block.
;
@startAutoSense
lea.l dummy(a4),a0 ; ptr to the dummy parameter block
;
; scsiQLink, scsiVersion, and scsiResult are not saved or changed.
; scsiBus, scsiReqID, and scsiReqLUN are not saved or changed.
; This leaves the auto Request Sense parameter block in its original
; position in the request queue.
;
move.l scsiCompletion(a5),scsiCompletion(a0) ; save the completion address
move.l zeroReg,scsiCompletion(a5) ; no completion routine for Req Sense
;
; scsiUsrData, scsiReqTO, scsiUsrFlags, and scsiSelTO are not saved or changed.
;
move.l scsiMgrFlags(a5),scsiMgrFlags(a0) ; save original SCSI flags
move.l #scsiFNoDisc,scsiMgrFlags(a5) ; sync virtual slow read, no disconnect
move.l scsiLinkCmd(a5),scsiLinkCmd(a0) ; save linked command pointer
move.l zeroReg,scsiLinkCmd(a5) ; no linked commands
@setupCmd
lea.l RSCmd(a4),a1 ; point to "Request Sense" command
move.b scsiSnsLen(a5),allocLen(a1) ; fill in allocation length
move.l scsiCmdBuf(a5),scsiCmdBuf(a0) ; save the command buffer pointer
move.l a1,scsiCmdBuf(a5) ; perform a "Request Sense"
@setupDC
move.l scsiDataLen(a5),scsiDataLen(a0) ; save the original data byte count
move.l zeroReg,d0 ; clear out d0
move.b scsiSnsLen(a5),d0 ; length of sense buffer
move.l d0,scsiDataLen(a5) ; sense buf length is data buf length
move.l scsiDataXfer(a5),scsiDataXfer(a0) ; save # of data bytes transferred
move.l zeroReg,scsiDataXfer(a5) ; no data bytes transferred, yet
lea.l RSDC(a4),a1 ; ptr to Req Sense data-chaining instr
move.l scsiDCInstr(a5),scsiDCInstr(a0) ; save data-chaining instruction ptr
move.l a1,scsiDCInstr(a5) ; data-chaining instr for Req Sense
move.l scsiSnsBuf(a5),d0 ; address of request's sense buffer
move.l d0,dcAddr(a1) ; buffer address in "dcAddr" field
move.l d0,dcStore(a1) ; buffer address in "dcStore" field
adda.w #dcSize,a1 ; point to next instruction
move.l scsiDataLen(a5),d0 ; number of bytes to transfer (set previously)
move.l d0,dcCount(a1) ; store in the "dcCount" field
move.l d0,dcStore(a1) ; store in the "dcStore" field
@setupAutoSense
move.l scsiSnsBuf(a5),scsiSnsBuf(a0) ; save the sense buffer pointer
move.l zeroReg,scsiSnsBuf(a5) ; disable auto Request Sense
move.b scsiSnsLen(a5),scsiSnsLen(a0) ; save the sense buffer length
move.b zeroReg,scsiSnsLen(a5) ; no sense buffer
;
; scsiSnsXfer is zero at this point, so it doesn't need initialization.
;
move.b scsiStatus(a5),scsiStatus(a0) ; hold on to the status byte
move.b #statusInitial,scsiStatus(a5) ; initial status byte value
@setupPrivate
;
; scsiTime, scsiMgrFlag1, scsiMgrFlag2, scsiMoveOffs,
; and the virtual/physical tranlation record are not changed or saved
;
movea.l scsiPrivate(a5),a1 ; point to private storage for this request
lea.l dummyPriv(a4),a0 ; point to dummy PB's private storage record
bset.b #scsiBAutoSns,scsiFlags(a1) ; mark this as the auto-Req Sense PB
move.b scsiState(a1),scsiState(a0) ; save the Message Out state variable
move.b #scsiMOID,scsiState(a1) ; back to normal "Identify" state
move.l scsiCmdOffs(a1),scsiCmdOffs(a0) ; save the temp command buffer ptr
move.l scsiCmdBuf(a5),scsiCmdOffs(a1) ; copy of scsiCmdBuf
move.l scsiDCSaved(a1),scsiDCSaved(a0) ; save the saved data pointer value
move.l zeroReg,scsiDCSaved(a1) ; start at beginning of table
move.l scsiDCOffs(a1),scsiDCOffs(a0) ; save the temp saved data pointer
move.l zeroReg,scsiDCOffs(a1) ; start at beginning of table
move.l zeroReg,activeReq(a4) ; the bus is currently free
move.b zeroReg,sICR+WrOffs(a3) ; deassert *ACK
bra.w @done ; done for now
@autoSense
;
; At this point, a5 is pointing at a "Auto Request Sense" parameter block.
; It is necessary to restore the fields in the parameter block to their
; original values.
;
@restorePB
lea.l dummy(a4),a0 ; point to the dummy parameter block
move.l scsiCompletion(a0),scsiCompletion(a5) ; restore completion routine ptr
move.l scsiMgrFlags(a0),scsiMgrFlags(a5) ; restore SCSI flags
move.l scsiLinkCmd(a0),scsiLinkCmd(a5) ; restore linked command pointer
move.l scsiCmdBuf(a0),scsiCmdBuf(a5) ; restore command buffer pointer
move.l scsiSnsBuf(a0),scsiSnsBuf(a5) ; restore sense buffer ptr
move.b scsiSnsLen(a0),scsiSnsLen(a5) ; restore sense buffer length
move.l scsiDataXfer(a5),d0 ; get number of bytes transferred
move.b d0,scsiSnsXfer(a5) ; save sense byte count in original PB
move.l scsiDataLen(a0),scsiDataLen(a5) ; restore data buffer length
move.l scsiDataXfer(a0),scsiDataXfer(a5) ; restore data transfer byte count
move.l scsiDCInstr(a0),scsiDCInstr(a5) ; restore data-chaining instruction ptr
move.b scsiStatus(a5),d0 ; get the status byte
move.b scsiStatus(a0),scsiStatus(a5) ; restore original status byte
andi.b #statusByteCode,d0 ; mask vendor-unique bits in Req Sense status
btst.l #statusRsrvBit,d0 ; is the reserved bit set ?
bne.s @restorePrivate ; if so, might not be "Check Cond" - continue
cmpi.b #statusChkCond,d0 ; "Check Condition" status ?
bne.s @restorePrivate ; no, so continue
move.b zeroReg,scsiSnsXfer(a5) ; Req Sense failed - report "no sense data"
@restorePrivate
movea.l scsiPrivate(a5),a1 ; point to private storage for this request
lea.l dummyPriv(a4),a0 ; point to dummy PB's private storage record
bclr.b #scsiBAutoSns,scsiFlags(a1) ; restore SCSI Mgr flags
move.b scsiState(a0),scsiState(a1) ; restore Message Out state variable
move.l scsiCmdOffs(a0),scsiCmdOffs(a1) ; restore temp command buffer ptr
move.l scsiDCSaved(a0),scsiDCSaved(a1) ; restore saved data ptr value
move.l scsiDCOffs(a0),scsiDCOffs(a1) ; restore temp saved data ptr value
@completed
moveq.l #noErr,d0 ; assume all is well
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
btst.b #scsiBStatus,scsiFlags(a0) ; is the Status byte valid ?
bne.s @validStatus ; if so, then exit
@noStatus
move.w #scsiNoStatusErr,d0 ; never received a Status byte <v1.2>
@validStatus
move.b zeroReg,sICR+WrOffs(a3) ; deassert ACK (avoids completion-routine deadlock)
movea.l jvClearState(a4),a0 ; clear SCSI Mgr state
jsr (a0) ; go to the routine
@linkLoop
move.l scsiLinkCmd(a5),d0 ; any linked commands left to be executed ?
movea.l d0,a5 ; point to the linked command <v1.5>
beq.s @done ; if not, we're truly done <v1.5>
move.w #scsiLinkFailErr,d0 ; this linked command was never started <v1.2>
jsr (a0) ; remove this request
bra.s @linkLoop ; keep checking
@done
rts ; continue
msgExtendedMsg:
;
; Currently, no extended messages are supported
;
msgSaveDataPtr:
move.l scsiDCInstr(a5),a0 ; point to data-chaining instructions
cmp.l a0,zeroReg ; is it nil ?
beq.s @dcStop ; if so, there is nothing to save
movea.l scsiPrivate(a5),a1 ; point to private storage for this request
move.l scsiDCOffs(a1),scsiDCSaved(a1) ; save the active "data" pointer state
moveq.l #dcLoop,d0 ; used for comparison
@repeat
move.l dcAddr(a0),d1 ; get the address or opcode
beq.s @dcStop ; if "dcStop", then we're done
cmp.l d0,d1 ; is it a "dcLoop" instruction ?
bne.s @save ; if not, save the address in "dcStore"
move.l dcCount(a0),d1 ; the loop count needs to be saved
@save
move.l d1,dcStore(a0) ; save the necessary value
adda.w #dcSize,a0 ; point to next data-chaining instruction
bra.s @repeat ; copy another instruction
@dcStop
move.b zeroReg,sICR+wrOffs(a3) ; deassert *ACK
rts ; continue
msgRestorePtrs:
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l scsiCmdBuf(a5),scsiCmdOffs(a0) ; restore command buffer pointer
move.l scsiDCSaved(a0),scsiDCOffs(a0) ; restore the active "data" pointer state
move.l scsiDCInstr(a5),a0 ; point to data-chaining instructions
cmp.l a0,zeroReg ; are there data-chaining instructions ?
beq.s @dcStop ; if not, then we're done
moveq.l #dcLoop,d0 ; used for comparison
@repeat
move.l dcStore(a0),d3 ; get the saved value
move.l dcAddr(a0),d1 ; get the address or opcode
beq.s @dcStop ; if "dcStop", then we're done
cmp.l d0,d1 ; is it a "dcLoop" instruction ?
bne.s @dcMove ; if not, it is a "dcMove" instruction
@dcLoop
move.l d3,dcCount(a0) ; restore the old count
bra.s @continue ; go to next instruction
@dcMove
move.l d3,dcAddr(a0) ; restore the old address
@continue
adda.w #dcSize,a0 ; point at next instruction
bra.s @repeat ; handle next instruction
@dcStop
move.b zeroReg,sICR+WrOffs(a3) ; deassert *ACK
rts ; continue
msgDisconnect:
move.l zeroReg,d1 ; clear out d1 for use as an offset
move.b scsiReqID(a5),d1 ; get ID of request
move.b scsiReqLUN(a5),d0 ; get the LUN of request
bset.b d1,discID(a4) ; mark ID as disconnected
lea.l discLUN(a4),a0 ; point to base of disconnected LUN table
bset.b d0,0(a0,d1.w) ; mark bit "scsiReqLUN" of "discLUN[scsiReqID]"
move.l zeroReg,activeReq(a4) ; no active request
move.w #scsiDisc,scsiResult(a5) ; mark this PB as disconnected
move.b zeroReg,sICR+WrOffs(a3) ; deassert *ACK
rts ; continue
msgMsgRejIn: ; for now, no processing if a Msg Rej occurs
IF dbSymbols THEN
_Debugger
ENDIF
move.b zeroReg,sICR+wrOffs(a3) ; deassert *ACK
rts
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
msgLCCF: ; Linked Command Complete with (or without) Flag
moveq.l #noErr,d0 ; assume all is well
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
btst.b #scsiBStatus,scsiFlags(a0) ; is the Status byte valid ?
bne.s @validStatus ; if so, then continue
@noStatus
move.w #scsiNoStatusErr,d0 ; never received a Status byte <v1.2>
@validStatus
move.b zeroReg,sICR+WrOffs(a3) ; deassert ACK (avoids completion-routine deadlock)
movea.l jvClearState(a4),a0 ; clear SCSI Mgr state
jsr (a0) ; go to the routine
IF dbSymbols THEN
_Debugger ; what now ??? (find a private record, Setup, etc.)
ENDIF
rts ; continue
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
;
; This is the Message Out phase handler
;
MsgOut:
@loop
moveq.l #scsiMONOP,d0 ; assume unexpected Message Out phase <v1.2>
tst.l activeReq(a4) ; check if we're processing a request <v1.2>
beq.s @handleMsgOut ; send No Op's - unexpected Msg Out phase <v1.2>
movea.l scsiPrivate(a5),a0 ; point at private storage for this request
move.l zeroReg,d0 ; clear out the offset in d0
move.b scsiState(a0),d0 ; get the Message Out state for this request
cmpi.b #numMsgVct-1,d0 ; is it in range of the Msg Out state routing table ?
bls.s @handleMsgOut ; if so, call the Msg Out handler
@invalid
moveq.l #scsiMOInvalid,d0 ; Msg Out state machine messed up - kill the request
@handleMsgOut
lea.l msgOutTbl(a4),a0 ; point to base of Message Out routing table
move.b 0(a0,d0.l),d0 ; get offset into msgTable
lea.l msgTable(a4),a0 ; point to msgTable
movea.l 0(a0,d0.l),a0 ; get address of message out state handler
jsr (a0) ; go handle the message out state
@wfReq
move.b sCSR(a3),d0 ; get a snapshot of the bus <v1.5>
btst.l #bBSY,d0 ; do we have BSY ? <v1.5>
beq.s @msgOutDone ; if not, get out <v1.2>
btst.l #bREQ,d0 ; wait for REQ before checking phase <v1.5>
beq.s @wfReq ; no REQ yet
btst.b #bPM,sBSR(a3) ; are we still in Message Out phase ?
bne.s @loop ; if so, continue running the state machine <v1.5>
@msgOutDone
rts
msgIdentifyOut:
;
; first send an Identify message to the appropriate LUN
;
move.b scsiReqLUN(a5),d1 ; set up "Identify" message
ori.b #scsiIdentFlg+scsiReselFlg,d1 ; "Identify and Disconnect" message
move.l scsiMgrFlags(a5),d0 ; get SCSI flags
btst.l #scsiBImmed,d0 ; is it a page-fault request ? <v1.1>
bne.s @discNotOK ; sorry, page-faults can't disconnect
btst.l #scsiBNoDisc,d0 ; check if disconnect is disabled
beq.s @discOK ; we may disconnect
@discNotOK
bclr.l #scsiReselBit,d1 ; no disconnect -- simple "Identify" message
@discOK
move.b d1,sODR+wrOffs(a3) ; send out Identify message
@chkBusDevReset
btst.l #scsiBBusDevRst,d0 ; should we send a Bus Device Reset msg ? <v1.5>
beq.s @noMessagesLeft ; if not, we're done <start>
move.b #iATN+iDB,sICR+wrOffs(a3) ; assert data bus - keep ATN on
move.b #iATN+iACK+iDB,sICR+wrOffs(a3) ; assert ACK and data bus
@wfnREQ0
btst.b #bREQ,sCSR(a3) ; wait for REQ to go away
bne.s @wfnREQ0 ; keep waiting
move.b #iATN,sICR+wrOffs(a3) ; keep ATN on - release data bus and ACK
move.l scsiPrivate(a5),a0 ; point to private storage
move.b #scsiMOBDR,scsiState(a0) ; set state machine to Bus Device Reset <end>
rts ; we're done for now <v1.5>
@noMessagesLeft
move.b zeroReg,sICR+wrOffs(a3) ; deassert ATN line - no messages left
move.b #iDB,sICR+wrOffs(a3) ; assert data bus
move.b #iACK+iDB,sICR+wrOffs(a3) ; assert ACK and data bus
@wfnREQ1
btst.b #bREQ,sCSR(a3) ; wait for REQ to go away
bne.s @wfnREQ1 ; keep waiting
move.b zeroReg,sICR+wrOffs(a3) ; deassert data bus and release ACK - we're done
move.l scsiPrivate(a5),a0 ; point to private storage
move.b #scsiMONOP,scsiState(a0) ; set state machine to No Operation
rts ; we're done for now
msgBusDevRst:
; <v1.5>
; send a Bus Device Reset message <start>
;
move.b #scsiBusDevReset,sODR+wrOffs(a3) ; send out a Bus Device Reset message
@noMessagesLeft
move.b zeroReg,sICR+wrOffs(a3) ; deassert ATN line - no messages left
move.b #iDB,sICR+wrOffs(a3) ; assert data bus
move.b #iACK+iDB,sICR+wrOffs(a3) ; assert ACK and data bus
@wfnREQ1
btst.b #bREQ,sCSR(a3) ; wait for REQ to go away
bne.s @wfnREQ1 ; keep waiting
move.b zeroReg,sICR+wrOffs(a3) ; deassert data bus and release ACK - we're done
@wfBusFree
move.b sCSR(a3),d0 ; get a snapshot of the bus
btst.l #bBSY,d0 ; is BSY still asserted ?
beq.s @accepted ; the device took the Bus Dev Reset message
btst.l #bREQ,d0 ; do we have a REQ ?
beq.s @wfBusFree ; if not, keep waiting for Bus Free
@rejected
;
; The target has asserted REQ, which probably means it is preparing to send a
; Message Reject in response to the Bus Device Reset message. We simply let the
; phase handler take care of this case.
;
move.l scsiPrivate(a5),a0 ; point to private storage
move.b #scsiMONOP,scsiState(a0) ; set state machine to No Operation
rts ; we're done for now
@accepted
move.w #scsiNoStatusErr,d0 ; error indicating the device took the BDR message
movea.l jvClearState(a4),a0 ; remove this request
jsr (a0)
movea.l zeroReg,a5 ; no active request <end>
rts ; we're done <v1.5>
msgMsgRejOut:
IF dbSymbols THEN
_Debugger ; performing a Message Reject sequence
ENDIF
move.b #scsiMsgReject,sODR+wrOffs(a3) ; send out a Message Reject message
move.b zeroReg,sICR+wrOffs(a3) ; deassert ATN line - no messages left
move.b #iDB,sICR+wrOffs(a3) ; assert data bus
move.b #iACK+iDB,sICR+wrOffs(a3) ; assert ACK and data bus
@wfnREQ1
btst.b #bREQ,sCSR(a3) ; wait for REQ to go away
bne.s @wfnREQ1 ; keep waiting
move.b zeroReg,sICR+wrOffs(a3) ; deassert data bus and release ACK - we're done
move.l scsiPrivate(a5),a0 ; point to private storage
move.b #scsiMONOP,scsiState(a0) ; set state machine to No Operation
rts ; see what happens next
msgInvalidOut:
IF dbSymbols THEN
_Debugger ; bad state machine state
ENDIF
rts ; get out
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
msgKillio: ; could be more elegant ???
IF dbSymbols THEN
_Debugger ; Send an Abort and/or Bus Device Reset message
ENDIF
movea.l jvResetBus(a4),a0 ; addr of reset SCSI bus routine
jsr (a0) ; reset the bus, killing all requests
rts ; we're done
IF padForOverPatch THEN ; padding <v1.3>
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
dc.w $4e71, $4e71, $4e71, $4e71 ; nop's
ENDIF
msgNoOp: ; If the target still wants message bytes, we send "No Operation" bytes.
move.b #scsiNoOperation,sODR+wrOffs(a3) ; send out a No Op message
move.b #iDB,sICR+wrOffs(a3) ; assert data bus
move.b #iACK+iDB,sICR+wrOffs(a3) ; assert ACK and data bus
@wfnREQ3
btst.b #bREQ,sCSR(a3) ; wait for REQ to go away
bne.s @wfnREQ3 ; keep waiting
move.b zeroReg,sICR+wrOffs(a3) ; release ACK and data bus
rts ; check phase, and repeat, if necessary
;
; DataIO -- data-chaining instruction interpreter
;
; Called by: PhaseHnd
;
; Calls: Transfer and the data transfer routines.
;
; On entry: a3 - ptr to SCSI read base address
; a4 - ptr to SCSI Mgr globals
; a5 - ptr to active request
; d5 - offset into the data-chaining instructions
;
; Function: Performs the data-chaining interpretation and calls the data transfer routines.
;
DataIO:
move.w #scsiOverrunErr,d0 ; assume we shouldn't be in a data phase <v1.2>
tst.l scsiDataLen(a5) ; is a data phase expected ? <v1.2>
beq.s @errorExit ; if not, exit <v1.2>
@expected
moveq.l #scsiFFast+scsiFWrite+scsiFPhysical,d4 ; d4 = mask data transfer bits <v1.2>
and.l scsiMgrFlags(a5),d4 ; d4 = {0,4,...,24,28} jump table offset <v1.2>
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l scsiDCOffs(a0),d5 ; get offset into data-chaining instr
btst.b #bIO,sCSR(a3) ; check if I/O is asserted
bne.s @input ; if asserted, we're in Data In phase
@output
btst.l #scsiBWrite,d4 ; test the write flag
bne.s @dc_interpreter ; we're in the correct phase - start data transfer
@badPhase
move.w #scsiTransferErr,d0 ; data phase doesn't match read/write flag <v1.2>
@errorExit
movea.l jvLogError(a4),a0 ; note SCSI error in PB <v1.2>
jsr (a0) ; go to the routine <v1.2>
bra.w @exit ; exit
@input
btst.l #scsiBWrite,d4 ; test the write flag
bne.s @badPhase ; we're in the wrong phase
move.b #iDMA,sMR+WrOffs(a3) ; set DMA mode <C478/10Dec86>
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ? <2.8>
beq.s @DMAdone ; if not, skip it <2.8>
move.l #iHSKEN,sDCTRL(a3) ; turn on hhsk <C914>
@DMAdone
move.b zeroReg,sIDMArx+WrOffs(a3) ; start read DMA <C478/10Dec86>
@dc_interpreter
move.l scsiDCInstr(a5),a1 ; point to the data-chaining buffer
@start
move.l dcCount(a1,d5.l),d2 ; get count
move.l dcAddr(a1,d5.l),d0 ; get the data-chaining instruction
beq.w @dcStop ; if d0 is 0, then it is "dcStop"
move.l d0,a2 ; hold on to it in case it's an address
addq.l #1,d0 ; check if it is a "dcLoop" instruction
beq.w @dcLoop ; if d0 was -1, then it is "dcLoop"
@dcMove ; a2 contains the address
btst.b #bDMAR,sBSR(a3) ; do we have a DRQ ? <v1.1><v1.2>
bne.s @chkPhase ; if so, we can check the phase
btst.b #bREQ,sCSR(a3) ; do we have a REQ ?
beq.s @dcMove ; if not, we can't accurately check the phase
@chkPhase
btst.b #bPM,sBSR(a3) ; are we still in phase ?
beq.w @dcStop ; if not, get out
@adjust
;
; If a data-chaining instruction was only partially executed (due to a Disconnect),
; then the "scsiMoveOffs" field in the private storage will be nonzero, reflecting
; the number of bytes already transferred during the instruction. If it is nonzero,
; then the residual data to be transferred is the DCInstr value minus the amount
; already transferred, and the address pointer is incremented by the amount of data
; already transferred. <v1.1>
;
move.l scsiMoveOffs(a0),d0 ; have we partially executed this DCInstr ? <v1.1>
beq.s @chkOverrun ; if not, no adjustment necessary <v1.1>
sub.l d0,d2 ; transfer only residual amount of data <v1.1>
adda.l d0,a2 ; place data in the remaining buffer space <v1.1>
@chkOverrun
move.l scsiDataXfer(a5),d0 ; get number of bytes transferred already
add.l d2,d0 ; expected bytes transferred after "dcMove"
cmp.l scsiDataLen(a5),d0 ; will we go over the limit ?
bls.s @noOverrun ; if not, start the transfer
move.w #scsiOverrunErr,d0 ; possible overrun condition <v1.2>
movea.l jvLogError(a4),a0 ; note SCSI error information in the PB <v1.2>
jsr (a0) ; go to the routine <v1.2>
bra.s @dcStop ; exit
@noOverrun
movea.l jvTransfer(a4),a0 ; transfer the data
jsr (a0) ; go to the routine
@update
tst.w d0 ; check if an error occurred
bne.s @dcStop ; we're finished for now
btst.b #waitingForIRQ,state1(a4) ; are we DMA-ing ? <v1.5>
bne.s @exit ; if so, bail for now <v1.1><v1.2>
movea.l scsiPrivate(a5),a0 ; point to private storage <v1.1><v1.2>
add.l d1,scsiDataXfer(a5) ; update bytes transferred <v1.1><v1.2>
cmp.l d1,d2 ; did we transfer everything ? <v1.1>
beq.s @all ; if so, update PB and start next DCInstr <v1.1>
@notAll
add.l d1,scsiMoveOffs(a0) ; update the amount of data transferred <v1.1><v1.2>
bra.s @start ; restart current instruction <v1.2>
@all
move.l dcOffset(a1,d5.l),d0 ; get offset for the buffer address <v1.2>
add.l d0,dcAddr(a1,d5.l) ; add the offset to the buffer address <v1.2>
move.l zeroReg,scsiMoveOffs(a0) ; clear the amount of data transferred <v1.1>
@nextInstr
moveq.l #dcSize,d0
add.l d0,d5 ; point to next instruction
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l d5,scsiDCOffs(a0) ; save value in PB for consistency
bra.s @start ; handle next instruction
@dcLoop
tst.l d2 ; check for loop count of zero
beq.s @nextInstr ; count exhausted, go to the next instr
subq.l #1,d2 ; decrement
move.l d2,dcCount(a1,d5.l) ; update count
beq.s @nextInstr ; if 0, count exhausted, go to next instr
move.l dcOffset(a1,d5.l),d0 ; get offset
asl.l #4,d0 ; multiply by 16
add.l d0,d5 ; adjust ptr to next instruction
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l d5,scsiDCOffs(a0) ; save value in PB for consistency
bra.s @start ; handle next instruction
@dcStop
move.b zeroReg,sMR+WrOffs(a3) ; turn off pDMA to be safe
btst.b #sSCSIDMAExists,G_Reserved0+3(a4) ; do we have a SCSI DMA chip ? <2.8>
beq.s @DMAdone2 ; if not, skip it <2.8>
move.l #iINTREN,sDCTRL(a3) ; turn off hardware-handshaking <v1.2>
@DMAdone2
@exit
moveq.l #noErr,d0 ; no problem <v1.2>
rts
;
; DataDMA -- DMA-related data-chaining interpreter for use after first DMA <v1.2>
; <start>
; Called by: SCSIDT
;
; Calls: The data transfer routines.
;
; On entry: a3 - ptr to SCSI read base address
; a4 - ptr to SCSI Mgr globals
; a5 - ptr to active request
;
; Function: Performs the data-chaining interpretation and calls the data transfer routines.
;
DataDMA:
move.l scsiDCInstr(a5),a1 ; point to data-chaining instructions
move.l scsiPrivate(a5),a0 ; point to private storage for the request
lea.l scsiVAddr(a0),a2 ; point to GetPhysical translation table
move.l scsiDCOffs(a0),d5 ; get offset into data-chaining instructions
move.l 12(a2),d1 ; get number of bytes transferred by DMA
add.l d1,scsiDataXfer(a5) ; update bytes transferred
tst.l 4(a2) ; is the residual equal to zero ?
beq.s @all ; if so, we're done with this dcInstr
@notAll ; not at data-chaining instruction boundary
add.l d1,scsiMoveOffs(a0) ; add DMA bytes to form new offset into dcInstr
bra.s @start ; restart the current instruction
@all ; at data-chaining instruction boundary
move.l dcOffset(a1,d5.l),d0 ; get offset for buffer address
add.l d0,dcAddr(a1,d5.l) ; add offset to buffer address
move.l zeroReg,scsiMoveOffs(a0) ; clear the amount of data transferred
@nextInstr
moveq.l #dcSize,d0 ; point to next dcInstr
add.l d0,d5
move.l d5,scsiDCOffs(a0) ; save offset into data-chaining table
@start
move.l dcCount(a1,d5.l),d2 ; get count
move.l dcAddr(a1,d5.l),d0 ; get the data-chaining instruction
beq.w @dcStop ; if d0 is 0, then it is "dcStop"
move.l d0,a2 ; hold on to it in case it's an address
addq.l #1,d0 ; check if it is a "dcLoop" instruction
beq.w @dcLoop ; if d0 was -1, then it is "dcLoop"
@dcMove ; a2 contains the address
btst.b #bDMAR,sBSR(a3) ; do we have a DRQ ?
bne.s @chkPhase ; if so, we can check the phase
btst.b #bREQ,sCSR(a3) ; do we have a REQ ?
beq.s @dcMove ; if not, we can't accurately check the phase
@chkPhase
btst.b #bPM,sBSR(a3) ; are we still in phase ?
beq.w @dcStop ; if not, get out
@adjust
;
; If a data-chaining instruction was only partially executed (due to a Disconnect),
; then the "scsiMoveOffs" field in the private storage will be nonzero, reflecting
; the number of bytes already transferred during the instruction. If it is nonzero,
; then the residual data to be transferred is the DCInstr value minus the amount
; already transferred, and the address pointer is incremented by the amount of data
; already transferred.
;
move.l scsiMoveOffs(a0),d0 ; have we partially executed this DCInstr ?
beq.s @chkOverrun ; if not, no adjustment necessary
sub.l d0,d2 ; transfer only residual amount of data
adda.l d0,a2 ; place data in the remaining buffer space
@chkOverrun
move.l scsiDataXfer(a5),d0 ; get number of bytes transferred already
add.l d2,d0 ; expected bytes transferred after "dcMove"
cmp.l scsiDataLen(a5),d0 ; will we go over the limit ?
bls.s @noOverrun ; if not, start the transfer
move.w #scsiOverrunErr,d0 ; possible overrun condition
movea.l jvLogError(a4),a0 ; note SCSI error information in the PB
jsr (a0) ; go to the routine
bra.s @dcStop ; exit
@noOverrun
moveq.l #noErr,d0 ; assume no error
move.l d2,d1 ; make a copy of the count - is it zero ?
beq.w @all ; if so, we've finished this instruction
moveq.l #scsiFFast+scsiFWrite+scsiFPhysical,d4 ; d4 = mask to leave data transfer bits
and.l scsiMgrFlags(a5),d4 ; d4 = {0,4,...,24,28} jump table offset
lea.l dataTable(a4),a0 ; point to data transfer table in globals
movea.l 0(a0,d4.l),a0 ; get the routine address
jmp (a0) ; go to the routine - it will return to SCSIDT
@dcLoop
tst.l d2 ; check for loop count of zero
beq.w @nextInstr ; count exhausted, go to the next instr
subq.l #1,d2 ; decrement
move.l d2,dcCount(a1,d5.l) ; update count
beq.w @nextInstr ; if 0, count exhausted, go to next instr
move.l dcOffset(a1,d5.l),d0 ; get offset
asl.l #4,d0 ; multiply by 16
add.l d0,d5 ; adjust ptr to next instruction
movea.l scsiPrivate(a5),a0 ; point to private storage for this request
move.l d5,scsiDCOffs(a0) ; save value in PB for consistency
bra.w @start ; handle next instruction
@dcStop
move.b zeroReg,sMR+WrOffs(a3) ; turn off pDMA to be safe
move.l #iINTREN,sDCTRL(a3) ; turn off hardware-handshaking
moveq.l #noErr,d0 ; no problem <end>
rts
ENDWITH
END