Nand Flash -详述4
?【内存技术设备,MTD(Memory Technology Device)】
MTD,是Linux的存储设备中的一个子系统。其设计此系统的目的是,对于内存类的设备,提供一个抽象层,一个接口,使得对于硬件驱动设计者来说,可以尽量少的去关心存储格式,比如FTL,FFS2等,而只需要去提供最简单的底层硬件设备的读/写/擦除函数就可以了。而对于数据对于上层使用者来说是如何表示的,硬件驱动设计者可以不关心,而MTD存储设备子系统都帮你做好了。 对于MTD字系统的好处,简单解释就是,他帮助你实现了,很多对于以前或者其他系统来说,本来也是你驱动设计者要去实现的很多功能。换句话说,有了MTD,使得你设计Nand Flash的驱动,所要做的事情,要少很多很多,因为大部分工作,都由MTD帮你做好了。 当然,这个好处的一个“副作用”就是,使得我们不了解的人去理解整个Linux驱动架构,以及MTD,变得更加复杂。但是,总的说,觉得是利远远大于弊,否则,就不仅需要你理解,而且还是做更多的工作,实现更多的功能了。
有限的通过出复用来实现输入输出命令和地址/数据等的IO接口,最小单位是页而不是常见的bit,写前需擦除等,导致了这类设备,不能像平常对待硬盘等操作一样去操作,只能采取一些特殊方法,这就诞生了MTD设备的统一抽象层。 MTD,将nand flash,nor flash和其他类型的flash等设备,统一抽象成MTD设备来管理,根据这些设备的特点,上层实现了常见的操作函数封装,底层具体的内部实现,就需要驱动设计者自己来实现了。具体的内部硬件设备的读/写/擦除函数,那就是你必须实现的了。 ? 表4.MTD设备和硬盘设备之间的区别 ? 关于mtd设备驱动,感兴趣的可以去参考 MTD原始设备与FLASH硬件驱动的对话 MTD原始设备与FLASH硬件驱动的对话-续 那里,算是比较详细地介绍了整个流程,方便大家理解整个mtd框架和nand flash驱动。 ? 【Nand flash驱动工作原理】 在介绍具体如何写Nand Flash驱动之前,我们先要了解,大概的,整个系统,和Nand Flash相关的部分的驱动工作流程,这样,对于后面的驱动实现,才能更加清楚机制,才更容易实现,否则就是,即使写完了代码,也还是没搞懂系统是如何工作的了。 让我们以最常见的,Linux内核中已经有的三星的Nand Flash驱动,来解释Nand Flash驱动具体流程和原理。 ? 此处是参考2.6.29版本的Linux源码中的driversmtdnands3c2410.c,以2410为例。 1.?? 在nand flash驱动加载后,第一步,调用对应的init函数 ---- s3c2410_nand_init: 去将nand flash驱动注册到Linux驱动框架中。 2.?? 驱动本身真正的开始,是从probe函数: s3c2410_nand_probe->s3c24xx_nand_probe, 3.?? 需要多解释一下的,是这部分代码: ?????? for (setno = 0; setno < nr_sets; setno++,nmtd++) { /*调用init chip去挂载你的nand驱动的底层函数到"nand flash的结构体"中,以及设置对应的"ecc mode",挂载ecc相关的函数 */ /* scan_ident,扫描nand 设备,设置nand flash的默认函数,获得物理设备的具体型号以及对应各个特性参数,这部分算出来的一些值,对于nand flash来说,是最主要的参数,比如nand flash的芯片的大小,块大小,页大小等。 */ ????????????? if (nmtd->scan_res == 0) { /*扫描的后一阶段,经过前面的scan_ident,我们已经获得对应nand flash的硬件的各个参数, /*add partion,根据你的nand flash的分区设置,去分区 */ ? 4.?? 等所有的参数都计算好了,函数都挂载完毕,系统就可以正常工作了。 上层访问你的nand falsh中的数据的时候,通过MTD层,一层层调用,最后调用到你所实现的那些底层访问硬件数据/缓存的函数中。 ?
关于上面提到的,在nand_scan_tail的时候,系统会根据你的驱动,如果没有实现一些函数的话,那么就用系统默认的。如果实现了自己的函数,就用你的。 1.?????? 对于驱动框架部分 其实,要了解,关于驱动框架部分,你所要做的事情的话,只要看看三星的整个nand flash驱动中的这个结构体,就差不多了: static struct platform_driver s3c2410_nand_driver = { 对于上面这个结构体,没多少要解释的。从名字,就能看出来:
而对于底层硬件操作的有些函数,总体上说,都可以在上面提到的s3c2410_nand_init_chip中找到: static void s3c2410_nand_init_chip(struct s3c2410_nand_info *info, ?????? struct nand_chip *chip = &nmtd->chip; ?????? chip->write_buf??? = s3c2410_nand_write_buf; ?????? switch (info->cpu_type) { /* nand flash控制器中,一般都有对应的数据寄存器,用于给你往里面写数据,表示将要读取或写入多少个字节(byte,u8)/字(word,u32) ,所以,此处,你要给出地址,以便后面的操作所使用 */ ????????????? chip->IO_ADDR_W??? = regs + S3C2410_NFDATA; ????????????? break; 。。。。。。 ????? } ?????? chip->IO_ADDR_R = chip->IO_ADDR_W; ?????? nmtd->info????????????? = info; ?????? if (hardware_ecc) { /* 此处,多数情况下,你所用的Nand Flash的控制器,都是支持硬件ECC的,所以,此处设置硬件ECC(HW_ECC) ,也是充分利用硬件的特性, ????????????? chip->ecc.mode???????? = NAND_ECC_HW;? ?//设置成了硬件方式校验ecc ????????????? switch (info->cpu_type) { ????????????? case TYPE_S3C2410: ???????????????????? chip->ecc.hwctl????? = s3c2410_nand_enable_hwecc; ???????????????????? break; ?????? } else { ?????? if (set->ecc_layout != NULL) ?????? if (set->disable_ecc) ? 而我们要实现的底层函数,也就是上面蓝色标出来的一些函数而已: (1)s3c2410_nand_write_buf和 s3c2410_nand_read_buf:这是两个最基本的操作函数,其功能,就是往你的nand flash的控制器中的FIFO读写数据。一般情况下,是MTD上层的操作,比如要读取一页的数据,那么在发送完相关的读命令和等待时间之后,就会调用到你底层的read_buf,去nand Flash的FIFO中,一点点把我们要的数据,读取出来,放到我们制定的内存的缓存中去。写操作也是类似,将我们内存中的数据,写到Nand Flash的FIFO中去。 当然,除了这些你必须实现的函数之外,在你更加熟悉整个框架之后,你可以根据你自己的nand flash的特点,去实现其他一些原先用系统默认但是效率不高的函数,而用自己的更高效率的函数替代他们,以提升你的nand flash的整体性能和效率。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |