view level1/modules/llide.asm @ 2662:07cc32e3d96f

More fixes
author Boisy Pitre <boisy.pitre@nuance.com>
date Fri, 13 Jul 2012 23:11:00 -0500
parents 17d43fd29ee2
children
line wrap: on
line source

*******************************************************************
* llide - Low-level IDE driver
*
* $Id$
*
* This low level driver works with both ATA and ATAPI devices.
*
* The type of device (ATA or ATAPI) is automatically detected
* by the 'IOSetup' routine.  Additionally, an ATA device is
* further detected as either an LBA or CHS device.
*
* Since only two physical drives are allowed (master/slave),
* there is a two entry "per drive static storage" that indicates
* if a drive has been initialized, its type (ATAPI or ATA, LBA or
* CHS), and if ATA, its geometry.
*
* Edt/Rev  YYYY/MM/DD  Modified by
* Comment
* ------------------------------------------------------------------
*     1    2004/04/08  Boisy G. Pitre
* Created.
*
*     2    2005/07/23  Christopher R. Hawks
* Fixes for persnickity ATAPI CDROMs.
*
*     3    2005/08/21  Christopher R. Hawks
* More fixes.
*
*     4    2005/12/13  Boisy G. Pitre
* Moved SS.VarSect code into RBSuper for performance

               NAM       llide               
               TTL       Low-level IDE driver

               IFP1      
               USE       defsfile
               USE       rbsuper.d
               USE       ide.d
               ENDC      

tylg           SET       Sbrtn+Objct
atrv           SET       ReEnt+rev
rev            SET       4


RW12           SET       0                   Use READ12/WRITE12 ATAPI commands (1 = yes)
WAITTIME       SET       10                  BUSY wait time (in approximate seconds)

*
* Status Register Flip/Mask Values
*
NBUSYDRDY      EQU       (BusyBit|DrdyBit)*256+(DrdyBit)
NBUSY          EQU       (BusyBit)*256+$00
NBUSYDRQ       EQU       (BusyBit|DrqBit)*256+(DrqBit)
NBUSYNDRQ      EQU       (BusyBit|DrqBit)*256+$00

               MOD       eom,name,tylg,atrv,start,0

               IFNE      RW12
READCODE       EQU       A$READ2
WRITCODE       EQU       A$WRITE2
               ELSE      
READCODE       EQU       A$READ
WRITCODE       EQU       A$WRITE
               ENDC      

NumRetries     EQU       8

* Low-level driver static memory area
               ORG       V.LLMem
* Master static storage
V.Master       RMB       1                   status byte (ATAPI or ATA (CHS or LBA))
               RMB       2                   Cylinders (CHS) or Bits 31-16 of LBA
               RMB       1                   Sides (CHS) or Bits 15-8 of LBA
               RMB       2                   Sectors (CHS) or Bits 7-0 of LBA
* Slave drive static storage
V.Slave        RMB       1
               RMB       2
               RMB       1
               RMB       2
* ATAPI Command Packet
V.ATAPICmd     RMB       18
V.SnsData      EQU       V.ATAPICmd          Sense Data is shared with ATAPI command block
* The following values are for device 0 and 1 respectively:
* Bit 0 = device inited (0 = false, 1 = true)
* Bit 1 = device type (0 = ATA, 1 = ATAPI)
* Bit 2 = device mode (0 = CHS, 1 = LBA)
V.CurStat      RMB       1
V.Retries      RMB       1
V.WhichDv      RMB       1                   contains devhead selection (made by IOSetup)
V.PhySct       RMB       3                   local copy of physical sector passed (V.PhySct)
V.SctCnt       RMB       1                   local copy of physical sector passed (V.SectCnt)
V.Sectors      RMB       1                   number of sectors (harvested directly from drive query)
V.CurDTbl      RMB       2
V.ATAVct       RMB       2


name           FCS       /llide/

start          bra       ll_init
               nop       
               lbra      ll_read
               lbra      ll_write
               lbra      ll_getstat
               lbra      ll_setstat

* ll_init - Low level init routine
*
* Entry:
*    Y  = address of device descriptor
*    U  = address of low level device memory area
*
* Exit:
*    CC = carry set on error
*    B  = error code
*
* Note: This routine is called ONCE: for the first device
* IT IS NOT CALLED PER DEVICE!
*
ll_init                  
*         clrb
*         rts             


* ll_term - Low level term routine
*
* Entry:
*    Y  = address of device descriptor
*    U  = address of device memory area
*
* Exit:
*    CC = carry set on error
*    B  = error code
*
* Note: This routine is called ONCE: for the last device
* IT IS NOT CALLED PER DEVICE!
*
ll_term                  
               clrb      
               rts       


