uboot的sd/mmc流程

arch/arm/cpu/armv7/start.S(_main[arch/arm/lib/ctr0.S])-->board_init_r //(common/spl/spl.c)-->spl_load_image -->  drivers/mmc/mmc_init(mmc_start_init)-->mmc_send_op_cond

-->mmc_send_op_cond_iter


下面是SD卡接口的完整规范

摘自:http://www.360doc.com/content/10/1206/10/3373961_75433630.shtml

特性: 

◎  兼容 规范版本1.01      ◎卡上错误校正        ◎支持CPRM

◎两个可选的通信协议:SD模式和SPI模式 

◎可变时钟频率0-25MHz

◎通信电压范围:2.0-3.6V

工作电压范围:2.0-3.6V

低电压消耗:自动断电及自动睡醒,智能电源管理 

无需额外编程电压       ◎卡片带电插拔保护 

◎正向兼容MMC卡               ◎高速串行接口带随即存取 

---支持双通道闪存交叉存取 

---快写技术:一个低成本的方案,能够超高速闪存访问和高可靠数据存储

---最大读写速率:10Mbyte/s

◎最大10个堆叠的卡(20MHz,Vcc=2.7-3.6V)

◎数据寿命:10万次编程/擦除

◎CE和FCC认证          ◎PIP封装技术 

◎尺寸:24mm宽×32mm长×1.44mm厚

说明:

本SD卡高度集成闪存,具备串行和随机存取能力。可以通过专用优化速度的串行接口访问,数据传输可靠。接口允许几个卡垛叠,通过他们的外部连接。接口完全符合最新的消费者标准,叫做SD卡系统标准,SD卡系统规范定义。 

SD卡系统是一个新的大容量存储系统,基于半导体技术的变革。 

它的出现,提供了一个便宜的、结实的卡片式的存储媒介,为了消费多媒体应用。

SD卡可以设计出便宜的播放器和驱动器而没有可移动的部分。 

一个低耗电和广供电电压的可以满足移动电话、电池应用比如音乐播放器、个人管理器、掌上电脑、电子书、电子百科全书、电子词典等等。

使用非常有效的数据压缩比如MPEG,SD卡可以提供足够的容量来应付多媒体数据。

框图: 

SD卡上所有单元由内部时钟发生器提供时钟。接口驱动单元同步外部时钟的DAT和CMD信号到内部所用时钟。 

本卡由6线SD卡接口控制,包括:CMD,CLK,DAT0-DAT3。

在多SD卡垛叠中为了标识SD,一个卡标识寄存器(CID)和一个相应地址寄存器(RCA)预先准备好。 

一个附加的寄存器包括不同类型操作参数。 

这个寄存器叫做CSD。 

使用SD卡线访问存储器还是寄存器的通信由SD卡标准定义。

卡有自己的电源开通检测单元。 

无需附加的主复位信号来在电源开启后安装卡。 

它防短路,在带电插入或移出卡时。 

无需外部编程电压。 

编程电压卡内生成。 

SD卡支持第二接口工作模式SPI。

如果接到复位命令(CMD0)时,CS信号有效(低电平),SPI模式启用。

(译者按:以下部分为本人翻译)

接口

该SD卡的接口可以支持两种操作模式:

。SD卡模式

。SPI模式

主机系统可以选择以上其中任一模式,SD卡模式允许4线的高速数据传输。 SPI模式允许简单通用的SPI通道接口, 这种模式相对于SD模式的不足之处是丧失了速度。

SD卡模式针脚定义

 

针脚

名称

类型

描述

1

CD DAT3

I/O/PP

卡监测数据位3

2

CMD

PP

命令/回复

3

Vss

S

4

Vcc

S

供电电压

5

CLK

I

时钟

6

Css2

S

7

DAT0

I/O/PP

数据位0

8

DAT1

I/O/PP

数据位1

9

DAT2

I/O/PP

数据位2

1:S:电源供电,I:输入O:输出 I/O:双向 PP:I/O使用推挽驱动

SD卡的总线概念

