view level1/modules/llcoco3fpga.asm @ 3139:96cb9d947e0b

New low-level driver for CoCo3FPGA SD card interface From Gary Becker. Add llcoco3fpga.asm to level1/modules
author Bill Pierce <merlinious999@gmail.com>
date Sat, 04 Feb 2017 11:06:18 +0100
parents
children c505ae3120c4
line wrap: on
line source

*******************************************************************
* llsd - Low-level SDHC/SD/MMC driver
*
* This driver talks to the SD card interface on the DE1 utilizing the CoCo3FPGA
*
* $FF64 (Control Register Write)
* Bit 7 : SD Enable
* Bit 6 : Enable Interrupt
* Bits 5-1 : not used
* Bit 0 : SPI Device Select #1 (DE1 SD Card)
* 
* $FF64 (Status Register Read)
* Bit 7 : IRQ Active
* Bits 6-2 : not used
* Bit 1 : SD Card Locked (Write Protect)
* Bit 0 : SD Installed
*
* $FF65 (Data Register R/W)
*
* Edt/Rev  YYYY/MM/DD  Modified by
* Comment
* ------------------------------------------------------------------
*     1    2012/06/16  Gary Becker
* Rewritten to work with 8 bit SPI on CoCo3FPGA
*     2    2015/06/06  Gary Becker
* Removed any code to enable multiple slots (single slot only)

*Level	EQU		2

		NAM		llcoco3fpga
		TTL		Low-level SDHC/SD/MMC driver

		USE		defsfile
		USE		rbsuper.d

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

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

		ORG		V.LLMem
* Low-level driver static memory area
SEC_CNT	RMB		1	Number of sectors to transfer
SEC_LOC	RMB		2	Where they are or where they go
SEC_ADD	RMB		3	LSN of sector
SDVersion	RMB		1	0 = Byte Addressable SD
* !0 = Sector Addressable SD
CMDStorage	RMB		1	Command storage area for read/write CMDs
SD_SEC_ADD	RMB		4	Four bytes because some devices are byte addressable
CMDCRC	RMB		1

**************************************
* Command bytes storage area
**************************************
CMD0		fcb		$40,$00,$00,$00,$00,$95
*CMD1		fcb		$41,$00,$00,$00,$00,$95
CMD8		fcb		$48,$00,$00,$01,$AA,$87
*CMD13	fcb		$4D,$00,$00,$00,$00,$95
CMD16		fcb		$50,$00,$00,$02,$00,$FF		was 95
ACMD41V1	fcb		$69,$00,$00,$00,$00,$FF		was 95
ACMD41V2	fcb		$69,$40,$00,$00,$00,$FF		was 95
CMD55		fcb		$77,$00,$00,$00,$00,$FF		was 95
CMD58		fcb		$7A,$00,$00,$00,$00,$FF		was 95

* Read/Write commands
CMDRead	EQU		$5100		Command to read a single block
CMDWrite	EQU		$5800		Command to write a sector
CMDEnd	EQU		$00FF		Every command ends with $95
* SPI Address Equates
* SPI Control Register
SPICTRL	EQU		0
SLOT_SEL_0	EQU		1
SPI_IRQ_EN	EQU		$40
SPI_EN	EQU		$80		Sets SPI enable and IRQ enable
* SPI Status Register
SPISTAT	EQU		0
CARD_DET_0	EQU		1
CARD_LOCK_0	EQU		2
IRQ_SLOT_0	EQU		$80
* SPI Transmit / Receive Register
SPIDAT	EQU		1
* Test 8 bit LED display
SPITRACE	EQU		$FF66

name		FCS		/llcoco3fpga/

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

EREAD
		ldd		#$0000+E$Read	A=Disable SPI Interface, but not CS
* B=Read Error
		lbra		RETERR

* ll_read - Low level read routine
*
* Entry:
*    Registers:
*      Y  = address of path descriptor
*      U  = address of device memory area
*    Static Variables of interest:
*      V.PhysSect = 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
*    Initialization errors are not flagged as an error
*    SDCards are hot pluggable and card might be plugged in later
ll_read
* Setup read command
		orcc		#IntMasks		disable interrupts
		ldx		V.Port-UOFFSET,u	Get address of hardware
		lda		V.SectCnt,u		Get number of sectors to read
		sta		SEC_CNT,u		Save it to our usable storage
		ldd		V.CchPSpot,u	get the location to copy the sector into
		std		SEC_LOC,u		Save it into our usable storage
		ldd		V.PhysSect,u	Copy Sector Adrress into our storage
		std		SEC_ADD,u
		lda		V.PhysSect+2,u
		sta		SEC_ADD+2,u
		lda		SPISTAT,x
		lsra
		bcc		EREAD			No card installed, so no reads