* Entry:   Y = address of per-drive static storage
ATADSize                 
               pshs      y,x,b               make room for space on stack (and save id byte)
* Determine if we are dealing with LBA or CHS
               bitb      #$04                LBA?
               bne       lba@
* Here we pull CHS values
chs@                     
               ldd       1,y                 get cylinders
               std       1,s
               lda       3,y                 get sides
               sta       ,s                  save sides on stack (B)
               ldd       4,y                 get sectors
               std       3,s                 save sectors/track on stack (Y)
               bra       m@
* Here we pull LBA values at words 60-61
lba@                     
               clr       ,s                  clear flag indicating LBA mdoe (B)
               ldd       3,y                 get bits 15-0
               std       3,s                 save bits 15-0 on stack (Y)
               ldd       1,y                 get bits 31-16
               std       1,s                 save bits 31-16 (X)
m@             lda       #$02                512 bytes/sector
ex@            puls      b,x,y,pc


* SSDSize - Get a disk medium's size
*
* GetStat Call SS.DSize:
*
* Entry: B = SS.DSize
* Exit:  Carry = 1; error with code in B
*        Carry = 0:
*          IF B = 0
*            A = Sector Size (1 = 256, 2 = 512, 4 = 1024, 8 = 2048)
*            X = Number of Sectors (bits 31-16)
*            Y = Number of Sectors (Bits 15-0)
*          IF B != 0
*            A = Sector Size (1 = 256, 2 = 512, 4 = 1024, 8 = 2048)
*            X = Number of Logical Cylinders
*            B = Number of Logical Sides
*            Y = Number of Logical Sectors/Track
*  
SSDSize        pshs      u,y
               bsr       DSize
               bcs       ex@
               ldu       ,s                  get path desc in U
               ldu       PD.RGS,u
               std       R$D,u
               stx       R$X,u
               sty       R$Y,u
               clrb      
ex@            puls      y,u,pc

DSize          lbsr      IOSetup
               bcs       ex@
* Determine if this device is ATAPI or ATA
               bitb      #$02                ATAPI?
               lbeq      ATADSize            no, it's ATA
* Note - for ATAPI version of SS.DSize, we use the obsolete
* READ CAPACITY call because it works on CD-ROMs, as opposed to
* READ FORMAT CAPACITIES.
ATAPIDSize               
               lbsr      ATAPIPreSend        prepare packet
* Populate packet buffer with STOP code and Eject
*         ldd   #$230C		ATAPI READ FORMAT CAPACITIES Code
               lda       #$25                ATAPI READ CAPACITY Code
               sta       V.ATAPICmd,u        write it
*         stb   V.ATAPICmd+8,u	and allocation length
* Send to data port
               bsr       ATAPISend           send command
               bcs       ex@
* Read 8 bytes of format capacity data
*         ldb   #6
               ldb       #4
               pshs      b
               leay      V.SnsData,u
read@          lda       DataReg,x
               ldb       Latch,x
               std       ,y++
               dec       ,s
               bne       read@
               puls      b
               ldx       V.SnsData+0,u
               ldy       V.SnsData+2,u
               leay      1,y
               bcc       b@
               leax      1,x
b@             lda       V.SnsData+6,u
               clrb      
ex@            rts       

*         ldy   PD.RGS,y
*         ldd   V.SnsData+0,u	get bits 31-16
*         std   R$X,y
*         ldd   V.SnsData+2,u	get bits 15-0
*         addd  #$0001		add 1
*         std   R$Y,y
*         bcc   b@
*         ldd   R$X,y
*         addd  #$0001
*         std   R$X,y
*b@       lda   V.SnsData+6,u	get bits 15-8 of block size
*         sta   R$A,y
*         clr   R$B,y		signal that this is LBA mode
*         clrb
*ex@      rts        


* ll_getstat - Low level GetStat routine
*
* Entry:
*    Y  = address of path descriptor
*    U  = address of device memory area
*
* Exit:
*    CC = carry set on error
*    B  = error code
*
ll_getstat               
               ldx       PD.RGS,y
               lda       R$B,x
               cmpa      #SS.DSize
               beq       SSDSize
               ldb       #E$UnkSvc
               coma      
ex1            rts       


* StopUnit - Park a drive
*
* ATA   Devices: This is a No-Op.
* ATAPI Devices: A STOP UNIT command is issued to the device.
*                (ejects media on ATAPI removable devices)
StopUnit       lbsr      IOSetup
               bcs       ex1
* Determine if this device is ATAPI or ATA
               bitb      #$02                ATAPI?
               beq       ex1                 no, ignore...
               lbsr      ATAPIPreSend        prepare packet
