問題描述:
TQ2440的官方裸跑程序中,對SD卡先進行讀操作,然后再寫,發(fā)現不能程序卡死。倘若對SD卡先寫后讀,程序可以正常運行,奇哉怪哉?
寫數據的關鍵代碼-->
while(i
調試與問題分析:
調試的時候發(fā)現,當不能在寫的時候,FIFO available detectfor Tx (TFDET)為0,也即是說是fifo滿了。此時,程序循環(huán)了16次(i=0x10)。循環(huán)一次寫入4個字節(jié),16次剛好是fifo的最大容量64字節(jié)。這證明了,寫入fifo中的數據,本應該發(fā)送給SD卡,騰空fifo以供用戶繼續(xù)寫,卻被擱置在fifo中,有進無出。就像下水道中轉站被堵,上游的污水就不能繼續(xù)排放一行的道理。
先寫后讀是可以正常工作的,我打印了執(zhí)行寫函數之后部分寄存器的值,如左圖所示。可以發(fā)現寫后的寄存器rSDIDCNT、rSDIDSTA都恢復到了初始值。右圖是執(zhí)行讀函數之后寄存器的值,可以發(fā)現執(zhí)行讀函數之后,rSDIDCNT、rSDIDSTA都沒有回到初始值,都仍然停留在讀函數執(zhí)行的狀態(tài)中。也就是說,讀函數沒有執(zhí)行徹底,SDMMC模塊沒有進入到空閑狀態(tài)。在沒有準備好的情況下,繼續(xù)進行寫操作,是不可能成功的。
修復
修復的方法主要是無論讀操作,還是寫操作,都確認SDIO總線空閑時,然后再才退出當前的函數。這樣可以保證在隨后的操作中,SDMMC模塊處于準備好的狀態(tài),而非遺留狀態(tài)。
讀函數
/**********************************************************************************
功 能:該函數用于從SD卡中讀出指定塊起始地址的單個數據塊
參 數:
U32 Addr 被讀塊的起始地址
U8* RxBuffer 用于接收讀出數據的緩沖區(qū)
返回值:
0 讀塊操作不成功
1 讀塊操作成功
舉 例:
在主調函數中定義一個數組作為接收緩沖區(qū),如U8 Rx_buffer[BlockSize];
然后開始調用Read_One_Block(addr,Rx_buffer);
**********************************************************************************/
U8 Read_One_Block(U32 Addr,U8 * RxBuffer)
{
U16 i=0;
U32 status=0;
U16 BlockSize; //定義塊大小
U16 BlockNumber;
BlockSize=1 << SDCard_BlockSize; //以byte為單位
BlockNumber = ((Addr >> SDCard_BlockSize) + 1) &0x0fff;
rSDIDTIMER=0x7fffff; // Set timeout count
rSDIBSIZE=0x200; // 512byte(128word)
rSDIFSTA=rSDIFSTA"(1<<16); // FIFO reset
rSDIDCON = (BlockNumber<<0)|(2<<12)|(1<<14)|(1<<16)|(1<<17)|(1<<19)|(0<<22);
while(CMD17(Addr)!=1) //發(fā)送讀單個塊指令
{
#ifdef __SD_MMC_DEBUG__
Uart_Printf("Send read addr failed!n");
#endif
}
/* 開始接收數據到緩沖區(qū) */
while(i
{
status = rSDIDSTA;
if(status&0x60) //檢查是否超時和CRC校驗是否出錯
{
rSDIDSTA=(0x3<<0x5); //清除超時標志和CRC錯誤標志
#ifdef __SD_MMC_DEBUG__
Uart_Printf("there is wrong when reading: %s.n",status&0x20 ? "time out" :"CRC error");
#endif
return 0;
}
status=rSDIFSTA;
if((status&0x1000)==0x1000) //如果接收FIFO中有數據
{
*RxBuffer=rSDIDAT;
RxBuffer++;
i++;
}
status = rSDIDSTA;
Delay(2); //延時2ms
rSDIDCON=rSDIDCON&~(7<<12); //結束SDMMC模塊的接收
rSDIDSTA = status; //清狀態(tài)標志位
/* 確認SD卡進入了空閑狀態(tài)--SDIO總線空閑 */
rSDIDCON=(0<<18)|(1<<17)|(1<<16)|(1<<14)|(1<<12)|(BlockNumber<<0);
rSDIDTIMER=0x7fffff;
status = rSDIDSTA;
while( !( ((status&0x08)==0x08) | ((status&0x20)==0x20)| ((status&0x800)==0x800) )){
status=rSDIDSTA;
}
if( (status&0x20) == 0x20 ){
rSDIDSTA = status;
return 0;
}
rSDIDSTA = status;
return 1;
}
寫函數
/**********************************************************************************
功 能:該函數用于向SD卡的一個數據塊寫入數據
參 數:
U32 Addr 被寫塊的起始地址
U8* TxBuffer 用于發(fā)送數據的緩沖區(qū)
返回值:
0 數據寫入操作失敗
1 數據寫入操作成功
舉 例:
在主調函數中定義一個數組作為發(fā)送緩沖區(qū),如U8 Tx_buffer[BlockSize];
然后開始調用Write_One_Block(addr,Tx_buffer);
**********************************************************************************/
U8 Write_One_Block(U32 Addr,const U8 * TxBuffer)
{
U16 i = 0;
U32 status = 0;
U16 BlockSize; //定義塊大小
U16 BlockNumber;
BlockSize = 1<< SDCard_BlockSize; //以byte為單位
BlockNumber = ((Addr >> SDCard_BlockSize) +1) &0x0fff;
rSDIDTIMER=0x7fffff; // Set timeout count
rSDIBSIZE=0x200; // 512 byte(128 word)
rSDIFSTA = rSDIFSTA|(1<<16); // FIFO reset
rSDIDCON = BlockNumber|(3<<12)|(1<<14)|(1<<16)|(1<<17)|(1<<20)|(0<<22);
while(CMD24(Addr)!=1) //發(fā)送寫單塊操作指令
{
#ifdef __SD_MMC_DEBUG__
Uart_Printf("Send write addr failed!n");
#endif
}
/* 開始傳遞數據到緩沖區(qū) */
while(i < BlockSize)
{
status=rSDIFSTA;
if((status&0x2000)==0x2000) //如果發(fā)送FIFO可用,即FIFO未滿
{
rSDIDAT = *TxBuffer;
TxBuffer++;
i++;
}
}
status = rSDIDSTA;
Delay(5);
rSDIDCON=rSDIDCON&~(7<<12); //結束SDMMC模塊的發(fā)送
rSDIDSTA = status;
/* 確認SD卡進入了空閑狀態(tài)--SDIO總線空閑 */
rSDIDCON=(0<<18)|(1<<17)|(1<<16)|(1<<14)|(1<<12)|(BlockNumber<<0);
rSDIDTIMER=0x7fffff; // Set timeout count
status = rSDIDSTA;
while( !( ((status&0x08)==0x08) | ((status&0x20)==0x20)| ((status&0x800)==0x800) )){
status=rSDIDSTA;
}
if( (status&0x20) == 0x20 ){
rSDIDSTA = status;
return 0;
}
rSDIDSTA = status;
return 1;
}
測試效果
以下操作都得到成功驗證:
單塊讀
單塊寫
多塊讀(調用單塊讀函數實現)
多塊寫(調用單塊寫函數實現)
先讀后寫
先寫后讀
移植fatfs文件系統測試:
大多數操作沒有故障出現,偶爾也會出現寫函數卡死的情況
仍然存在