You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
			
		
		
		
		
			
		
			
				
	
	
		
			650 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
			
		
		
	
	
			650 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
| /*
 | |
|  * Slightly modified for M128 using
 | |
|  * maxx_ir 28.05.2015
 | |
|  * 
 | |
|  * PS. SPI Speed (4/8Mhz on F_CPU=16Mhz) set-up at: <power_on()>
 | |
|  */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* MMCv3/SDv1/SDv2 (in SPI mode) control module                          */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /*
 | |
| /  Copyright (C) 2014, ChaN, all right reserved.
 | |
| /
 | |
| / * This software is a free software and there is NO WARRANTY.
 | |
| / * No restriction on use. You can use, modify and redistribute it for
 | |
| /   personal, non-profit or commercial products UNDER YOUR RESPONSIBILITY.
 | |
| / * Redistributions of source code must retain the above copyright notice.
 | |
| /
 | |
| /-------------------------------------------------------------------------*/
 | |
| 
 | |
| #include <avr/io.h>
 | |
| #include "diskio.h"
 | |
| #include "globals.h"
 | |
| 
 | |
| /* Port controls  (Platform dependent) */
 | |
| 
 | |
| //!! M128
 | |
| //#define SCK            1  /* - Output: SPI Serial Clock (SCLK) - ATMEGA128 PORTB, PIN1 */
 | |
| //#define MOSI           2  /* - Output: SPI Master out - slave in (MOSI) - ATMEGA128 PORTB, PIN2 */
 | |
| //#define MISO           3  /* - Input:  SPI Master in - slave out (MISO) - ATMEGA128 PORTB, PIN3 */
 | |
| //#define CSN            0  /*SPI - SS*/
 | |
| //#define SD_CS          6       /* PB.6 Output as CS*/
 | |
| 
 | |
| //!! ATMEGA644/1284
 | |
| #define SCK            7  /* - Output: SPI Serial Clock (SCLK) - ATMEGA644/1284 PORTB, PIN7 */
 | |
| #define MOSI           5  /* - Output: SPI Master out - slave in (MOSI) -  ATMEGA644/1284 PORTB, PIN5 */
 | |
| #define MISO           6  /* - Input:  SPI Master in - slave out (MISO) -  ATMEGA644/1284 PORTB, PIN6 */
 | |
| #define CSN            4  /*SPI - SS*/
 | |
| //#define SD_CS       2       /* PB.2 Output as CS*/
 | |
| #define SD_CS       0       /* PB.0 Output as CS*/
 | |
| 
 | |
| 
 | |
| #define CS_LOW()	PORTB &= ~(1<<SD_CS)			/* CS=low */
 | |
| #define	CS_HIGH()	PORTB |= (1<<SD_CS)			/* CS=high */
 | |
| #define MMC_CD		(1)	/* Card detected.   yes:true, no:false, default:true */
 | |
| #define MMC_WP		(0)		/* Write protected. yes:true, no:false, default:false */
 | |
| #define	FCLK_SLOW()	SPCR = ((1<<SPE) | (1<<MSTR) | (1<<SPR1)) /*SPCR = 0x52*/		/* Set slow clock (F_CPU / 64) */
 | |
| #define	FCLK_FAST()	SPCR = ((1<<SPE) | (1<<MSTR)) /*SPCR = 0x50*/		/* Set fast clock (F_CPU / 4) */
 | |
| 
 | |
| 
 | |
| /*--------------------------------------------------------------------------
 | |
| 
 | |
|    Module Private Functions
 | |
| 
 | |
| ---------------------------------------------------------------------------*/
 | |
| 
 | |
| /* Definitions for MMC/SDC command */
 | |
| #define CMD0	(0)			/* GO_IDLE_STATE */
 | |
| #define CMD1	(1)			/* SEND_OP_COND (MMC) */
 | |
| #define	ACMD41	(0x80+41)	/* SEND_OP_COND (SDC) */
 | |
| #define CMD8	(8)			/* SEND_IF_COND */
 | |
| #define CMD9	(9)			/* SEND_CSD */
 | |
| #define CMD10	(10)		/* SEND_CID */
 | |
| #define CMD12	(12)		/* STOP_TRANSMISSION */
 | |
| #define ACMD13	(0x80+13)	/* SD_STATUS (SDC) */
 | |