* Populate packet buffer with STOP code and Eject
ok@            ldd       #A$STOP*256+$02     ATAPI STOP Code and Eject byte
               sta       V.ATAPICmd,u        write it and RSV to zero
               stb       V.ATAPICmd+4,u
* Send to data port
               bra       ATAPISend           send command


* ll_setstat - Low level SetStat routine
*
* Entry:
*    Y  = address of path descriptor
*    U  = address of device memory area
*
* Exit:
*    CC = carry set on error
*    B  = error code
*
ll_setstat               
               ldx       PD.RGS,y
               lda       R$B,x
               cmpa      #SS.SQD
               beq       StopUnit
               IFNE      0
               cmpa      #SS.DCmd
               bne       n@
               pshs      x                  save pointer to caller registers
               bsr       DCmd               call DCmd
               puls      x                  get pointer to caller registers
               sta       R$A,x              save status byte in A
               ENDC
n@             clrb      
ssex           rts       


               IFNE    0
BadType        comb
               ldb     #E$BTyp
               rts

* Entry:
*    X   = caller regs
*    Y   = path descriptor
*
*    R$B = SS.DCmd
*    R$X = Transfer buffer
*    R$Y = ATAPI command packet
DCmd                     
               pshs      y
               os9       F$ID                get the user ID of the calling process
               cmpy      #$0000              is it 0 (superuser)?
               puls      y
               bne       noperms             no, don't allow the call
               lbsr      IOSetup
** SS.DCmd only works with ATAPI devices.
               bitb      #$02                ATAPI?
               beq       BadType             branch if not
               ldy       R$X,x               get caller's transfer buffer
               sty       V.UTxBuf,u          save off in mem for later
               ldx       R$Y,x               get ptr to caller's command buffer
               IFGT      Level-1
               ldy       D.Proc              get current process ptr
               lda       P$Task,y            get task # for current process
               ldb       D.SysTsk            get system task #
               ldy       #ATAPIPkLn          max size of ATAPI command
               pshs      u                   save on stack
               leau      V.ATAPICmd,u        point to ATAPI command buffer in our statics
               os9       F$Move              copy from caller to temporary task
               puls      u
               bcs       ex                  error copying, exit
               ELSE      
               ldb       #ATAPIPkLn
               leay      V.ATAPICmd,u
cl@            lda       ,x+
               sta       ,y+
               decb      
               bne       cl@
               ENDC      
               ldy       V.PORT-UOFFSET,u    get hw address (because we overwrite Y earlier)
*               inc       V.OS9Err,u          we want real errors returned
               inc       V.CchDirty,u        and make cache dirty
*               leax      retry@,pcr
*               stx       V.RetryVct,u
retry@         lbsr      ATAPISend
               bcs       ex
               IFGT      Level-1
               ldx       D.Proc              get current process ptr
               ldb       P$Task,x            get task # for current process
               ENDC      
               ldx       V.UTxBuf,u

msgloop@       lbsr      Wait4REQ            wait for REQ to be asserted
               bita      #CMD                command phase?
               lbne      PostXfr             yes, return
io@            bita      #INOUT              data coming in or going out?
               bne       in@                 branch if coming in...
               IFGT      Level-1
               os9       F$LDABX
               leax      1,x
               ELSE      
               lda       ,x+
               ENDC      
               sta       SCSIDATA,y
               bra       msgloop@
in@            lda       SCSIDATA,y
               IFGT      Level-1
               os9       F$STABX
               leax      1,x
               ELSE      
               sta       ,x+
               ENDC      
               bra       msgloop@
               ENDC

* ATAPISend - Sends the command packet to the device
*
* Entry:   X = HW address
*          V.WhichDv = DevHead device selection value
* Exit:    Carry = 1; error code in B
*          Carry = 0; command successfully sent
ATAPISend                
* First, select the device and wait for /BUSY
               lda       V.WhichDv,u
               sta       DevHead,x           select device
* ATAPI says we wait for !BUSY
               ldd       #NBUSYDRDY          wait for NBUSY and DEVREADY too - CRH
               lbsr      StatusWait          wait for proper condition
               bcs       timeout             branch if error
               lda       #NumRetries         get retry count
               sta       V.Retries,u         and save
retry@                   
               clr       Features,x          clear feature byte
               clr       SectCnt,x           clear TAG field
               ldd       #$FFFF              maximum read in PIO mode
               std       CylLow,x
               lda       #$A0                ATAPI PACKET CODE
               sta       Command,x           write it to device
* Check for error
               ldd       #NBUSYDRQ           /BUSY and DRQ
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
* Send to data port
               lda       #6                  packet size / 2
               pshs      a,y
               leay      V.ATAPICmd,u
l@             ldd       ,y++
               stb       Latch,x
               sta       DataReg,x
               dec       ,s
               bne       l@
               puls      a,y