SD总线允许强大的1线到4线数据信号设置。当默认的上电后,SD卡使用DAT0。 初始化之后,主机可以改变线宽(译者按:即改为2根线,3根线。。。)。混和的SD卡连接方式也适合于主机。在混和连接中Vcc,Vss和CLK的信号连接可以通用。但是,命令,回复,和数据(DAT0~3)这几根线,各个SD卡必须从主机分开。

这个特性使得硬件和系统上交替使用。SD总线上通信的命令和数据比特流从一个起始位开始,以停止位中止。

CLK:每个时钟周期传输一个命令或数据位。频率可在0~25MHz之间变化。SD卡的总线管理器可以不受任何限制的自由产生0~25MHz的频率。

CMD

命令从该CMD线上串行传输。一个命令是一次主机到从卡操作

 

回复从该CMD线上串行传输。一个命令是对之前命令的回答。回复可以来自单机或所有卡。 

DAT0~3:数据可以从卡传向主机或副versa。数据通过数据线传输。

 

SD卡总线拓扑

 

SPI模式针脚定义

针脚

名称

类型

描述

1

CS

I

片选(负有效)

2

DI

I

数据输入

3

Vss

S

4

Vcc

S

供电电压

5

CLK

I

时钟

6

Vss2

S

7

DO

O

数据输出

8

RSV

--

 

9

RSV

--

 

1:S:电源供电,I:输入O:输出 I/O:双向 PP:I/O使用推挽驱动

注意:SPI模式时,这些信号需要在主机端用10~100K欧的电阻上拉。

SPI 总线概念

SPI总线允许通过2通道(数据入和出)传输比特数据。SPI兼容模式使得MMC主机系统通过很小的改动就可以使用SD卡。SPI模式使用字节传输。

所有的数据被融合到一些字节中并aligned to the CS signal(可能是:同过CS信号来校正)。SPI模式的优点就是简化主机的设计。特别的,MMC主机需要小的改动。SPI模式相对于SD模式的不足之处是丧失了速度性能。

 

SD卡的电特性

 SD卡的连接电路图

 

 

直流特性

完全最大值评估

最大值评估指即使在瞬间也不能超出限制电压。当你在归定的最大值评估范围内使用该产品,不会出现永久性损坏。但是这并不能保证正常的逻辑操作。


1、 简介:
    SD卡(Secure Digital Memory Card)是一种为满足安全性、容量、性能和使用环境等各方面的需求而设计的一种新型存储器件,SD卡允许在两种模式下工作,即SD模式和SPI模式,本 系统采用SPI模式。本小节仅简要介绍在SPI模式下,STM32处理器如何读写SD卡,如果读者如希望详细了解SD卡,可以参考相关资料。
SD 卡内部结构及引脚如下图所示:

 


SD卡内部图




    2、SD卡管脚图:


SD卡图.JPG




    3、SPI模式下SD各管脚名称为:
    sd 卡:


SPI模式下SD各管脚名称 为.JPG




    注: 一般SD有两种模式:SD模式和SPI模式,管脚定义如下:
    (A)、SD MODE 1、CD/DATA3   2、CMD 3、VSS1 4、VDD 5、CLK 6、VSS2 7、DATA0 8、DATA1 9、DATA2
    (B)、SPI MODE 1、CS             2、DI 3、VSS 4、VDD 5、SCLK 6、VSS2 7、DO           8、RSV 9、RSV

    SD 卡主要引脚和功能为:
    CLK:时钟信号,每个时钟周期传输一个命令或数据位,频率可在0~25MHz之间变化,SD卡的总线管理器可以不受任何限制的自由产生0~25MHz 的频率;
    CMD:双向命令和回复线,命令是一次主机到从卡操作的开始,命令可以是从主机到单卡寻址,也可以是到所有卡;回复是对之前命令的回答,回复可以来自单 卡或所有卡;
    DAT0~3:数据线,数据可以从卡传向主机也可以从主机传向卡。
    SD卡以命令形式来控制SD卡的读写等操作。可根据命令对多块或单块进行读写操作。在SPI模式下其命令由6个字节构成,其中高位在前。SD卡命令 的格式如表1所示,其中相关参数可以查阅SD卡规范。


    4、MicroSD卡管脚图:


