單片機對SD卡讀寫系列(二)
- //-----------------------------------------------------------------------------------------------
向SD卡中寫入命令,并返回回應(yīng)的第二個字節(jié) - //-----------------------------------------------------------------------------------------------
- unsigned
char Write_Command_SD(unsigned char *CMD) - {
unsigned char tmp; unsigned char retry=0; unsigned char i; //禁止SD卡片選 SPI_CS=1; //發(fā)送8個時鐘信號 Write_Byte_SD(0xFF); //使能SD卡片選 SPI_CS=0; //向SD卡發(fā)送6字節(jié)命令 for (i=0;i<0x06;i++) { Write_Byte_SD(*CMD++); } //獲得16位的回應(yīng) Read_Byte_SD(); //read the first byte,ignore it. do { //讀取后8位 tmp = Read_Byte_SD(); retry++; } while((tmp==0xff)&&(retry<100)); return(tmp); - }
2)
SD卡的初始化是非常重要的,只有進行了正確的初始化,才能進行后面的各項操作。在初始化過程中,SPI的時鐘不能太快,否則會造初始化失敗。在初始化成功后,應(yīng)盡量提高SPI的速率。在剛開始要先發(fā)送至少74個時鐘信號,這是必須的。在很多讀者的實驗中,很多是因為疏忽了這一點,而使初始化不成功。隨后就是寫入兩個命令CMD0與CMD1,使SD卡進入SPI模式
- //--------------------------------------------------------------------------
初始化SD卡到SPI模式 - //--------------------------------------------------------------------------
- unsigned
char SD_Init() - {
unsigned char retry,temp; unsigned char i; unsigned char CMD[] = {0x40,0x00,0x00,0x00,0x00,0x95}; SD_Port_Init(); //初始化驅(qū)動端口 Init_Flag=1; //將初始化標(biāo)志置1 for (i=0;i<0x0f;i++) { Write_Byte_SD(0xff); //發(fā)送至少74個時鐘信號 } //向SD卡發(fā)送CMD0 retry=0; do { //為了能夠成功寫入CMD0,在這里寫200次 temp=Write_Command_SD(CMD); retry++; if(retry==200) { //超過200次 return(INIT_CMD0_ERROR);//CMD0 Error! } } while(temp!=1); //回應(yīng)01h,停止寫入 //發(fā)送CMD1到SD卡 CMD[0] = 0x41; //CMD1 CMD[5] = 0xFF; retry=0; do { //為了能成功寫入CMD1,寫100次 temp=Write_Command_SD(CMD); retry++; if(retry==100) { //超過100次 return(INIT_CMD1_ERROR);//CMD1 Error! } } while(temp!=0);//回應(yīng)00h停止寫入 Init_Flag=0; //初始化完畢,初始化標(biāo)志清零 SPI_CS=1; //片選無效 return(0); //初始化成功 - }
3)
CID寄存器存儲了SD卡的標(biāo)識碼。每一個卡都有唯一的標(biāo)識碼。
CID寄存器長度為128位。它的寄存器結(jié)構(gòu)如下:
名稱 | 域 | 數(shù)據(jù)寬度 | CID劃分 |
生產(chǎn)標(biāo)識號 | MID | 8 | [127:120] |
OEM/應(yīng)用標(biāo)識 | OID | 16 | [119:104] |
產(chǎn)品名稱 | PNM | 40 | [103:64] |
產(chǎn)品版本 | PRV | 8 | [63:56] |
產(chǎn)品序列號 | PSN | 32 | [55:24] |
保留 | - | 4 | [23:20] |
生產(chǎn)日期 | MDT | 12 | [19:8] |
CRC7校驗合 | CRC | 7 | [7:1] |
未使用,始終為1 | - | 1 | [0:0] |
它的讀取時序如下:
- //------------------------------------------------------------------------------------
讀取SD卡的CID寄存器 16字節(jié) 成功返回0 - //-------------------------------------------------------------------------------------
- unsigned
char Read_CID_SD(unsigned char *Buffer) - {
//讀取CID寄存器的命令 unsigned char CMD[] = {0x4A,0x00,0x00,0x00,0x00,0xFF}; unsigned char temp; temp=SD_Read_Block(CMD,Buffer,16); //read 16 bytes return(temp); - }
4)讀取CSD
CSD(Card-Specific Data)寄存器提供了讀寫SD卡的一些信息。其中的一些單元可以由用戶重新編程。
- //-----------------------------------------------------------------------------------------
讀SD卡的CSD寄存器 共16字節(jié) 返回0說明讀取成功 - //-----------------------------------------------------------------------------------------
- unsigned
char Read_CSD_SD(unsigned char *Buffer) - {
//讀取CSD寄存器的命令 unsigned char CMD[] = {0x49,0x00,0x00,0x00,0x00,0xFF}; unsigned char temp; temp=SD_Read_Block(CMD,Buffer,16); //read 16 bytes return(temp); - }
4)
綜合上面對CID與CSD寄存器的讀取,可以知道很多關(guān)于SD卡的信息,以下程序可以獲取這些信息。如下:
- //-----------------------------------------------------------------------------------------------
- //返回
- //
SD卡的容量,單位為M - //
sector count and multiplier MB are in - u08
== C_SIZE / (2^(9-C_SIZE_MULT)) - //
SD卡的名稱 - //-----------------------------------------------------------------------------------------------
- void
SD_get_volume_info() - {
unsigned char i; unsigned char c_temp[5]; VOLUME_INFO_TYPE SD_volume_Info,*vinf; vinf=&SD_volume_Info; //Init the pointoer; - /讀取CSD寄存器
Read_CSD_SD(sectorBuffer.dat); - //獲取總扇區(qū)數(shù)
vinf->sector_count = sectorBuffer.dat[6] & 0x03; vinf->sector_count <<= 8; vinf->sector_count += sectorBuffer.dat[7]; vinf->sector_count <<= 2; vinf->sector_count += (sectorBuffer.dat[8] & 0xc0) >> 6; // 獲取multiplier vinf->sector_multiply = sectorBuffer.dat[9] & 0x03; vinf->sector_multiply <<= 1; vinf->sector_multiply += (sectorBuffer.dat[10] & 0x80) >> 7; - //獲取SD卡的容量
vinf->size_MB = vinf->sector_count >> (9-vinf->sector_multiply); // get the name of the card Read_CID_SD(sectorBuffer.dat); vinf->name[0] = sectorBuffer.dat[3]; vinf->name[1] = sectorBuffer.dat[4]; vinf->name[2] = sectorBuffer.dat[5]; vinf->name[3] = sectorBuffer.dat[6]; vinf->name[4] = sectorBuffer.dat[7]; vinf->name[5] = 0x00; //end flag - }
以上程序?qū)⑿畔⒀b載到一個結(jié)構(gòu)體中,這個結(jié)構(gòu)體的定義如下: - typedef
struct SD_VOLUME_INFO - {
//SD/SD Card info unsigned int size_MB; unsigned char sector_multiply; unsigned int sector_count; unsigned char name[6]; - }
VOLUME_INFO_TYPE;
5)
扇區(qū)讀是對SD卡驅(qū)動的目的之一。SD卡的每一個扇區(qū)中有512個字節(jié),一次扇區(qū)讀操作將把某一個扇區(qū)內(nèi)的512個字節(jié)全部讀出。過程很簡單,先寫入命令,在得到相應(yīng)的回應(yīng)后,開始數(shù)據(jù)讀取。
扇區(qū)讀的時序:
- unsigned
char SD_Read_Sector(unsigned long sector,unsigned char *buffer) - {
unsigned char retry; //命令16 unsigned char CMD[] = {0x51,0x00,0x00,0x00,0x00,0xFF}; unsigned char temp; //地址變換 由邏輯塊地址轉(zhuǎn)為字節(jié)地址 sector = sector << 9; //sector = sector * 512 CMD[1] = ((sector & 0xFF000000) >>24 ); CMD[2] = ((sector & 0x00FF0000) >>16 ); CMD[3] = ((sector & 0x0000FF00) >>8 ); //將命令16寫入SD卡 retry=0; do { //為了保證寫入命令 一共寫100次 temp=Write_Command_MMC(CMD); retry++; if(retry==100) { return(READ_BLOCK_ERROR); //block write Error! } } while(temp!=0); //Read Start Byte form MMC/SD-Card (FEh/Start Byte) //Now data is ready,you can read it out. while (Read_Byte_MMC() != 0xfe); readPos=0; SD_get_data(512,buffer) ; //512字節(jié)被讀出到buffer中 return 0; - }
- 其中SD_get_data函數(shù)如下:
- //----------------------------------------------------------------------------
獲取數(shù)據(jù)到buffer中 - //----------------------------------------------------------------------------
- void
SD_get_data(unsigned int Bytes,unsigned char *buffer) - {
unsigned int j; for (j=0;j<="" span="" style="word-wrap: break-word;"> *buffer++ = Read_Byte_SD(); - }
6)
扇區(qū)寫是SD卡驅(qū)動的另一目的。每次扇區(qū)寫操作將向SD卡的某個扇區(qū)中寫入512個字節(jié)。過程與扇區(qū)讀相似,只是數(shù)據(jù)的方向相反與寫入命令不同而已。
扇區(qū)寫的程序例程:
- //--------------------------------------------------------------------------------------------
寫512個字節(jié)到SD卡的某一個扇區(qū)中去 返回0說明寫入成功 - //--------------------------------------------------------------------------------------------
- unsigned
char SD_write_sector(unsigned long addr,unsigned char *Buffer) - {
unsigned char tmp,retry; unsigned int i; //命令24 unsigned char CMD[] = {0x58,0x00,0x00,0x00,0x00,0xFF}; addr = addr << 9; //addr = addr * 512 CMD[1] = ((addr & 0xFF000000) >>24 ); CMD[2] = ((addr & 0x00FF0000) >>16 ); CMD[3] = ((addr & 0x0000FF00) >>8 ); //寫命令24到SD卡中去 retry=0; do { //為了可靠寫入,寫100次 tmp=Write_Command_SD(CMD); retry++; if(retry==100) { return(tmp); //send commamd Error! } } while(tmp!=0); //在寫之前先產(chǎn)生100個時鐘信號 for (i=0;i<100;i++) { Read_Byte_SD(); } //寫入開始字節(jié) Write_Byte_MMC(0xFE); //現(xiàn)在可以寫入512個字節(jié) for (i=0;i<512;i++) { Write_Byte_MMC(*Buffer++); } //CRC-Byte Write_Byte_MMC(0xFF); //Dummy CRC Write_Byte_MMC(0xFF); //CRC Code tmp=Read_Byte_MMC(); // read response if((tmp & 0x1F)!=0x05) // 寫入的512個字節(jié)是未被接受 { SPI_CS=1; return(WRITE_BLOCK_ERROR); //Error! } //等到SD卡不忙為止 - //因為數(shù)據(jù)被接受后,SD卡在向儲存陣列中編程數(shù)據(jù)
while (Read_Byte_MMC()!=0xff){}; //禁止SD卡 SPI_CS=1; return(0);//寫入成功 - }
單片機采用STC89LE單片機(SD卡的初始化電壓為2.0V~3.6V,操作電壓為3.1V~3.5V,因此不能用5V單片機,或進行分壓處理),工作于22.1184M的時鐘下,由于所采用的單片機中沒硬件SPI,采用軟件模擬SPI,因此讀寫速率都較慢。如果要半SD卡應(yīng)用于音頻、視頻等要求高速場合,則需要選用有硬件SPI的控制器,或使用SD模式,有了 SPI模式的基礎(chǔ),SD模式應(yīng)該不是什么難事。
評論