* Added by CRH - Some drives require that we wait for much more than
* 400ns.  Hence this code forces a slow-down and checks the status
* to see if things are ok.
               clrb      
slow@          decb                          CRH wait for much more than 400ns
               bne       slow@
               lda       Status,x            CRH Sometimes error on first try
               lsra      
               bcc       ok@
               dec       V.Retries,u
               bne       retry@
* As per ATA/ATAPI-6 spec (T13/1410D Revision 3A), page 161, we wait for /BUSY
* and DRQ after sending a command
* That IS NOT the case when sending START/STOP unit commands, so we don't check
* for DRQ.
ok@            ldd       #NBUSY              /BUSY
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lda       #A$STOP             CRH see above
               cmpa      V.ATAPICmd,u
               beq       nodata@
               ldd       #NBUSYDRQ           /BUSY and DRQ
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
nodata@        lsra                          shift in CHECK CONDITION bit
               bcc       ex@                 branch if ok
* Note: if the device returns error, DRQ won't be set...
err@           lbsr      HandleSenseKey      go check sense key
               bcs       ex@
ret@           dec       V.Retries,u         else retry until terminal
               bne       retry@              continue retrying
timeout        comb                          set carry
               ldb       #E$DevBsy+4         SHOULD BE AN E$TimeOut ERROR!
ex@            rts       

* Prepare the ATAPI Packet
ATAPIPreSend             
               pshs      x
* Clear 12 byte packet.
               leax      V.ATAPICmd,u
               ldb       #12
c@             clr       ,x+
               decb      
               bne       c@
ex@            puls      x,pc


** ATAPI REQUEST SENSE Command
** Should only be called for ATAPI devices
*ReqSense 
*         bsr    ATAPIPreSend	prepare packet
** Populate packet buffer with REQUEST SENSE
*         ldd   #$0312		ATAPI REQUEST SENSE Code and allocation length byte
*         sta   V.ATAPICmd,u
*         stb   V.ATAPICmd+4,u
** Send to data port
*         lbsr  ATAPISend	send command
*         lda   Status,x		get status code
*         lsra			shift in CHECK CONDITION bit
*         bcs   ex@		branch if ok
** Read 18 bytes of sense data
*         ldb   #$12
*         pshs  y,b
*         leay  V.SnsData,u
*read@    lda   DataReg,x 
*         ldb   Latch,x   
*         std   ,y++      
*         dec   ,s        
*         bne   read@     
*         puls  b,y,pc
*ex@      rts


* IOSetup - Sets up the device for I/O
*
* The device is selected (master or slave), then the device is
* checked for previous initialization.
*
* If the device has not been initialized, it is queried for its
* mode (ATAPI/ATA, LBA or CHS) and size.  That information is
* saved in the driver's static storage for later use by other
* routines.
*
* Entry:  Y = path descriptor pointer
*         U = static memory pointer
* Exit:   B = status byte for device
*         X = HW address
*         Y = pointer to current device table
IOSetup        ldx       V.PORT-UOFFSET,u    get hw address
               lda       PD.DNS,y            get device ID bit
               lsra                          shift device ID into carry
               bcs       slave@
               lda       #%10100000          master byte
               leay      V.Master,u
               bra       t@
slave@         lda       #%10110000          slave byte
               leay      V.Slave,u           else point to slave status byte
t@             sty       V.CurDTbl,u
               sta       V.WhichDv,u         save for later
* Select the device -- on power-up, the status register is usually 0
               sta       DevHead,x           select device
* According to page 320 of the ATA/ATAPI-6 document, we must wait for BOTH
* BUSY and DRQ to be clear before proceeding. (HI2: Device_Select State)
               ldd       #NBUSYNDRQ          /BUSY and /DRQ
               lbsr      StatusWait          wait for proper condition
               lbcs      ex@                 branch if error
* Determine if this device has already been initialized
               tst       ,y                  test device's stat byte
               lbne      initdone            if not zero, init already done
* Here we must initialize the device by IDENTIFYING it.
* First, try sending the ATA IDENTIFY DRIVE code
               lda       #$EC                ATA identify command
               sta       Command,x           write it
               ldd       #NBUSY              /BUSY
               lbsr      StatusWait          wait for proper condition
               lbcs      ex@                 branch if error
* Check if there's an error
               lsra                          shift error bit into carry
               bcc       ATAIdent            if no error, then probably ATA
* If here, we got an error sending $EC, so try ATAPI's $A1
               lda       #$A1
               sta       Command,x
               clrb      
slow@          decb                          CRH wait for much more than 400ns
               bne       slow@
               ldd       #NBUSY
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lsra                          shift error bit into carry
               bcs       timeout             if not error, we're ok
               ldd       #NBUSYDRQ
               lbsr      StatusWait          wait for proper condition
               lbcs      ex@                 branch if error
