上一片写了一个u-boot命令saveenv的实现。里面使用到的nand_write等函数。那就来学习一些u-boot里nand flash 的驱动。
在学习nand flash驱动之前,我们要先熟悉u-boot中nand flash驱动架构以及nand flash操作原理。
在u-boot启动过程中调用了nand_init函数,这就是nand flash驱动初始化的入口点。
#if defined(CONFIG_CMD_NAND) puts ("NAND: "); nand_init(); /* go init the NAND */ #endif |
我们看,这个函数被调用的前提条件是CONFIG_CMD_NAND宏被定义,所以如果你要操作nand flash,这个宏一定要在配置文件中被定义,我们先记在这里。进入nand_init函数中。
void nand_init(void) { int i; unsigned int size = 0; for (i = 0; i < CONFIG_SYS_MAX_NAND_DEVICE; i++) { nand_init_chip(&nand_info[i], &nand_chip[i], base_address[i]); size += nand_info[i].size / 1024; if (nand_curr_device == -1) nand_curr_device = i; } printf("%u MiB\n", size / 1024); #ifdef CONFIG_SYS_NAND_SELECT_DEVICE /* * Select the chip in the board/cpu specific driver */ board_nand_select_device(nand_info[nand_curr_device].priv, nand_curr_device); #endif } |
一个for循环,这里又有一个宏CONFIG_SYS_MAX_NAND_DEVICE,表示有几个nand flash设备,一般开发板中只有一片nand flash,所以你需要在配置文件中定义这个宏为1。在进入nand_init_chip函数之前我们先将nand_init函数看完,首先计算出nand flash设备总大小,nand_curr_device表示当前nand flash设备编号,初始值为-1,由于我们这里只有一个nand flash设备,所以这个值应该用于为0才对。for循环结束之后打印出nand flash设备总的大小。你如果没有定义CONFIG_SYS_NAND_SELECT_DEVICE这个宏,那么这个函数也就结束了,nand flash也就初始化完成了。
进入nand_init_chip函数中。
在看这个函数之前,我们要看传递给这个函数的三个参数,nand_info、nand_chip和base_address。这三个参数它们都是定义在nand.c中的三个全局变量,用于保存nand flash的相关信息,这就是初始化要的关键。nand_info主要和芯片本身相关,比如记录nand flash的大小等等。
nand_chip这个结构主要记录nand flash它的操作相关,比如read、wirte等等。而base_address是记录的nand flash主控制器的寄存器基地址。它是这样定义的。
#ifndef CONFIG_SYS_NAND_BASE_LIST
static ulong base_address[CONFIG_SYS_MAX_NAND_DEVICE] = CONFIG_SYS_NAND_BASE_LIST; |
你如果没有定义CONFIG_SYS_NAND_BASE_LIST,那么寄存器基地址就是CONFIG_SYS_NAND_BASE,对于S3C2440呢这个值就为0x4E000000,所以你又需要在配置文件中定义这个宏。
static void nand_init_chip(struct mtd_info *mtd, struct nand_chip *nand, ulong base_addr) { int maxchips = CONFIG_SYS_NAND_MAX_CHIPS; int __attribute__((unused)) i = 0; if (maxchips < 1) maxchips = 1; mtd->priv = nand; nand->IO_ADDR_R = nand->IO_ADDR_W = (void __iomem *)base_addr; if (board_nand_init(nand) == 0) { if (nand_scan(mtd, maxchips) == 0) { if (!mtd->name) mtd->name = (char *)default_nand_name; else mtd->name += gd->reloc_off; #ifdef CONFIG_MTD_DEVICE /* * Add MTD device so that we can reference it later * via the mtdcore infrastructure (e.g. ubi). */ sprintf(dev_name[i], "nand%d", i); mtd->name = dev_name[i++]; add_mtd_device(mtd); #endif } else mtd->name = NULL; } else { mtd->name = NULL; mtd->size = 0; } } |
首先将nand flash操作相关指针nand_chip赋值给nand_info中的一个私有数据指针。
修改nand_chip的读写基地址为base_address,也就是0x4E000000。
然后是if语句中的board_nand_init函数,board_nand_init一看就和具体板子相关了。因为我们的处理器是S3C2440和S3C2410接近,而u-boot中又没有对S3C2440做移植,所以我们只有分析2410的。
int board_nand_init(struct nand_chip *nand) nand->ecc.correct = s3c2410_nand_correct_data; |
这个board_nand_init函数首先获取时钟电源管理的寄存器基地址,然后打开nand flash控制器的时钟使能。
然后设置nand flash控制器的配置寄存器。
又设置nand flash读写基地址为0x4e00000c。最后是对nand_chip结构的一些赋值,board_nand_init初始化完成。
没写完。等有时间续上。。。。