lphr		lda		SEC_CNT,u
		ldd		#CMDRead
		std		CMDStorage,u	Read command and clear MSB of address
		ldd		#CMDEnd
		std		SD_SEC_ADD+3,u	Clear LSB of address and CRC
rd_loop	lda		#(SPI_EN+SLOT_SEL_0)	but not IRQ enable
		bsr		LSNMap0		Setup the appropriate LSN value for the card, build command,
* setup SPI to access the card, and sends command
		bcs		EREAD			If we timed out, branch with error
		bne		EREAD			If the R1 was not 0

		ldy		SEC_LOC,u		Get the sector buffer address
		ldb		#$00			256 loops of 2 reads of 1 bytes is 512 bytes
lprd		lda		SPIDAT,x		Send FF (receive a byte) until we get FE
		cmpa		#$FE			Do we need more cycles??????
		nop
		bne		lprd
* Read the 512 Byte sector
* we need a minumum of 4 CPU cycles to read in the 8 bits
RDSectorLoop	lda	SPIDAT,x
		sta		,y+		? cycles ?????????????
		nop				might be too much ???????
		lda		SPIDAT,x
		sta		,y+
		decb
		bne		RDSectorLoop
* Get the last two bytes of the sector (CRC bytes)
		lda		SPIDAT,x	Send 2x FF to get the CRC
		sty		SEC_LOC,u	Save out buffer pointer
*		nop				Might be too many cycles ?????????
		lda		SPIDAT,x	We ignore the CRC
		dec		SEC_CNT,u	decrement # of hw sectors to read
		beq		finird	if zero, we are finished
*Increment sector number by 1 for sector addressable or $200 for byte addressable
incsec	inc		SEC_ADD+2,u	add one to 3 byte LSN
		bne		lphr		if we are at 0 then we need to add
		inc		SEC_ADD+1,u	the carry to the next byte
		bne		lphr
		inc		SEC_ADD,u
		bra		lphr

* No errors, exit
finird
		ldd		#$0000	Disable SPI and exit
		sta		SPICTRL,x
		andcc		#^(IntMasks+Carry)	Renable Interrupts and clear carry
		rts				return

**************
* LSNMap
* Take physical LSN and convert into SDHC/SD/MMC LSN
* SD/MMC uses a 32 bit byte mapping for the LSN, so we must shift the logical LSN up one bit
* and then clear the 4th byte to build the correct LSN string
* SDHC uses a 32 bit 512 byte sector mapping for the LSN.So there is no need to shift the LSN
* we can just write it as is and clear out the upper LSN byte, because we only get 3 bytes from coco for LSN
* does not preserve Reg A
**************
LSNMap0
		sta		SPICTRL,x
		nop
		lda		SPIDAT,x	Send 1 FF
		lda		SDVersion,u
		bne		secadd	GoTo Sector Address type
*Save the sector number into the command
		ldd		SEC_ADD+1,u	bytes 1 and 2 (middle and LSB)
		aslb				Byte address needs to be shifter one more bit
		rola
		std		SD_SEC_ADD+1,u	and stored in the first 3 bytes of a 4 byte address
		lda		SEC_ADD,u	MSB
		rola
		sta		SD_SEC_ADD,u
		bra		merge
secadd
		ldd		SEC_ADD+1,u		Save the sector number into our storage
		std		SD_SEC_ADD+2,u	Store in the last three bytes of the 4 byte address
		lda		SEC_ADD,u
		sta		SD_SEC_ADD+1,u
merge		lda		SPIDAT,x	Send 1 FF
LSNMap1
		leay		CMDStorage,u
* Fall through to cmdsend

* cmdsend - Sends a 6 byte command
* Entry:  X = HW addr
*         Y = Address of first byte of command sequence
* Exit:
* Registers preserved: all but A/B/X
cmdsend
		lda		,y
		sta		$FF66
		ldb		#6
cslp		lda		,y+
		sta		SPIDAT,x
		decb
		bne		cslp
* If finished, fall through to GETR1

* Entry:  X = HW addr
* Exit:   A = R1
*         CC.C = 0 OK
*         CC.C = 1 ERROR
* Registers preserved: all but A/B
GetR1
		andcc		#^Carry	Clear Carry
		ldb		#$00		Probably too much
lpgtr1	lda		SPIDAT,x
		bpl		finigtr1                
		decb
		bne		lpgtr1
		comb				set carry for error
finigtr1	rts