* Here, we have identified an ATAPI device.
ATAPIIdent               
               ldb       #$03                ATAPI
               stb       ,y
* We flush the ATAPI data but don't reference it
dread@         ldb       DataReg,x           CRH flush ALL bytes
*         ldb   Latch,x 	but save time by not reading latch
               lda       Status,x
               anda      #8
               bne       dread@
               bra       initdone
* ATAIdent - process an ATA device
* This routine is called by IOSetup when it deduces that the device
* being queried is an ATA device.  This device is called ONCE -- the
* first time the device is accessed.
* This routine will set up our per-drive static storage to indicate that
* it is an ATA device.  It will also determine if it is an LBA or CHS mode
* device, and save the appropriate CHS or LBA sector values.
ATAIdent                 
               ldd       #NBUSYDRDY          /BUSY and DRDY
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
* Harvest C/H/S and LBA sector values.
               ldb       DataReg,x           ignore bytes 0-1
               ldb       DataReg,x           bytes 2-3 = no. of cylinders
               lda       Latch,x
               std       1,y                 save cylinders in our private static area
               ldb       DataReg,x           ignore bytes 4-5
               ldb       DataReg,x           bytes 6-7 = no. of heads
               lda       Latch,x
               stb       3,y                 save sides on stack (B)
               ldb       DataReg,x           ignore bytes 8-9
               ldb       DataReg,x           ignore bytes 10-11
               ldb       DataReg,x           bytes 12-13 = no. of sectors/track
               lda       Latch,x
               std       4,y                 save sectors/track on stack (Y)
* Throw away the next 42 (7-48) words
               ldb       #43
l@             tst       DataReg,x
               lda       Latch,x
               decb      
               bne       l@
* A holds byte with LBA bit
               incb                          B was 0, now 1
               anda      #%00000010          LBA allowed on this drive?
               beq       nope@
               orb       #$04                set LBA mode
               stb       ,y                  save updated status byte
* Since we're LBA mode, get the number of LBA sectors in words 60-61
               ldb       #10                 skip to the LBA sectors (words 60-61)
more@          tst       DataReg,x           simply read the data register like this...
               decb      
               bne       more@
               ldb       DataReg,x           get word 60
               lda       Latch,x
               std       3,y
               ldb       DataReg,x           and 61
               lda       Latch,x
               std       1,y
               lda       #256-61             how many words we have left
               bra       left@               go on.
nope@          stb       ,y                  save updated status byte
* Read remaining 256-50 words
               lda       #256-50
left@          ldb       DataReg,x
               deca      
               bne       left@
initdone       ldb       ,y                  get status byte of drive
               clra                          clear carry
ex@            rts       


* ATAPI Write Routine -- Independent of ATA Read
ATAPIWrite               
               lbsr      ATAPIPreSend        prepare packet
* Populate packet buffer with WRITE code and sector information
again@         ldb       V.PhySct,u          get bits 23-16 of sector
               stb       V.ATAPICmd+3,u
               ldd       V.PhySct+1,u        get bits 15-0 of sector
               std       V.ATAPICmd+4,u
               ldd       #WRITCODE*256+$01   ATAPI WRITE Code and transfer length
               sta       V.ATAPICmd,u        write it
               IFNE      RW12
               stb       V.ATAPICmd+9,u      write to byte 9
               ELSE      
               stb       V.ATAPICmd+8,u      write to byte 8
               ENDC      
* Send to data port
               lbsr      ATAPISend           send command
               bcs       ex@
* Shift data from device
o@             pshs      d
               lda       V.Log2Phys,u
               sta       1,s                 set up our logical sector counter
inc@           clr       ,s                  set up our byte counter
wr@            ldd       ,y++
               stb       Latch,x
               sta       DataReg,x
               inc       ,s
               bpl       wr@
               dec       1,s
               bne       inc@
               puls      d
* Increment physical sector
               inc       V.PhySct+2,u
               bcc       go@
               inc       V.PhySct+1,u
               bcc       go@
               inc       V.PhySct,u
go@            dec       V.SctCnt,u          decrement # of hw sectors to read
               bne       again@              if not zero, do it again
               ldd       #NBUSY              /BUSY
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lsra                          error bit set?
               bcc       ex@                 yep...
               bsr       HandleSenseKey
ex@            puls      x,pc



* ATAPI Read Routine -- Independent of ATA Read
ATAPIRead                
               ldy       V.CchPSpot,u        get pointer to spot in cache to put sector
               lbsr      ATAPIPreSend        do command packet setup stuff