| #define CMD16	(16)		/* SET_BLOCKLEN */
 | |
| #define CMD17	(17)		/* READ_SINGLE_BLOCK */
 | |
| #define CMD18	(18)		/* READ_MULTIPLE_BLOCK */
 | |
| #define CMD23	(23)		/* SET_BLOCK_COUNT (MMC) */
 | |
| #define	ACMD23	(0x80+23)	/* SET_WR_BLK_ERASE_COUNT (SDC) */
 | |
| #define CMD24	(24)		/* WRITE_BLOCK */
 | |
| #define CMD25	(25)		/* WRITE_MULTIPLE_BLOCK */
 | |
| #define CMD32	(32)		/* ERASE_ER_BLK_START */
 | |
| #define CMD33	(33)		/* ERASE_ER_BLK_END */
 | |
| #define CMD38	(38)		/* ERASE */
 | |
| #define CMD55	(55)		/* APP_CMD */
 | |
| #define CMD58	(58)		/* READ_OCR */
 | |
| 
 | |
| 
 | |
| static volatile
 | |
| DSTATUS Stat = STA_NOINIT;	/* Disk status */
 | |
| 
 | |
| static volatile
 | |
| BYTE Timer1, Timer2;	/* 100Hz decrement timer */
 | |
| 
 | |
| static
 | |
| BYTE CardType;			/* Card type flags */
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Power Control  (Platform dependent)                                   */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* When the target system does not support socket power control, there   */
 | |
| /* is nothing to do in these functions and chk_power always returns 1.   */
 | |
| 
 | |
| static
 | |
| void power_on (void)
 | |
| {
 | |
| 	
 | |
| 	// Chang init
 | |
| 	//~ PORTB |= 0b01000101;	/* Configure SCK/MOSI/CS/SS as output */
 | |
| 	//~ DDRB  |= 0b01000111;
 | |
| 	//~ 
 | |
| 	//~ SPCR = 0x52;			/* Enable SPI function in mode 0 */
 | |
| 	//~ //SPSR = 0x01;			/* SPI 2x mode 8MHZ */
 | |
| 	//~ SPSR = 0x00;			/* SPI 2x mode off 4MHZ */
 | |
| 	
 | |
| 	// My init (Contiki-style)
 | |
| 	DDRB	|= _BV(SD_CS); // CS to OUT && Disable
 | |
| 	CS_HIGH();
 | |
| 
 | |
| 	/* Initalize ports for communication with SPI units. */
 | |
| 	/* CSN=SS and must be output when master! */
 | |
| 	DDRB  |= _BV(MOSI) | _BV(SCK) | _BV(CSN);
 | |
| 	PORTB |= _BV(MOSI) | _BV(SCK);
 | |
| 
 | |
| 
 | |
| #if defined(SPI_8_MHZ)
 | |
| 	/* Enables SPI, selects "master", clock rate FCK / 2, and SPI mode 0 */
 | |
| 	// SPI 8Mhz
 | |
| 	SPCR = _BV(SPE) | _BV(MSTR);
 | |
| 	SPSR = _BV(SPI2X); //FCK / 2 - 8Mhz
 | |
| #elif defined (SPI_4_MHZ)
 | |
| 	/* Enables SPI, selects "master", clock rate FCK / 4, and SPI mode 0 */
 | |
| 	// SPI 4Mhz
 | |
| 	SPCR = _BV(SPE) | _BV(MSTR);
 | |
| 	SPSR = 0x0; //FCK / 4 - 4Mhz
 | |
| #else
 | |
| 	/* Enables SPI, selects "master", clock rate FCK / 4, and SPI mode 0 */
 | |
| 	// SPI 4Mhz
 | |
| 	SPCR = _BV(SPE) | _BV(MSTR);
 | |
| 	SPSR = 0x0; //FCK / 4 - 4Mhz
 | |
| #endif
 | |
| 
 | |
| }
 | |
| 
 | |
| static
 | |
| void power_off (void)
 | |
