SD卡读写包括两种模式

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

SD卡读写包括两种模式:SD模式和SPI模式。其中SD模式又可以分为1bit 和4bit两种传输模式。SD卡缺省使用专有的SD模式。SD卡规范中主要讲了一些命令,响应和CRC效验等等,整个规范的内容还是很多的。

SD卡上电后,卡处于空闲状态,主机发送CMD0复位SD卡,然后通过CMD55和ACMD41判断当前电压是否在卡的工作范围内。在得到了正确的响应后,主机可以继续通过CMD10读取SD卡的CID寄存器,通过CMD16设置数据块长度,通过CMD9读取卡的CSD寄存器。从CSD寄存器中,主机可以获知卡容量,支持的命令集等重要参数。此时,卡以进入了传输状态,主机就可以通过CMD17/18和CMD24/25对卡进行读写。CRC校验是为了防止SD卡的命令,应答,数据传输出现错误。每个命令和应答信号都会产生CRC效验码,每个数据块的传输也会长生CRC效验码。

这段程序是友善之臂推出的mini2440开发板中带的ADS测试源码。整个阅读代码的过程是对这S3C2440的芯片手册和SD卡规范来看的,对于MMC卡没有给出注释,其实和SD卡是大同小异。由于是初次接触ARM,对SD规范的认识也不是很深入,再加上自己水平有限,还不能完全读懂源代码,其中的肯定存在一些错误,欢迎大家一起交流讨论。

#define INT 1

#define DMA 2

int CMD13(void);// Send card status

int CMD9(void);

unsigned int*Tx_buffer;//128[word]*16[blk]=8192[byte] unsigned int*Rx_buffer;//128[word]*16[blk]=8192[byte] volatile unsigned int rd_cnt;//读数据计数器

volatile unsigned int wt_cnt;//写数据计数器

volatile unsigned int block;//读写块总数

volatile unsigned int TR_end=0;

int Wide=0;// 0:1bit, 1:4bit

int MMC=0;// 0:SD , 1:MMC

int Maker_ID;

char Product_Name[7];

int Serial_Num;

volatile int RCA;

void Test_SDI(void)

{

U32 save_rGPEUP, save_rGPECON;

RCA=0;

MMC=0;

block=3072;//3072Blocks=1.5MByte,

((2Block=1024Byte)*1024Block=1MByte)

save_rGPEUP=rGPEUP;

save_rGPECON=rGPECON;

//**配置SD/MMC控制器

rGPEUP = 0xf83f;// SDCMD, SDDAT[3:0] => PU En. rGPECON = 0xaaaaaaaa;//SDCMD, SDDAT[3:0]

Uart_Printf("\nSDI Card Write and Read Test\n");

if(!SD_card_init())//等待SD卡初始化完成

return;

TR_Buf_new();//发送数据缓冲区初始化

Wt_Block();//写卡

Rd_Block();//读卡

View_Rx_buf();

if(MMC)

TR_Buf_new();

if(MMC)

{

rSDICON |=(1<<5);// YH 0519, MMC Type SDCLK

Wt_Stream();

Rd_Stream();

View_Rx_buf();

}

Card_sel_desel(0);// Card deselect

if(!CMD9())

Uart_Printf("Get CSD fail!!!\n");

rSDIDCON=0;//tark???

rSDICSTA=0xffff;

rGPEUP=save_rGPEUP;

rGPECON=save_rGPECON;

}

void TR_Buf_new(void)//发送数据缓冲区初始化

{

//-- Tx & Rx Buffer initialize

int i, j;

Tx_buffer=(unsigned int*)0x31000000;

j=0;

for(i=0;i<2048;i++)//128[word]*16[blk]=8192[byte]

*(Tx_buffer+i)=i+j;

Flush_Rx_buf();

}

void Flush_Rx_buf(void)//接收数据缓冲区清0

{

//-- Flushing Rx buffer

int i;

Rx_buffer=(unsigned int*)0x31800000;

for(i=0;i<2048;i++)//128[word]*16[blk]=8192[byte]

*(Rx_buffer+i)=0;

Uart_Printf("End Rx buffer flush\n");

}

void View_Rx_buf()

{

//-- Display Rx buffer

int i,error=0;

Tx_buffer=(unsigned int*)0x31000000;

Rx_buffer=(unsigned int*)0x31800000;

Uart_Printf("Check Rx data\n");

for(i=0;i<128*block;i++)

{

if(Rx_buffer[i]!= Tx_buffer[i])

{

Uart_Printf("\nTx/Rx error\n");

Uart_Printf("%d:Tx-0x%08x, Rx-0x%08x\n",i,Tx_buffer[i], Rx_buffer[i]);

error=1;

break;

}

}

相关文档
最新文档