MicroSD卡管脚图.JPG



    5、MicroSD卡管脚名称:


MicroSD卡管脚名 称.JPG




    SD 卡与MicroSD卡仅仅是封装上的不同,MicroSD卡更小,大小上和一个SIM卡差不多,但是协议与SD卡相同。
    一般我们用单片机操作SD 卡时,都不需要对FAT分区表信息做处理,原因如下:
    1)、操作FAT分区表要增加程序代码量、增加SRAM的消耗,对于便携应用来说代码大小和 占用SRAM的多少至关重要。
    2)、即使我们对FAT分区表不做任何了解,实际上我们一样可以向SD卡上写入数据,这就表明使用FAT对我们做数 据存储应用来说如同鸡肋。
    3)、耗费大量经历和时间去了解FAT分区表对于我们做嵌入式软件开发的人来说有些得不偿失。
    4)、SD卡支持 两种操作模式,SD模式和SPI模式,SPI模式做SD数据操作时根本不需要知道FAT,这时候SD卡对于我们来说实际上就是个大的、快速的、方便的、容 量可变的外部存储器。
    基于以上原因,一般情况下对SD卡的操作只需要了解SPI通讯就可以了,而现在大部分单片机都有SPI接口,那么操作SD卡 易如反掌。


    以下是做SD卡试验时使用的电路图:


SD卡试验时使用的电路 图.JPG



    SD_CS/ 连接到单片机的片选SD管脚,只有单片机设置SD_CS/为低电平时才可以操作SD卡。
    MOSI连接单片机SPI总线的MOSI管脚(SPI数据 输入),单片机从这个管脚读取SD卡内的数据。
    MISO连接单片机SPI总线的MISO管脚(SPI数据输出)、单片机通过这个管脚向SD卡内写 入数据。
    SCK连接单片机SPI总线的SCK(SPI时钟)
    SD管脚实际上在SD卡内部连接到了GND,当SD插座上没插入SD卡时,单 片机从这个管脚能读到高电平(前提是使用单片机内部上拉输入,或者外部增加一个上拉电阻),一旦插入SD卡,这个管脚就变成低电平,这个功能用来检测是否 插入SD卡。
    RSV1和RSV2是保留功能管脚,不需要操作。
    MicroSD卡的连接和SD卡大同小异,只是MicroSD卡比SD卡少 一个GND管脚,所以不能使用上面做的这种插入卡的检测,实际上现在很多SD卡/MicroSD卡插座都有插入检测管脚,当然,一分钱一分货,价格上当然 也要贵一些
顺便提一下,普通SD卡插座最多5块钱。

    SPI命令格式
   

Byte 1

Byte2-5

Byte 6

7

6

5        0

31       0

7

0

0

1

Command

Command Argument

CRC

1


    以下是一个简单的测试SD卡读 写的程序,程序是基于Atmega128单片机编写的,对于Atmega的其他单片机仅需要做管脚改动就可以使用,其他单片机更改要更大。
sd.h
//******************************************************************
//SPI 各线所占用的端口
#define SD_SS          PB6                          
#define SD_SCK         PB1
#define SD_MOSI        PB2
#define SD_MISO        PB3
//******************************************************************

#define SD_DDR         DDRB
#define SD_PORT        PORTB
#define SD_PIN         PINB

#define SD_SS_H        SD_PORT |= (1<
#define SDSS_L        SD_PORT &= ~(1<
#define SD_SCK_H       SD_PORT |= (1<
#define SD_SCK_L       SD_PORT &= ~(1<
#define SD_MOSI_H      SD_PORT |= (1<
#define SD_MOSI_L      SD_PORT &= ~(1<