* Populate packet buffer with READ code and sector information
again@         ldb       V.PhySct,u          get,u bits 23-16 of sector
               stb       V.ATAPICmd+3,u
               ldd       V.PhySct+1,u        get bits 15-0 of sector
               std       V.ATAPICmd+4,u
               ldd       #READCODE*256+$01   ATAPI Read Code and transfer length
               sta       V.ATAPICmd,u        write it and RSV to zero
               IFNE      RW12
               stb       V.ATAPICmd+9,u      write to byte 9
               ELSE      
               stb       V.ATAPICmd+8,u      write to byte 8
               ENDC      
* Send to data port
               lbsr      ATAPISend           send command
               bcs       ex@
* Shift data from device
o@             pshs      d
               lda       V.Log2Phys,u
               sta       1,s                 set up our logical sector counter
inc@           clr       ,s                  set up our byte counter
read@          lda       DataReg,x
               ldb       Latch,x
               std       ,y++
               inc       ,s
               bpl       read@
               dec       1,s
               bne       inc@
               puls      d
* Increment physical sector
               inc       V.PhySct+2,u
               bcc       go@
               inc       V.PhySct+1,u
               bcc       go@
               inc       V.PhySct,u
go@            dec       V.SctCnt,u          decrement # of hw sectors to read
               bne       again@              if not zero, do it again
               ldd       #NBUSY              /BUSY
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lsra                          error bit set?
               bcc       ex@                 nope...
               bsr       HandleSenseKey
ex@            puls      x,pc


* Handle ATAPI Sense Key
* If the resulting error in the look-up table is zero,
* we return with carry clear
* Returns: B =  0 (carry clear, no error)
*          B != 1 (carry set, error)
HandleSenseKey           
               pshs      x,a
               ldb       ErrorReg,x          get error register value
               lsrb                          shift sense key into place
               lsrb      
               lsrb      
               lsrb      
               leax      SenseMap,pcr        point to Sense Key Map
               clra                          clear carry
               ldb       b,x                 get appropriate error
               beq       ok@                 if error is zero, return ok
               coma                          set carry
ok@            puls      a,x,pc

* ll_read - Low level read routine
*
* Entry:
*    Registers:
*      Y  = address of path descriptor
*      U  = address of device memory area
*    Static Variables of interest:
*      V.PhySct = starting physical sector to read from
*      V.SectCnt  = number of physical sectors to read
*      V.SectSize = physical sector size (0=256,1=512,2=1024,3=2048)
*      V.CchPSpot = address where physical sector(s) will go
*
* Exit:
*    All registers may be modified
*    Static variables may NOT be modified
ll_read                  
               pshs      x                   make some space on the stack
               lbsr      IOSetup             initialize the device
               lbcs      ex@
* Copy V.PhySct and V.SectCnt to our local copy
* since we cannot modify them.
               lda       V.PhysSect,u
               ldy       V.PhysSect+1,u
               sta       V.PhySct,u
               sty       V.PhySct+1,u
               lda       V.SectCnt,u
               sta       V.SctCnt,u
               bitb      #$02                ATAPI device?
               lbne      ATAPIRead           yes, go do it
* ATA Read Routine
ATARead                  
* stb   V.CurStat,u	save status of current drive
               bitb      #$04                LBA drive?
               bne       lba@                branch if so
               leay      DoCHS,pcr           else point Y to CHS routine
               bra       skip@
lba@           leay      DoLBA,pcr
skip@          sty       V.ATAVct,u          save pointer
               ldy       V.CchPSpot,u        get pointer to spot in cache to put sector
loop@                    
               ldd       #NBUSY              /BUSY
               bsr       StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               jsr       [V.ATAVct,u]        do proper ATA preparation
               bcs       ex@                 branch if error
cont@          lda       #$01
               sta       SectCnt,x           store it
               lda       #S$READ
               sta       Command,x
               ldd       #NBUSY              /BUSY
               bsr       StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lsra                          error bit set?
               bcc       w@                  branch if not
               lbsr      ATAError
               bra       ex@
w@             ldd       #NBUSYDRQ           /BUSY and DRQ
               bsr       StatusWait          wait for proper condition
               bcs       ex@                 branch if error
               lda       V.Log2Phys,u
               sta       1,s                 set up our logical sector counter
inc@           clr       ,s                  set up our byte counter
read@          lda       DataReg,x
               ldb       Latch,x
               std       ,y++
               inc       ,s
               bpl       read@
               dec       1,s
               bne       inc@
* Increment physical sector
               inc       V.PhySct+2,u
               bcc       go@
               inc       V.PhySct+1,u
               bcc       go@
               inc       V.PhySct,u
go@            dec       V.SctCnt,u          decrement # of hw sectors to read
               bne       loop@               if not zero, do it again
               clrb      
ex@            puls      x,pc