* ll_write - Low level write routine
*
* Entry:
*    Registers:
*      Y  = address of path descrip
*      U  = address of device memory area
*    Static Variables of interest:
*      V.PhysSect = 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
		orcc		#IntMasks	disable interrupts
		ldx		V.Port-UOFFSET,u	Get address of hardware
		lda		V.SectCnt,u		Get number of sectors to write`
		sta		SEC_CNT,u	Save it to our usable storage
		ldd		V.CchPSpot,u	get the location to of the sector send
		std		SEC_LOC,u	Save it into our usable storage
		ldd		V.PhysSect,u	Copy Sector Adrress into our storage
		std		SEC_ADD,u
		lda		V.PhysSect+2,u
		sta		SEC_ADD+2,u
		lda		SPISTAT,x
		lsra
		lbcc		EWRITE	No card installed, so no writes
		lsra
		lbcs		EWP	Write Protected, then exit with WP error
* The big read sector loop comes to here
lphw		ldd		#CMDWrite
		std		CMDStorage,u
		ldd		#CMDEnd
		std		SD_SEC_ADD+3,u	LSB of address and CRC
wr_loop	lda		#(SPI_EN+SLOT_SEL_0)
		bsr		LSNMap0	Setup the appropriate LSN value for the card, build command,
* setup SPI to access the card, and sends command
		bcs		EWRITE
		bne		EWRITE

		lda		SPIDAT,x	2 bytes >= 1 byte after R1
		nop				Might not be enough ?????
		nop
		lda		SPIDAT,x
		ldd		#$FE00	Start of sector byte and clear counter
		ldy		SEC_LOC,u	get the location of the sectors(s) to write
		sta		SPIDAT,x	Mark the start of the sector
		nop				Too much ???????
* Write the 512 Byte sector
WRSectorLoop	lda	,y+
		sta		SPIDAT,x
		nop
		lda		,y+
		sta		SPIDAT,x
		decb
		bne		WRSectorLoop
		lda		SPIDAT,x	send two FFs as CRC
		sty		SEC_LOC,u	Save out buffer pointer
		nop				Might not be enough ???????
		stb		SPIDAT,x	Second FF (send 0 to check)
		cmpa		#$E5		Response - Data accepted token
		beq		fnd0		First byte? if not, check four more bytes.
* Make sure the response was accepted
		lda		SPIDAT,x	This should be the data we got back during the write of the last CRC
*		nop				But we do send another ff so we can get the E5
		cmpa		#$E5		Response - Data accepted token
		beq		fnd0		First byte? if not, check three more bytes.
		lda		SPIDAT,x
		cmpa		#$E5		Response - Data accepted token if this is not it, then we have an issue
		beq		fnd0		First byte? if not, check two more bytes.
		lda		SPIDAT,x
		cmpa		#$E5		Response - Data accepted token if this is not it, then we have an issue
		beq		fnd0		First byte? if not, check one more byte.
		lda		SPIDAT,x
		cmpa		#$E5		Response - Data accepted token if this is not it, then we have an issue
		bne		EWRITE	Write error
* Check to see if the write is complete
fnd0		lda		SPIDAT,x
		beq		lpwr2
		bra		fnd0	Ths beq and bra could be a bne, but I want the extra cycles
lpwr2		lda		SPIDAT,x
		cmpa		#$FF
		beq		wfin
		bra		lpwr2
wfin		ldb		#10	Lets send 16 more FF just in case
finlp		lda		SPIDAT,x
		decb
		bne		finlp
		dec		SEC_CNT,u	decrement # of hw sectors to read
		beq		finiwr	if zero, we are finished
		inc		SEC_ADD+2,u	add one to 3 byte LSN
		bne		lphw		if we are at 0 then we need to add
		inc		SEC_ADD+1,u	the carry to the next byte
		lbne		lphw
		inc		SEC_ADD,u
		lbra		lphw
* No errors, exit
finiwr
		ldd		#$0000	Diable SPI and exit
		sta		SPICTRL,x
		andcc		#^(IntMasks+Carry)	Renable Interrupts and clear carry
		rts

* Error handlers
EWRITE
		ldd		#$0000+E$Write	A=Enable SPI Interface, but not CS
* B=Write Error
		bra		RETERR
EWP
		ldd		#$0000+E$WP		A=Enable SPI Interface, but not CS
* B=Write Protect Error

RETERR
		sta		SPICTRL,x	Set the hardware
		andcc		#^IntMasks	Enable interrupts
		coma				Set Carry
		rts
NOTRDY
		ldd		#$0000+E$NotRdy	Turn off controller and turn on IRQ and flag not ready error
		bra		RETERR
* 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
		orcc		#IntMasks	disable interrupts
		lda		$FFD9		Speed up
		ldx		V.PORT-UOFFSET,u	load x with the hw address for the IRQ routine
		lda		SPISTAT,x
		lsra
		bcc		NOTRDY	If there is no card, nothing to do
* Enable SPI
		ldd		#SPI_EN*256+$10	Enable SPI Interface, but not CS
* 16*8 cycles is >= 74
		sta		SPICTRL,x

*send at least 74 clock cycles with no SS, 12*8 = 96
lpff		lda		SPIDAT,x	Send FF
		decb				2 cycles, need 4
*		nop				2 cycles
		bne		lpff		3 Cycles
*Initialize card 0
CRD0		lda		#SPI_EN+SLOT_SEL_0	Enable SPI and lower CS
		sta		SPICTRL,x

* sdinit - initializes SD Cards, if installed
sdinit
* Send CMD0
		lda		SPIDAT,x	Send 1 FF
		nop				????????? enough
		leay		CMD0,pcr	Might need more cycles ???????
		lda		SPIDAT,x	Send 1 more FF
		lbsr		cmdsend	Also does a GETR1
		bcs		NOTRDY
		anda		#$7E		Idle is ok
		bne		NOTRDY	but nothing else

* Send CMD8
		lda		SPIDAT,x	Send 1 FF
		nop				?????? enough
		leay		CMD8,pcr	Might need more cycles ??????
		lda		SPIDAT,x	Sens 1 more FF
		lbsr		cmdsend	Also does an GETR1
		bcs		SDV1
		anda		#$7E
		bne		SDV1
		lda		SPIDAT,x	Byte 1 of R3/R7, through it away
		nop
		nop		might need more ????????
		nop
		lda		SPIDAT,x	Byte 2 of R3/R7, throught it away
		nop
		nop
		nop
		lda		SPIDAT,x	Byte 3 of R3/R7, should be 1
		cmpa		#$01		2 cycles
		bne		NOTRDY	2 cycles
		nop
		lda		SPIDAT,x	Byte 4 of R3/R7, should be $AA
		cmpa		#$AA		2 cycles
		bne		NOTRDY	2 cycles
		nop

* Send ACMD41 by first CMD55
loop41V2	lda		SPIDAT,x	Send 1 FF
		nop
		leay		CMD55,pcr	might need more ??????
		lda		SPIDAT,x	Send 1 FF
		lbsr		cmdsend	Also does an GETR1
		bcs		NOTRDY
		anda		#$7E		Idle is ok
		bne		NOTRDY	but nothing else

* Then send ACMD41
		lda		SPIDAT,x
		nop
		leay		ACMD41V2,pcr
		lda		SPIDAT,x
		lbsr		cmdsend
		bcs		NOTRDY	No response
		beq		Send58	If 0 then CMD58
		cmpa		#$01		if 1 then try again
		beq		loop41V2
		lbra		NOTRDY

* Send CMD58 to V2 card
Send58	lda		SPIDAT,x
		nop				?????? ENOUGH
		leay		CMD58,pcr	Read OCR
		lda		SPIDAT,x
		lbsr		cmdsend
		lbcs		NOTRDY
		lda		SPIDAT,x	Byte 1 of OCR
		anda		#$40		CCS bit 1= sector 0= byte
		sta		SDVersion,u
		lda		SPIDAT,x	Byte 2 of R3/R7, through it away
		nop
		nop
		lda		SPIDAT,x	Byte 3 of R3/R7, through it away
		nop
		nop
		lda		SPIDAT,x	Byte 4 of R3/R7, through it away
		lda		SDVersion,u	0 = byte addressable, !0 = block addressable
		bne		FININIT
		bra		Send16

* Send ACMD41 by first CMD55
SDV1

loop41V1	lda		SPIDAT,x	Get extra bytes in case we got a bad R7 previously
		nop
		lda		SPIDAT,x
*		nop
		clr		SDVersion,u		Byte addressable
		lda		SPIDAT,x
*		nop
		leay		CMD55,pcr
		lda		SPIDAT,x
		lbsr		cmdsend
		lbcs		NOTRDY
		anda		#$7E		Idle is ok
		lbne		NOTRDY	but nothing else

* Then send ACMD41
		lda		SPIDAT,x
*		nop
		leay		ACMD41V1,pcr
		lda		SPIDAT,x
		lbsr		cmdsend
		lbcs		NOTRDY
		beq		Send16	If 0 then CMD16
		cmpa		#$01		if 1 then try again
		beq		loop41V1
		lbra		NOTRDY
* Send CMD16
Send16	lda		SPIDAT,x
*		nop
		leay		CMD16,pcr
		lda		SPIDAT,x
		lbsr		cmdsend
		lbne		NOTRDY	but nothing else
* Finish INIT
FININIT
		lda		SPIDAT,x	Send last FF
*		nop
*		lda		SDVersion,u
*		sta		$ff66
*		lda		#SPI_EN+SPI_IRQ_EN	Turn on SPI and Interrupt and turn off CS
		lda		#SPI_EN	Turn on SPI and turn off CS
		sta		SPICTRL,x

*Finished with initialization
*Use the stat routine to return

* ll_getstat - Low level GetStat routine
* 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_getstat
ll_setstat
		andcc		#^Carry
		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
		andcc		#^Carry
		rts

		EMOD      
eom		EQU		*
		END