SD 卡(Secure Digital Memory Card)是由日本松下、东芝及美国 SanDisk 公司于 1999 年 8 月在 MMC 卡(Multi Media Card)基础上共同研制而成的具有大容量、高性能、安全性好等特点的多功能存储卡。

 

由于 SD 卡通过 9 针的硬件接口与专门的驱动器接口相连接,不需要外接电源维持记忆信息,而且作为一体化固体介质,没有任何移动部件,所以不用担心机械移动带来的损坏。由于 SD 卡读/写速度快、移动灵活性好、安全性强,加之体积很小,SD 卡被广泛使用于嵌入式便携移动装置上作为嵌入式系统的数据存储设备,如:数码相机、PDA 和多媒体播放器等。SD 卡用于 ReWorks 嵌入式系统,必须要开发相应的驱动程序。


1 SD 卡操作流程

SD 卡具备 SPI 和 SD 两种总线模式,这两种模式下的卡的初始化过程和读/写过程各有特点,具体驱动程序的实现细节不同。SD 卡有 SD 和 SPI 两种通信模式,前者可获得比后者更高的通信速率。在 SD 模式下的总线协议类似于问答方式,先由主机向卡发送 CMD 指令接着被寻址的卡做出响应,根据接收的主机指令的不同,响应帧内容和长度也不同,具体有 4 种 R1,R2,R3 和 R6,其中 R1,R3,R6 总长为 48 位,R2 总长为 136 位,响应帧格式如图 1 所示。帧传输总是以高位开始,低位结束。本文采用 SD 总线模式对 SD 卡进行操作。

 

 

1.1 SD 卡初始化过程

在对 SD 卡进行读/写操作前必须对卡进行初始化,在卡的初始化过程中所有的主机与卡的通信都只走命令接口。在识别模式下主机对卡进行软复位,由卡发布 RCA(Relative Card Address)地址,主机对其确认,此后主机用 RCA 地址对相应的卡进行操作。上电复位或者发送 CMD0 可使 SD 卡进入空闲模式(IdleState),总线激活后,在 SD 模式下应该用 CMD55(APP_CMD)命令(其中 RCA=0x0000)对卡进行操作,以使卡将下一指令解释为应用指令即 ACMD,再用 ACMD41(SD_SEND_OP_COND)对卡进行操作,卡对该指令的响应将记录在操作条件寄存器中,同时不兼容韵卡将进入非活动状态(Inactive State),此后主机发出 CMD2(ALL_SEND_CID)指令,获取卡的识别编码,获取识别码后,主机将发出 CMD3(SEND_RE-LATIVE_ADDR)指令获取 RCA 地址。以上为 SD 卡初始化的基本流程,此后主机将用 RCA 地址对卡进行读/写操作,SD 卡在 SD 模式下的初始化的详细状态转换图如图 2 所示。

 

1.2 SD 卡读/写过程

卡的读/写过程是主机通过 CMD 物理接口向卡发出相关指令,对卡进行读或者写操作,读和写操作都有单块和多块的区别,根据 SD 卡协议规范,在进行读/写操作之前,先发送 CMD16(SET_BLOCKLEN)命令来设置块大小。实现单块读和写操作分别用 CMD17(READ_SINGLE_BLOCK)和 CMD24(WRITE_BLOCK)命令,实现多块读和写操作分别用 CMD18(READ_MULTIPLE_BLOCK)和 CMD25(WRITE_MUL TIPLE_BLOCK)命令。主机和 SD 卡之间通过 FIFO 管道进行数据交换,因此读和写函数模块中且在数据交换之前,必须要对 FIFO 进行复位,通过对 S3C2440 的 SD 卡接口模块中的 SDI-FSTA 寄存器操作即可完成 FIFO 复位。在对卡读/写的过程中必须要注意在进行多块读和写操作完成后必须用 CMD12(STOP_TRANSMISSION)指令结束读/写操作,将卡转入传输状态(Transfer State)。

 

2 、ReWorks 平台 SD 卡设备驱动程序的设计

同其他操作系统一样 ReWorks 抽象了对设备的处理。所有的硬件设备都像常规文件一样看待,它们可以使用和操作文件相同的、标准的系统调用来打开、关闭和读/写。系统中的每一个设备都用一个设备文件代表。

 

在创建具体的 SD 卡设备之前必须要在内存区中开辟一块空间,实现形式为:dev_table=(SDCARD_DEV*)malloc(sizeof(SDCARD_DEV)*max_SD_devices);SDCARD_DEV 结构体包含了 block_device_operations 结构和 SD 卡相关属性,block_device_operations 结构体如下所示:

 

 

该结构体对于块设备驱动程序来讲非常重要。因为用户进程通过设备文件同硬件打交道,对设备文件的操作方式不外乎就是一些系统调用,如读/写等。ReWorks 中就是用 block_device_operations 这个关键的数据结构将系统调用和驱动程序关联起来的。

 

在该数据结构中每一个函数指针都对应了一个系统调用,用户进程利用系统调用在对设备文件进行诸如 read/write 操作时,系统调用通过设备文件的主设备号找到相应的设备驱动程序,然后读取这个数据结构相应的函数指针,接着把控制权交给该函数。该结构体包含了块设备操作所需的函数接口和必要的设备属性。

 

空间开辟成功后,便向系统注册设备,实现形式为:register_block_driver(SD_MAJOR,max_SD_devices,(block_ops*)dev_table,sizeof(SDCARD_DEV),”SDdriver”,&ret_major);其实就是向系统设备表中写入相关信息,将上面开辟的 SDCARD_DEV 类型的内存空间和所要创建的块设备建立关联,再对 SDCARD_DEV 类型的内存空间进行必要的初始化,为下面创建 SD 卡块设备做好准备。

 

接下来便是创建 SD 卡设备。ReWorks 系统采用 mknod 创建块设备,并使用主(major)和次(minor)设备编号来描述设备。函数原型如下:

 

int mknod(const char*pathname,mode_t mode,dev_t dev);

 

其参数说明如表 1 所示。

 

 

在实际开发中的实现形式为:mknod(dev_name,S IFBLK|0777,MKDEV(SD_MAJOR,ram_minor));

 

至此,SD 卡块设备的创建工作已经基本完成。下面所要做的就是功能函数的编写,首先用 block_device_operations*pSdDev;声明一个结构体指针,再用实际对卡操作的函数名填充该结构体如下:

 

pSdDev-》bd_blkRd=SD_blk_Read;pSdDev-》bd_blkWrt=SD_blk_Write;pSdDev-》bd_ioctl=SD_Ioctl;

 

最后实现所有的对卡进行实际操作的功能函数,如下:

 

 

至此,ReWorks 平台上 SD 卡驱动程序的总体结构基本完成。项目中实际使用的主机控制器是 S3C2440,在实际的开发中还需要按照前面描述的 SD 卡基于 SD 模式下的总线协议的操作流程,对卡进行必要的初始化工作,主要是对 S3C2440 的 SD 卡接口寄存器按照一定的程序进行操作,包括卡的波特率设置、工作模式设置、FIFO 管道初始化、CMD 命令控制寄存器的操作、对卡的响应信息的解析、数据路径编程和所有的过程所需的中断控制寄存器的操作等工作。

 

3、 结语

经测试,在 ReWorks 嵌入式系统平台上,利用系统建立块设备的工作流程和关键数据结构创建的 SD 模式下的 SD 卡驱动程序,很好地完成了对 SD 卡读/写操作的 BSP 层支持,数据读写安全可靠,已成功应用于某控制系统。