#define SD_MISO_IN     (SD_PIN&(1<
//-------------------------------------------------------------
// 错误号
//-------------------------------------------------------------
#define INIT_CMD0_ERROR         0xFF
#define INIT_CMD1_ERROR         0xFE
#define WRITE_BLOCK_ERROR       0xFD
#define READ_BLOCK_ERROR        0xFC
#define TRUE                    0x01
//------------------------------------------------------------- 
// MMC/SD 命令(命令号从40开始,只列出基本命令,并没有都使用)
//-------------------------------------------------------------
#define SD_RESET                0x40 + 0                   
#define SD_INIT                0x40 + 1
#define SD_READ_CSD            0x40 + 9
#define SD_READ_CID            0x40 + 10
#define SD_STOP_TRANSMISSION    0x40 + 12
#define SD_SEND_STATUS            0x40 + 13
#define SD_SET_BLOCKLEN        0x40 + 16
#define SD_READ_BLOCK            0x40 + 17
#define SD_READ_MULTI_BLOCK    0x40 + 18
#define SD_WRITE_BLOCK            0x40 + 24
#define SD_WRITE_MULTI_BLOCK    0x40 + 25

//片选关(MMC/SD-Card Invalid)
#define SD_Disable()   SD_SS_H
//片选开 (MMC/SD-Card Active)
#define SD_Enable()    SD_SS_L

SD_TEST.C
//****************************************************************************************/
//ICC-AVR application builder : 03-5-20 8:39:11
// Target : M128
// Crystal: 3.6864Mhz

#include
#include
#include 'sd.h'
void uart0_init(void);
void putchar(unsigned char content);
void putstr(unsigned char *s);
void SD_Port_Init(void);
unsigned char SD_Init(void);
unsigned char SD_write_sector(unsigned long addr,unsigned char *Buffer);
unsigned char SD_read_sector(unsigned long addr,unsigned char *Buffer);
unsigned char SPI_TransferByte(unsigned char byte);
unsigned char Write_Command_SD(unsigned char cmd,unsigned long address);
unsigned long SD_find(void);
//**************************************************************************
// 串口调试程序
//**************************************************************************
void uart0_init(void)
{
UCSR0B = 0x00; //disable while setting baud rate
UCSR0A = 0x00;
UCSR0C = 0x06; // 00000110 UART0设置为异步模式、无奇偶校验、1位停止位、8位数据位
UBRR0L = 0x17; //set baud rate lo
UBRR0H = 0x00; //set baud rate hi 设置UART0口通信速率9600
UCSR0B = 0x18;
}
void putchar(unsigned char content)
{

while(!(UCSR0A & (1 << UDRE0))); /*    判断上次发送有没有完成    */
UDR0 = content;    /*    发送数据    */

}
void putstr(unsigned char *s)
{

while(*s)
{
putchar(*s);
s++;
}

}