| {
 | |
| 	
 | |
| 	//SPCR = 0;
 | |
| 					/* Disable SPI function */
 | |
| 
 | |
| 	//DDRB  &= ~0b00110111;	
 | |
| 	/* Set SCK/MOSI/CS as hi-z, INS#/WP as pull-up */
 | |
| 	//PORTB &= ~0b00000111;
 | |
| 	//PORTB |=  0b00110000;
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Transmit/Receive data from/to MMC via SPI  (Platform dependent)       */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| /* Exchange a byte */
 | |
| static
 | |
| BYTE xchg_spi (		/* Returns received data */
 | |
| 	BYTE dat		/* Data to be sent */
 | |
| )
 | |
| {
 | |
| 	SPDR = dat;
 | |
| 	loop_until_bit_is_set(SPSR, SPIF);
 | |
| 	return SPDR;
 | |
| }
 | |
| 
 | |
| /* Send a data block fast */
 | |
| static
 | |
| void xmit_spi_multi (
 | |
| 	const BYTE *p,	/* Data block to be sent */
 | |
| 	UINT cnt		/* Size of data block (must be multiple of 2) */
 | |
| )
 | |
| {
 | |
| 	do {
 | |
| 		SPDR = *p++; loop_until_bit_is_set(SPSR,SPIF);
 | |
| 		SPDR = *p++; loop_until_bit_is_set(SPSR,SPIF);
 | |
| 	} while (cnt -= 2);
 | |
| }
 | |
| 
 | |
| /* Receive a data block fast */
 | |
| static
 | |
| void rcvr_spi_multi (
 | |
| 	BYTE *p,	/* Data buffer */
 | |
| 	UINT cnt	/* Size of data block (must be multiple of 2) */
 | |
| )
 | |
| {
 | |
| 	do {
 | |
| 		SPDR = 0xFF; loop_until_bit_is_set(SPSR,SPIF); *p++ = SPDR;
 | |
| 		SPDR = 0xFF; loop_until_bit_is_set(SPSR,SPIF); *p++ = SPDR;
 | |
| 	} while (cnt -= 2);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Wait for card ready                                                   */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| static
 | |
| int wait_ready (	/* 1:Ready, 0:Timeout */
 | |
| 	UINT wt			/* Timeout [ms] */
 | |
| )
 | |
| {
 | |
| 	BYTE d;
 | |
| 
 | |
| 
 | |
| 	Timer2 = wt / 10;
 | |
| 	do
 | |
| 		d = xchg_spi(0xFF);
 | |
| 	while (d != 0xFF && Timer2);
 | |
| 
 | |
| 	return (d == 0xFF) ? 1 : 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Deselect the card and release SPI bus                                 */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| static
 | |
| void deselect (void)
 | |
| {
 | |
| 	CS_HIGH();		/* Set CS# high */
 | |
| 	xchg_spi(0xFF);	/* Dummy clock (force DO hi-z for multiple slave SPI) */
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Select the card and wait for ready                                    */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| static
 | |
| int select (void)	/* 1:Successful, 0:Timeout */
 | |
| {
 | |
| 	CS_LOW();		/* Set CS# low */
 | |
| 	xchg_spi(0xFF);	/* Dummy clock (force DO enabled) */
 | |
| 	if (wait_ready(500)) return 1;	/* Wait for card ready */
 | |
| 
 | |
| 	deselect();
 | |
| 	return 0;	/* Timeout */
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Receive a data packet from MMC                                        */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| static
 | |
| int rcvr_datablock (
 | |
| 	BYTE *buff,			/* Data buffer to store received data */
 | |
| 	UINT btr			/* Byte count (must be multiple of 4) */
 | |
| )
 | |
| {
 | |
| 	BYTE token;
 | |
| 
 | |
| 
 | |
| 	Timer1 = 20;
 | |
| 	do {							/* Wait for data packet in timeout of 200ms */
 | |
| 		token = xchg_spi(0xFF);
 | |
| 	} while ((token == 0xFF) && Timer1);
 | |
| 	if (token != 0xFE) return 0;	/* If not valid data token, retutn with error */
 | |
| 
 | |
| 	rcvr_spi_multi(buff, btr);		/* Receive the data block into buffer */
 | |
| 	xchg_spi(0xFF);					/* Discard CRC */
 | |
| 	xchg_spi(0xFF);
 | |
| 
 | |
| 	return 1;						/* Return with success */
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Send a data packet to MMC                                             */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| #if	_USE_WRITE
 | |
| static
 | |
| int xmit_datablock (
 | |
| 	const BYTE *buff,	/* 512 byte data block to be transmitted */
 | |
| 	BYTE token			/* Data/Stop token */
 | |
| )
 | |
| {
 | |
| 	BYTE resp;
 | |
| 
 | |
| 
 | |
| 	if (!wait_ready(500)) return 0;
 | |
| 
 | |
| 	xchg_spi(token);					/* Xmit data token */
 | |
| 	if (token != 0xFD) {	/* Is data token */
 | |
| 		xmit_spi_multi(buff, 512);		/* Xmit the data block to the MMC */
 | |
| 		xchg_spi(0xFF);					/* CRC (Dummy) */
 | |
| 		xchg_spi(0xFF);
 | |
| 		resp = xchg_spi(0xFF);			/* Reveive data response */
 | |
| 		if ((resp & 0x1F) != 0x05)		/* If not accepted, return with error */
 | |
| 			return 0;
 | |
| 	}
 | |
| 
 | |
| 	return 1;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Send a command packet to MMC                                          */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| static
 | |
| BYTE send_cmd (		/* Returns R1 resp (bit7==1:Send failed) */
 | |
| 	BYTE cmd,		/* Command index */
 | |
| 	DWORD arg		/* Argument */
 | |
| )
 | |
| {
 | |
| 	BYTE n, res;
 | |
| 
 | |
| 
 | |
| 	if (cmd & 0x80) {	/* ACMD<n> is the command sequense of CMD55-CMD<n> */
 | |
| 		cmd &= 0x7F;
 | |
| 		res = send_cmd(CMD55, 0);
 | |
| 		if (res > 1) return res;
 | |
| 	}
 | |
| 
 | |
| 	/* Select the card and wait for ready except to stop multiple block read */
 | |
| 	if (cmd != CMD12) {
 | |
| 		deselect();
 | |
| 		if (!select()) return 0xFF;
 | |
| 	}
 | |
| 
 | |
| 	/* Send command packet */
 | |
| 	xchg_spi(0x40 | cmd);				/* Start + Command index */
 | |
| 	xchg_spi((BYTE)(arg >> 24));		/* Argument[31..24] */
 | |
| 	xchg_spi((BYTE)(arg >> 16));		/* Argument[23..16] */
 | |
| 	xchg_spi((BYTE)(arg >> 8));			/* Argument[15..8] */
 | |
| 	xchg_spi((BYTE)arg);				/* Argument[7..0] */
 | |
| 	n = 0x01;							/* Dummy CRC + Stop */
 | |
| 	if (cmd == CMD0) n = 0x95;			/* Valid CRC for CMD0(0) + Stop */
 | |
| 	if (cmd == CMD8) n = 0x87;			/* Valid CRC for CMD8(0x1AA) Stop */
 | |
| 	xchg_spi(n);
 | |
| 
 | |
| 	/* Receive command response */
 | |
| 	if (cmd == CMD12) xchg_spi(0xFF);		/* Skip a stuff byte when stop reading */
 | |
| 	n = 10;								/* Wait for a valid response in timeout of 10 attempts */
 | |
| 	do
 | |
| 		res = xchg_spi(0xFF);
 | |
| 	while ((res & 0x80) && --n);
 | |
| 
 | |
| 	return res;			/* Return with the response value */
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*--------------------------------------------------------------------------
 | |
| 
 | |
|    Public Functions
 | |
| 
 | |
| ---------------------------------------------------------------------------*/
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Initialize Disk Drive                                                 */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| DSTATUS disk_initialize (
 | |
| 	BYTE pdrv		/* Physical drive nmuber (0) */
 | |
| )
 | |
| {
 | |
| 	BYTE n, cmd, ty, ocr[4];
 | |
| 
 | |
| 
 | |
| 	if (pdrv) return STA_NOINIT;		/* Supports only single drive */
 | |
| 	power_off();						/* Turn off the socket power to reset the card */
 | |
| 	if (Stat & STA_NODISK) return Stat;	/* No card in the socket */
 | |
| 	power_on();							/* Turn on the socket power */
 | |
| 	FCLK_SLOW();
 | |
| 	for (n = 10; n; n--) xchg_spi(0xFF);	/* 80 dummy clocks */
 | |
| 
 | |
| 	ty = 0;
 | |
| 	if (send_cmd(CMD0, 0) == 1) {			/* Enter Idle state */
 | |
| 		Timer1 = 100;						/* Initialization timeout of 1000 msec */
 | |
| 		if (send_cmd(CMD8, 0x1AA) == 1) {	/* SDv2? */
 | |
| 			for (n = 0; n < 4; n++) ocr[n] = xchg_spi(0xFF);		/* Get trailing return value of R7 resp */
 | |
| 			if (ocr[2] == 0x01 && ocr[3] == 0xAA) {				/* The card can work at vdd range of 2.7-3.6V */
 | |
| 				while (Timer1 && send_cmd(ACMD41, 1UL << 30));	/* Wait for leaving idle state (ACMD41 with HCS bit) */
 | |
| 				if (Timer1 && send_cmd(CMD58, 0) == 0) {		/* Check CCS bit in the OCR */
 | |
| 					for (n = 0; n < 4; n++) ocr[n] = xchg_spi(0xFF);
 | |
| 					ty = (ocr[0] & 0x40) ? CT_SD2 | CT_BLOCK : CT_SD2;	/* SDv2 */
 | |
| 				}
 | |
| 			}
 | |
| 		} else {							/* SDv1 or MMCv3 */
 | |
| 			if (send_cmd(ACMD41, 0) <= 1) 	{
 | |
| 				ty = CT_SD1; cmd = ACMD41;	/* SDv1 */
 | |
| 			} else {
 | |
| 				ty = CT_MMC; cmd = CMD1;	/* MMCv3 */
 | |
| 			}
 | |
| 			while (Timer1 && send_cmd(cmd, 0));			/* Wait for leaving idle state */
 | |
| 			if (!Timer1 || send_cmd(CMD16, 512) != 0)	/* Set R/W block length to 512 */
 | |
| 				ty = 0;
 | |
| 		}
 | |
| 	}
 | |
| 	CardType = ty;
 | |
| 	deselect();
 | |
| 
 | |
| 	if (ty) {			/* Initialization succeded */
 | |
| 		Stat &= ~STA_NOINIT;		/* Clear STA_NOINIT */
 | |
| 		FCLK_FAST();
 | |
| 	} else {			/* Initialization failed */
 | |
| 		power_off();
 | |
| 	}
 | |
| 
 | |
| 	return Stat;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Get Disk Status                                                       */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| DSTATUS disk_status (
 | |
| 	BYTE pdrv		/* Physical drive nmuber (0) */
 | |
| )
 | |
| {
 | |
| 	if (pdrv) return STA_NOINIT;	/* Supports only single drive */
 | |
| 	return Stat;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Read Sector(s)                                                        */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| DRESULT disk_read (
 | |
| 	BYTE pdrv,			/* Physical drive nmuber (0) */
 | |
| 	BYTE *buff,			/* Pointer to the data buffer to store read data */
 | |
| 	DWORD sector,		/* Start sector number (LBA) */
 | |
| 	UINT count			/* Sector count (1..128) */
 | |
| )
 | |
| {
 | |
| 	BYTE cmd;
 | |
| 
 | |
| 
 | |
| 	if (pdrv || !count) return RES_PARERR;
 | |
| 	if (Stat & STA_NOINIT) return RES_NOTRDY;
 | |
| 
 | |
| 	if (!(CardType & CT_BLOCK)) sector *= 512;	/* Convert to byte address if needed */
 | |
| 
 | |
| 	cmd = count > 1 ? CMD18 : CMD17;			/*  READ_MULTIPLE_BLOCK : READ_SINGLE_BLOCK */
 | |
| 	if (send_cmd(cmd, sector) == 0) {
 | |
| 		do {
 | |
| 			if (!rcvr_datablock(buff, 512)) break;
 | |
| 			buff += 512;
 | |
| 		} while (--count);
 | |
| 		if (cmd == CMD18) send_cmd(CMD12, 0);	/* STOP_TRANSMISSION */
 | |
| 	}
 | |
| 	deselect();
 | |
| 
 | |
| 	return count ? RES_ERROR : RES_OK;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Write Sector(s)                                                       */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| #if _USE_WRITE
 | |
| DRESULT disk_write (
 | |
| 	BYTE pdrv,			/* Physical drive nmuber (0) */
 | |
| 	const BYTE *buff,	/* Pointer to the data to be written */
 | |
| 	DWORD sector,		/* Start sector number (LBA) */
 | |
| 	UINT count			/* Sector count (1..128) */
 | |
| )
 | |
| {
 | |
| 	if (pdrv || !count) return RES_PARERR;
 | |
| 	if (Stat & STA_NOINIT) return RES_NOTRDY;
 | |
| 	if (Stat & STA_PROTECT) return RES_WRPRT;
 | |
| 
 | |
| 	if (!(CardType & CT_BLOCK)) sector *= 512;	/* Convert to byte address if needed */
 | |
| 
 | |
| 	if (count == 1) {	/* Single block write */
 | |
| 		if ((send_cmd(CMD24, sector) == 0)	/* WRITE_BLOCK */
 | |
| 			&& xmit_datablock(buff, 0xFE))
 | |
| 			count = 0;
 | |
| 	}
 | |
| 	else {				/* Multiple block write */
 | |
| 		if (CardType & CT_SDC) send_cmd(ACMD23, count);
 | |
| 		if (send_cmd(CMD25, sector) == 0) {	/* WRITE_MULTIPLE_BLOCK */
 | |
| 			do {
 | |
| 				if (!xmit_datablock(buff, 0xFC)) break;
 | |
| 				buff += 512;
 | |
| 			} while (--count);
 | |
| 			if (!xmit_datablock(0, 0xFD))	/* STOP_TRAN token */
 | |
| 				count = 1;
 | |
| 		}
 | |
| 	}
 | |
| 	deselect();
 | |
| 
 | |
| 	return count ? RES_ERROR : RES_OK;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Miscellaneous Functions                                               */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| 
 | |
| #if _USE_IOCTL
 | |
| DRESULT disk_ioctl (
 | |
| 	BYTE pdrv,		/* Physical drive nmuber (0) */
 | |
| 	BYTE cmd,		/* Control code */
 | |
| 	void *buff		/* Buffer to send/receive control data */
 | |
| )
 | |
| {
 | |
| 	DRESULT res;
 | |
| 	BYTE n, csd[16], *ptr = buff;
 | |
| 	DWORD csize;
 | |
| 
 | |
| 
 | |
| 	if (pdrv) return RES_PARERR;
 | |
| 
 | |
| 	res = RES_ERROR;
 | |
| 
 | |
| 	if (Stat & STA_NOINIT) return RES_NOTRDY;
 | |
| 
 | |
| 	switch (cmd) {
 | |
| 	case CTRL_SYNC :		/* Make sure that no pending write process. Do not remove this or written sector might not left updated. */
 | |
| 		if (select()) res = RES_OK;
 | |
| 		break;
 | |
| 
 | |
| 	case GET_SECTOR_COUNT :	/* Get number of sectors on the disk (DWORD) */
 | |
| 		if ((send_cmd(CMD9, 0) == 0) && rcvr_datablock(csd, 16)) {
 | |
| 			if ((csd[0] >> 6) == 1) {	/* SDC ver 2.00 */
 | |
| 				csize = csd[9] + ((WORD)csd[8] << 8) + ((DWORD)(csd[7] & 63) << 16) + 1;
 | |
| 				*(DWORD*)buff = csize << 10;
 | |
| 			} else {					/* SDC ver 1.XX or MMC*/
 | |
| 				n = (csd[5] & 15) + ((csd[10] & 128) >> 7) + ((csd[9] & 3) << 1) + 2;
 | |
| 				csize = (csd[8] >> 6) + ((WORD)csd[7] << 2) + ((WORD)(csd[6] & 3) << 10) + 1;
 | |
| 				*(DWORD*)buff = csize << (n - 9);
 | |
| 			}
 | |
| 			res = RES_OK;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case GET_BLOCK_SIZE :	/* Get erase block size in unit of sector (DWORD) */
 | |
| 		if (CardType & CT_SD2) {	/* SDv2? */
 | |
| 			if (send_cmd(ACMD13, 0) == 0) {	/* Read SD status */
 | |
| 				xchg_spi(0xFF);
 | |
| 				if (rcvr_datablock(csd, 16)) {				/* Read partial block */
 | |
| 					for (n = 64 - 16; n; n--) xchg_spi(0xFF);	/* Purge trailing data */
 | |
| 					*(DWORD*)buff = 16UL << (csd[10] >> 4);
 | |
| 					res = RES_OK;
 | |
| 				}
 | |
| 			}
 | |
| 		} else {					/* SDv1 or MMCv3 */
 | |
| 			if ((send_cmd(CMD9, 0) == 0) && rcvr_datablock(csd, 16)) {	/* Read CSD */
 | |
| 				if (CardType & CT_SD1) {	/* SDv1 */
 | |
| 					*(DWORD*)buff = (((csd[10] & 63) << 1) + ((WORD)(csd[11] & 128) >> 7) + 1) << ((csd[13] >> 6) - 1);
 | |
| 				} else {					/* MMCv3 */
 | |
| 					*(DWORD*)buff = ((WORD)((csd[10] & 124) >> 2) + 1) * (((csd[11] & 3) << 3) + ((csd[11] & 224) >> 5) + 1);
 | |
| 				}
 | |
| 				res = RES_OK;
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	/* Following commands are never used by FatFs module */
 | |
| 
 | |
| 	case MMC_GET_TYPE :		/* Get card type flags (1 byte) */
 | |
| 		*ptr = CardType;
 | |
| 		res = RES_OK;
 | |
| 		break;
 | |
| 
 | |
| 	case MMC_GET_CSD :		/* Receive CSD as a data block (16 bytes) */
 | |
| 		if (send_cmd(CMD9, 0) == 0		/* READ_CSD */
 | |
| 			&& rcvr_datablock(ptr, 16))
 | |
| 			res = RES_OK;
 | |
| 		break;
 | |
| 
 | |
| 	case MMC_GET_CID :		/* Receive CID as a data block (16 bytes) */
 | |
| 		if (send_cmd(CMD10, 0) == 0		/* READ_CID */
 | |
| 			&& rcvr_datablock(ptr, 16))
 | |
| 			res = RES_OK;
 | |
| 		break;
 | |
| 
 | |
| 	case MMC_GET_OCR :		/* Receive OCR as an R3 resp (4 bytes) */
 | |
| 		if (send_cmd(CMD58, 0) == 0) {	/* READ_OCR */
 | |
| 			for (n = 4; n; n--) *ptr++ = xchg_spi(0xFF);
 | |
| 			res = RES_OK;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case MMC_GET_SDSTAT :	/* Receive SD statsu as a data block (64 bytes) */
 | |
| 		if (send_cmd(ACMD13, 0) == 0) {	/* SD_STATUS */
 | |
| 			xchg_spi(0xFF);
 | |
| 			if (rcvr_datablock(ptr, 64))
 | |
| 				res = RES_OK;
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	case CTRL_POWER_OFF :	/* Power off */
 | |
| 		power_off();
 | |
| 		Stat |= STA_NOINIT;
 | |
| 		res = RES_OK;
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		res = RES_PARERR;
 | |
| 	}
 | |
| 
 | |
| 	deselect();
 | |
| 
 | |
| 	return res;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* Device Timer Interrupt Procedure                                      */
 | |
| /*-----------------------------------------------------------------------*/
 | |
| /* This function must be called in period of 10ms                        */
 | |
| 
 | |
| void disk_timerproc (void)
 | |
| {
 | |
| 	BYTE n, s;
 | |
| 
 | |
| 
 | |
| 	n = Timer1;				/* 100Hz decrement timer */
 | |
| 	if (n) Timer1 = --n;
 | |
| 	n = Timer2;
 | |
| 	if (n) Timer2 = --n;
 | |
| 
 | |
| 	s = Stat;
 | |
| 
 | |
| 	if (MMC_WP)				/* Write protected */
 | |
| 		s |= STA_PROTECT;
 | |
| 	else					/* Write enabled */
 | |
| 		s &= ~STA_PROTECT;
 | |
| 
 | |
| 	if (MMC_CD)				/* Card inserted */
 | |
| 		s &= ~STA_NODISK;
 | |
| 	else					/* Socket empty */
 | |
| 		s |= (STA_NODISK | STA_NOINIT);
 | |
| 
 | |
| 	Stat = s;				/* Update MMC status */
 | |
| }
 |