*
* Convert LSN to LBA values
*
* Entry:  V.PhySct = bits 23-0 of LSN
*         X      = ptr to hardware
*
* Exit:   CHS values placed directly in HW
*
DoLBA          lda       V.WhichDv,u         get devhead value populated by IOSetup (CHS mode)
               ora       #%01000000          OR in LBA bit
               sta       DevHead,x
               ldd       #NBUSYNDRQ          /BUSY and /DRQ
               bsr       StatusWait
               bcs       ex@
               ldb       V.PhySct,u          get bits 23-16 of sector
               stb       CylHigh,x           store it
               ldd       V.PhySct+1,u        get bits 15-0 of sector
               stb       SectNum,x           store it
               sta       CylLow,x
ex@            rts       


* Wait for a set of conditions in the status register to be TRUE
* This yields a delay of about 4 seconds.
*
* Entry: X = HW address
*        A = flip (if bit set, that bit is tested)
*        B = mask (result must match this byte)
* Exit:  A = status
StatusWait               
               pshs      y,b,a
               IFEQ      Level-1
               ldb       #WAITTIME/2
               ldy       #$0000
               ELSE      
               ldb       #WAITTIME
               ldy       #$0000
               ENDC      
l@             lda       Status,x
               anda      ,s                  apply flip
               cmpa      1,s                 compare to mask
               bne       dec@                branch if not equal (not what we want)
               clrb                          clear carry
               bra       ok@
dec@           leay      -1,y                count down
               bne       l@
               decb                          decrement bits 23-16
               bpl       l@                  if >=0, keep going
err@           comb                          set carry
               ldb       #E$DevBsy
ok@            leas      2,s
               lda       Status,x            get status again
               puls      y,pc

* Wait for 1 tick (1/60 second)
*Delay1Tk
*         pshs  x
*         IFGT  Level-1
*         ldx   D.Proc		get proc descriptor
*         cmpx  D.SysPrc	system?
*         beq   hw@		yep, system cannot sleep
*         ENDC
*         ldx   D.AProcQ	get active proc queue
*         beq   hw@		if empty, do hard wait
*         ldx   #1
*         os9   F$Sleep		give up worst case: 1 tick (1/60 second)
*         puls  x,pc		return to caller
** In case we can't sleep... do a hard 1/60 second delay
*hw@ 
*         IFEQ  Level-1
*         ldx   #$E52E/2		(5) (4)
*         ELSE
*         ldx   #$E52E		(5) (4)
*         ENDC
*w@       leax  -1,x		(4+) (4+)
*         bne   w@		(3) (3)
*         puls  x,pc		return to caller
*


* ll_write - Low level write routine
*
* Entry:
*    Registers:
*      Y  = address of path descriptor
*      U  = address of device memory area
*    Static Variables of interest:
*      V.PhySct = starting physical sector to write to
*      V.SectCnt  = number of physical sectors to write
*      V.SectSize = physical sector size (0=256,1=512,2=1024,3=2048)
*      V.CchPSpot = address of data to write to device
*
* Exit:
*    All registers may be modified
*    Static variables may NOT be modified
ll_write                 
               pshs      x                   make some space on the stack
               lbsr      IOSetup             initialize the device
               lbcs      ex@
* Copy V.PhySct to our local copy
               lda       V.PhysSect,u
               ldy       V.PhysSect+1,u
               sta       V.PhySct,u
               sty       V.PhySct+1,u
               lda       V.SectCnt,u
               sta       V.SctCnt,u
*
               ldy       V.CchPSpot,u        get pointer to spot in cache where physical sector is
               bitb      #$02                ATAPI device?
               lbne      ATAPIWrite          yes, go do it
* ATA Write Routine
ATAWrite       stb       V.CurStat,u         save status of current drive
loop@                    
               ldd       #NBUSY              /BUSY
               bsr       StatusWait          wait for proper condition
               bcs       ex@                 branch if ok
* Check for LBA mode
               ldb       V.CurStat,u         get status of current drive
               bitb      #$04                LBA bit set?
               bne       lba@                branch if so
* Here, we use CHS
               bsr       DoCHS
               bcs       ex@
               bra       cont@
lba@           lbsr      DoLBA
cont@          lda       #$01
               sta       SectCnt,x           store it
               lda       #S$WRITE
               sta       Command,x
               ldd       #NBUSY              /BUSY
               bsr       StatusWait          wait for proper condition
               bcs       ex@                 branch if ok
               lsra                          error bit set?
               bcc       g@                  branch if not
               lbsr      ATAError
               bra       ex@
g@             ldd       #NBUSYDRQ           /BUSY and DRQ
               lbsr      StatusWait          wait for proper condition
               bcs       ex@                 branch if ok