//****************************************************************************
// 端口初始化
void SD_Port_Init(void)
//****************************************************************************
{
SD_PORT        |= (1<<
SD_DDR         |= (1<<<
SD_DDR         &= ~(1<
}

//****************************************************************************
// 初始化 MMC/SD 卡为SPI模式
unsigned char SD_Init(void)
//****************************************************************************
{
unsigned char retry,temp;
unsigned char i;


SPCR=0x53;                           //设定SPI为128分频,慢速进行初始化
SPSR=0x00;

for (i=0;i<0x0f;i++)           
{
SPI_TransferByte(0xff); //延迟74个以上的时钟
}

SD_Enable();                    //开片选

SPI_TransferByte(SD_RESET);    //发送复位命令
SPI_TransferByte(0x00);
SPI_TransferByte(0x00);
SPI_TransferByte(0x00);
SPI_TransferByte(0x00);
SPI_TransferByte(0x95);           

SPI_TransferByte(0xff);
SPI_TransferByte(0xff);

retry=0;
do{
temp="Write"_Command_SD(SD_INIT,0);      //发送初始化命令
retry++;
if(retry==100)                             //重试100次
{
SD_Disable();                     //关片选

return(INIT_CMD1_ERROR);      //如果重试100次失败返回错误号

}
}while(temp!=0);                             

MSD_Disable();                                  //关片选

SPCR=0x50;                                        //设置SPI为2分频。进行高速读写
SPSR=0x01;

return(TRUE);                                  //返回成功
}

//****************************************************************************
// 发送命令给 MMC/SD卡
//Return: 返回MMC/SD卡对命令响应的第2字节,作为命令成功判断
unsigned char Write_Command_SD(unsigned char cmd,unsigned long address)
//****************************************************************************
{
unsigned char tmp;
unsigned char retry="0";

SD_Disable();
SPI_TransferByte(0xFF);

SD_Enable();

SPI_TransferByte(cmd);                     //将32位地址进行移位作为地址字节                                     
SPI_TransferByte(address>>24);
SPI_TransferByte(address>>16);
SPI_TransferByte(address>>8);
SPI_TransferByte(address);
SPI_TransferByte(0xFF);

SPI_TransferByte(0xFF);

do{
tmp = SPI_TransferByte(0xFF);    //发送8个时钟接受最后一个字节                                                                   
retry++;
}while((tmp==0xff)&&(retry<8));
return(tmp);

}


//****************************************************************************
// 写一个扇区(512Byte) to MMC/SD-Card
//如果写完成返回TRUE
unsigned char SD_write_sector(unsigned long addr,unsigned char *Buffer)
//****************************************************************************
{
unsigned char temp;
unsigned int i;

SPI_TransferByte(0xFF);                                    //延迟8个时钟             
SD_Enable();                                            //开片选

temp = Write_Command_MMC(MMC_WRITE_BLOCK,addr<<9);        //发送写扇区命令
if(temp != 0x00)
{
SD_Disable();
return(temp);
}

SPI_TransferByte(0xFF);
SPI_TransferByte(0xFF);
SPI_TransferByte(0xFE);

for (i=0;i<512;i++)
{
SPI_TransferByte(*Buffer++); //发送512字节数据
}

//CRC-Byte
SPI_TransferByte(0xFF); //Dummy CRC
SPI_TransferByte(0xFF); //CRC Code

temp = SPI_TransferByte(0xFF);   //读SD卡运行响应
if((temp & 0x1F)!=0x05)             //如果最后4位为0101,为操作成功。否则为操作失败。
{
SD_Disable();

return(WRITE_BLOCK_ERROR); //返回错误
}

while (SPI_TransferByte(0xFF) != 0xFF);

SD_Disable();
return(TRUE);                            //返回成功 
}
//****************************************************************************
// 读512字节 from MMC/SD-Card
//如果成功返回TRUE
unsigned char SD_read_sector(unsigned long addr,unsigned char *Buffer)
//****************************************************************************
{
unsigned char temp;
unsigned int i;
unsigned char data;



SPI_TransferByte(0xff);                                                 

MMC_Enable();

temp = Write_Command_SD(SD_READ_BLOCK,addr<<9);//发送读扇区命令

if(temp != 0x00)
{
SD_Disable();

return(READ_BLOCK_ERROR);                //返回错误号                
}

while(SPI_TransferByte(0xff) != 0xfe);


for(i=0;i<512;i++)
{
data = SPI_TransferByte(0xff);        //存数据 

*Buffer++=data;


}

SPI_TransferByte(0xff);                        //读CRC码
SPI_TransferByte(0xff);                        //读CRC码

SD_Disable();

return(TRUE);                                    //返回成功
}
//**************************************************************************
// 查找数据开始标志(预设DATASTART)根据实际需要删改
//**************************************************************************
unsigned long SD_find(void)                       
{     
unsigned long tmp="400";
unsigned char data[512];

do
{
SD_read_sector(tmp,data);                    //从0扇区开始查找
tmp++;                                        //查找DATASTART       

}while(!((data[0]=='D')&&(data[1]=='A')&&(data[2]=='T')&&(data[3]=='A')&&(data[4]=='S')&&(data[5]=='T')&&(data[6]=='A')&&(data[7]=='R')&&(data[8]=='T')));
return tmp;                                    //返回开始标志的下一个扇区
}     
//**************************************************************************
// 发送一个字节
//**************************************************************************
unsigned char SPI_TransferByte(unsigned char byte)
{
SPDR = byte;
while (!(SPSR & 0x80));                               //检测线路是否空闲                       
return SPDR;
}

//**************************************************************************
// 主程序例子
//**************************************************************************
void main(void)
{    
unsigned long temp;
unsigned char data[512];
unsigned char data2[512]={'sssssssssssssssssssssssss'};
unsigned char comm1[]={'/r/nhello world/r/n'};
unsigned char comm2[]={'/r/nSD_INIT OK/r/n'};
uart0_init();
SD_Port_Init();                     //端口初始化
if(SD_Init()== 0x01)
{                                      //SD卡初始化,并读取返回值
putstr(comm2);
}
temp="SD"_find();                      //查找DATASTART数据开始标志,返回下一扇区地址
SD_read_sector(1001,data);             //读取temp地址的512字节数据,512字节数据存入data数组
putstr(data);                
SD_write_sector(temp,data2);         //将data2数组512字节数据写入temp扇区
}

测试程序很简单,仅仅是做了一下读写SD卡的测试。

    关于SD卡的几点注意事项。
    1、无论我们愿意不愿意,SD卡每次读写数据的最小单位是1个扇区,即512个字节。
    2、SD卡与单片机连接的 SPI总线不能太长,要尽量短。这样的好处是速度可以更快,也不容易出错。
    3、虽然我们并不关心FAT文件表,但是我们仍然要关心SD卡的存储结构,如果我们不想使用PC机来读取保存在SD卡上的数据那我们就不用关心SD存储结构了。但,作为一个大容量的可移动存储设备,不能用PC机来读取是个很大的遗憾,我解决这个遗憾的方法如下:
    3-1、因为我不了解FAT复杂的结构,所以我做的程序没法去按照FAT表的各项功能来进行创建文件、删除文件、创建目录等等操作。
    3-2、虽然我们的单片机不能创建文件,但是PC机是可以创建文件的啊!所以我使用PC机将SD卡格式化,之后在SD卡上创建一个大文件,比如我的128M的SD卡上我建立了一个100M的文件。这里需要注意一下,一般使用windows创建文件的功能时是没有办法指定创建文件的大小的,空文件就是0个字节的长度,而我们是需要一个固定长度的文件的,所以我用VC编写了一个小软件,这个软件可以为我创建一个100M长度的空文件,记住,这点很重要:一个固定长度的空文件
    3-3、虽然我们建立了个文件在SD卡上,可是我们因为不去了解FAT表,所以我们一样不知道这个文件到底位于SD卡的什么地方,不要以为它会在0字节的地方开始,为了找到这个文件的开始位置,我们可以在建立的那个空文件的开头写上几个字符,比如我程序里面写的“DATASTART”,接下来我们要做的就是一个扇区一个扇区的去找这个几个特殊的字符,这是个笨方法,但却是最简单直观的方法。这个方法有两个缺点:a、如果文件建立在整个SD卡的后面,那找到这个文件需要漫长的等待。b、如果碰巧某个文件里面也有我们定义的那个特殊字符串的话,那就乱套了!不过好在我们使用的SD卡一般都是专用的,并不能拿去做其他应用,比如从公司copy点文件回家之类的,那就能保证这个SD卡上文件的简单性,即只有我们需要的那个文件,其他文件并不存在,而且这个文件肯定会从SD卡开始的那些扇区中的某一个开始。这样说来的话找到这个字符串也不是那么慢嘛!^_^。不过这里要建议一下,在使用SD卡之前最好用windows将它完全格式话一下。
    3-4、一旦我们找到了我们要写入文件的起始位置(它一般表示为一个扇区号),那我们就可以在这个起始扇区的下一个扇区写入数据了。
    4、OK,看起来很简单!有了这种存储方式我们还需要IIC接口的 EEPROM干吗呢?

 

 

 

 

 

 


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值