again@         lda       V.Log2Phys,u
               sta       1,s                 set up our sector counter
inc@           clr       ,s                  set up our byte counter
wr@            ldd       ,y++
               stb       Latch,x
               sta       DataReg,x
               inc       ,s
               bpl       wr@
               dec       1,s
               bne       inc@
* Increment physical sector
               inc       V.PhySct+2,u
               bcc       go@
               inc       V.PhySct+1,u
               bcc       go@
               inc       V.PhySct,u
go@                      
               dec       V.SctCnt,u          decrement # of hw sectors to read
               bne       loop@               if not zero, do it again
               clrb      
ex@            puls      x,pc

*
* Convert LSN to C/H/S values and write to IDE hardware
*
* Entry:  V.PhySct = bits 23-0 of LSN
*         X      = ptr to hardware
*
* Exit:   CHS values placed directly in HW
*
DoCHS                    
* Select device
               lda       V.WhichDv,u         get devhead value made by IOSetup (already CHS)
               sta       DevHead,x
               ldd       #NBUSYNDRQ          /BUSY and /DRQ
               lbsr      StatusWait          wait for proper condition
               bcc       start@              branch if ok
               rts       
* Start computation
start@         pshs      y                   save original Y
               ldy       V.CurDTbl,u
               lda       3,y                 get device's head
               ldb       5,y                 and sector
               stb       V.Sectors,u
               mul                           multiply H*S
               beq       ZeroProd            if zero, error out
               pshs      d                   save product of H*S
               ldd       V.PhySct+1,u        get bits 15-0 of LSN
               ldy       #-1                 start Y at -1
               inc       V.PhySct,u          increment physical sector
* Here we are doing physLSN/(H*S) to get cylinder for physLSN
a@             leay      1,y                 increment count to compensate
               subd      ,s                  subtract (H*S) from physLSN
               bhs       a@                  if D>=0 then continue
               dec       V.PhySct,u          decrement phys sector bits 23-16
               bne       a@                  if not zero, continue divide
               addd      ,s++                add in (H*S) to make non-negative
               pshs      d                   D now holds cylinder, save on stack 
               tfr       y,d                 Y now holds cylinder value
               exg       a,b                 swap
               std       CylLow,x            store computed cylinder in HW
               puls      d                   restore saved cylinder
* Now we will compute the sector/head value
               ldy       #-1
b@             leay      1,y
               subb      V.Sectors,u
               sbca      #0
               bcc       b@
               addb      V.Sectors,u
               incb                          add 1 to B, which is sector
               stb       SectNum,x           store computed sector in HW
               tfr       y,d
               orb       DevHead,x           OR in with value written earlier
               stb       DevHead,x
               clrb      
               puls      y,pc
ZeroProd       ldb       #E$Sect
               coma      
               puls      y,pc

* ATAError - Checks the ATA error register and maps
*            to a NitrOS-9 error message.
*
* Called if the error bit in the status register is set.
ATAError                 
               lda       ErrorReg,x
               ldb       #7
l@             lsra      
               bcs       LookUp
               decb      
               bne       l@
LookUp         leax      Errs,pcr
x@             ldb       b,x
               coma      
               rts       

* This is the ATAPI Sense Key -> NitrOS-9 Error Table
* The Sense Key Table is on page 50 of the ATAPI Removable
* Rewritable Specification, Revision 1.3 Proposed.
* If an error number is zero, then no error is returned.
SenseMap       FCB       0                   sense key 0 (NO SENSE)
               FCB       0                   sense key 1 (RECOVERED ERROR)
               FCB       E$NotRdy            sense key 2 (NOT READY)
               FCB       E$Sect              sense key 3 (MEDIUM ERROR)
               FCB       E$Unit              sense key 4 (HARDWARE ERROR)
               FCB       E$IllArg            sense key 5 (ILLEGAL REQUEST)
               FCB       0                   sense key 6 (UNIT ATTENTION)
               FCB       E$WP                sense key 7 (DATA PROTECT)
               FCB       0                   sense key 8 (BLANK CHECK)
               FCB       0                   sense key 9 (VENDOR SPECIFIC)
               FCB       0                   sense key A (RESERVED)
               FCB       1                   sense key B (ABORTED COMMAND)
               FCB       0                   sense key C (RESERVED)
               FCB       0                   sense key D (VOLUME OVERFLOW)
               FCB       0                   sense key E (MISCOMPARE)
               FCB       0                   sense key F (RESERVED)

* ERROR REG Bit   0      1     2       3       4        5      6     7
Errs           FCB       E$Unit,E$CRC,E$UnkSvc,E$Sect,E$UnkSvc,E$DIDC,E$Seek,E$Sect

               EMOD      
eom            EQU       *
               END