-boot移植(十一)---代码修改---支持nandflash

发布者:sjjawx831最新更新时间:2023-06-08 来源: elecfans关键字:nandflash  擦除 手机看文章 扫描二维码
随时随地手机看文章

一、移植前的修改

1.1 include/configs/jz2440修改

  原来的定义:

  

  可以看出,要先定义CONFIG_CMD_NAND才能使能NANDFlash。

  这个在我们文件中的82行有定义,所以不需要定义了。

  

  将里面的S3C2410全部改为S3C2440:

  

1.2 drivers/mtd/nand/修改

  拷贝s3c2410_nand.c 成  s3c2440_nand.c

  

  在此目录的makefile中添加 s3c2440_nand.c

  

  修改s3c2440_nand.c 文件,从board_nand_init 逐行检测修改,代码如下:

  1 #include

  2 

  3 #include

  4 #include

  5 #include

  6 

  7 /* NFCONF 寄存器定义 */

  8 #define S3C2440_NFCONF_EN          (1<<15)

  9 #define S3C2440_NFCONF_512BYTE     (1<<14)

 10 #define S3C2440_NFCONF_4STEP       (1<<13)

 11 #define S3C2440_NFCONF_INITECC     (1<<12)

 12 #define S3C2440_NFCONF_nFCE        (1<<1)

 13 #define S3C2440_NFCONF_TACLS(x)    ((x)<<12)

 14 #define S3C2440_NFCONF_TWRPH0(x)   ((x)<<8)

 15 #define S3C2440_NFCONF_TWRPH1(x)   ((x)<<4)

 16 

 17 /* NFCONF 寄存器定义 */

 18 #define S3C2440_NFCONT_LOCKTIGHT                  (1<<13)

 19 #define S3C2440_NFCONT_SOFTLOCK                  (1<<12)

 20 #define S3C2440_NFCONT_ENLLLEGALINT              (1<<10)

 21 #define S3C2440_NFCONT_ENRnBINT                 (1<<9)

 22 #define S3C2440_NFCONT_Rn                         (1<<8)

 23 #define S3C2440_NFCONT_SPAREECCLOCK             (1<<6)

 24 #define S3C2440_NFCONT_MAINECCLOCK                 (1<<5)

 25 #define S3C2440_NFCONT_ECC                      (1<<4)

 26 #define S3C2440_NFCONT_nCE                      (1<<1)

 27 #define S3C2440_NFCONT_MODE                      (1<<0)

 28 

 29 

 30 

 31 

 32 #define S3C2440_ADDR_NALE 4

 33 #define S3C2440_ADDR_NCLE 8

 34 

 35 #ifdef CONFIG_NAND_SPL

 36 

 37 /* in the early stage of NAND flash booting, printf() is not available */

 38 #define printf(fmt, args...)

 39 

 40 static void nand_read_buf(struct mtd_info *mtd, u_char *buf, int len)

 41 {

 42     int i;

 43     struct nand_chip *this = mtd->priv;

 44 

 45     for (i = 0; i < len; i++)

 46         buf[i] = readb(this->IO_ADDR_R);

 47 }

 48 #endif

 49 

 50 /* ctrl:表示做什么,选中芯片/取消片选,是发命令还是发地址

 51  * cmd :命令值或地址值

 52  */

 53 static void s3c24x0_hwcontrol(struct mtd_info *mtd, int cmd, unsigned int ctrl)

 54 {

 55     struct s3c24x0_nand *nand = s3c24x0_get_base_nand();

 56 

 57     debug("hwcontrol(): 0x%02x 0x%02xn", cmd, ctrl);

 58 

 59     if (ctrl & NAND_CLE) {

 60         /* 发命令 */

 61         writeb(cmd, &nand->nfcmd);

 62     }

 63     else if (ctrl & NAND_ALE) {

 64         /* 发地址 */

 65         writeb(cmd, &nand->nfaddr);

 66     }

 67 

 68 }

 69 

 70 /**

 71  * nand_select_chip - [DEFAULT] control CE line

 72  * @mtd: MTD device structure

 73  * @chipnr: chipnumber to select, -1 for deselect

 74  *

 75  * Default select function for 1 chip devices.

 76  */

 77 void s3c2440_nand_select(struct mtd_info *mtd, int chipnr)

 78 {

 79     struct s3c24x0_nand *nand = s3c24x0_get_base_nand();

 80 

 81     switch (chipnr) {

 82     case -1: /* 取消选中 */

 83         nand->nfcont |= S3C2440_NFCONT_nCE;

 84         break;

 85     case 0: /* 选中 */

 86         nand->nfcont &= ~S3C2440_NFCONT_nCE;

 87         break;

 88 

 89     default:

 90         BUG();

 91     }

 92 }

 93 

 94 static int s3c24x0_dev_ready(struct mtd_info *mtd)

 95 {

 96     struct s3c24x0_nand *nand = s3c24x0_get_base_nand();

 97     debug("dev_readyn");

 98     return readl(&nand->nfstat) & 0x01;

 99 }

100 

101 #ifdef CONFIG_S3C2440_NAND_HWECC

102 void s3c24x0_nand_enable_hwecc(struct mtd_info *mtd, int mode)

103 {

104     struct s3c24x0_nand *nand = s3c24x0_get_base_nand();

105     debug("s3c24x0_nand_enable_hwecc(%p, %d)n", mtd, mode);

106     writel(readl(&nand->nfconf) | S3C2440_NFCONF_INITECC, &nand->nfconf);

107 }

108 

109 static int s3c24x0_nand_calculate_ecc(struct mtd_info *mtd, const u_char *dat,

110                       u_char *ecc_code)

111 {

112     struct s3c24x0_nand *nand = s3c24x0_get_base_nand();

113     ecc_code[0] = readb(&nand->nfecc);

114     ecc_code[1] = readb(&nand->nfecc + 1);

115     ecc_code[2] = readb(&nand->nfecc + 2);

116     debug("s3c24x0_nand_calculate_hwecc(%p,): 0x%02x 0x%02x 0x%02xn",

117           mtd , ecc_code[0], ecc_code[1], ecc_code[2]);

118 

119     return 0;

120 }

121 

122 static int s3c24x0_nand_correct_data(struct mtd_info *mtd, u_char *dat,

123                      u_char *read_ecc, u_char *calc_ecc)

124 {

125     if (read_ecc[0] == calc_ecc[0] &&

126         read_ecc[1] == calc_ecc[1] &&

127         read_ecc[2] == calc_ecc[2])

128         return 0;

129 

130     printf("s3c24x0_nand_correct_data: not implementedn");

131     return -1;

132 }

133 #endif

134 

135 int board_nand_init(struct nand_chip *nand)

136 {

137     u_int32_t cfg = 0;

138     u_int8_t tacls, twrph0, twrph1;

139     struct s3c24x0_clock_power *clk_power = s3c24x0_get_base_clock_power();

140     struct s3c24x0_nand *nand_reg = s3c24x0_get_base_nand();

141 

142     writel(readl(&clk_power->clkcon) | (1 << 4), &clk_power->clkcon);

143 

144     /* 时序设置 */

145     tacls = 4;

146     twrph0 = 8;

147     twrph1 = 8;

148     

149     /* 初始化时序 */

150     cfg |= S3C2440_NFCONF_TACLS(tacls - 1);

151     cfg |= S3C2440_NFCONF_TWRPH0(twrph0 - 1);

152     cfg |= S3C2440_NFCONF_TWRPH1(twrph1 - 1);

153     writel(cfg, &nand_reg->nfconf);

154 

155     /* 使能NANDFLASH控制器,初始化ECC,禁止片选 */

156     writel(S3C2440_NFCONT_MODE | S3C2440_NFCONT_nCE | S3C2440_NFCONT_ECC, &nand_reg->nfcont);

157     

158     /* initialize nand_chip data structure */

159     nand->IO_ADDR_R = (void *)&nand_reg->nfdata;

160     nand->IO_ADDR_W = (void *)&nand_reg->nfdata;

161 

162     nand->select_chip = s3c2440_nand_select;

163 

164     /* hwcontrol always must be implemented */

165     nand->cmd_ctrl = s3c24x0_hwcontrol;

166 

167     nand->dev_ready = s3c24x0_dev_ready;

168 

169 #ifdef CONFIG_S3C2410_NAND_HWECC

170     nand->ecc.hwctl = s3c24x0_nand_enable_hwecc;

171     nand->ecc.calculate = s3c24x0_nand_calculate_ecc;

172     nand->ecc.correct = s3c24x0_nand_correct_data;

173     nand->ecc.mode = NAND_ECC_HW;

174     nand->ecc.size = CONFIG_SYS_NAND_ECCSIZE;

175     nand->ecc.bytes = CONFIG_SYS_NAND_ECCBYTES;

176     nand->ecc.strength = 1;

177 #else

178     nand->ecc.mode = NAND_ECC_SOFT;

179 #endif

180 

181 #ifdef CONFIG_S3C2440_NAND_BBT

182     nand->bbt_options |= NAND_BBT_USE_FLASH;

183 #endif

184 

185     return 0;

186 }

 修改过程如下:

 1 u-boot-2016.05commonboard_r:board_init_r函数中的初始化序列init_sequence_r中的:

 2 initr_nand

 3     nand_init

 4         nand_init_chip

 5             board_nand_init

 6                 设置nand_chip结构体, 提供底层的操作函数

 7             nand_scan

 8                 nand_scan_ident

 9                     nand_set_defaults

10                         chip->select_chip = nand_select_chip;

11                         chip->cmdfunc = nand_command;

[1] [2]
关键字:nandflash  擦除 引用地址:-boot移植(十一)---代码修改---支持nandflash

上一篇:平台设备驱动之平台驱动
下一篇:分析uboot移植第一阶段的各种代码

推荐阅读最新更新时间:2024-11-12 01:09

u-boot-2014.10移植(4)识别dm9000
在smdk2440.h里面注释掉CS8900相关的 #if 0 #define CONFIG_DRIVER_CS8900 1 /* we have a CS8900 on-board */ #define CS8900_BASE 0x19000300 #define CS8900_BUS16 1 /* the Linux driver does accesses as shorts */ #endif #if !defined(CONFIG_DRIVER_CS8900) #define CONFIG_DRIVER_DM9000 1 #def
[单片机]
移植u-boot-1.1.6之NOR的支持
u-boot-1.1.6里面默认配置文件里面支持的nor型号是 1 #if 0 2 #define CONFIG_AMD_LV400 1 /* uncomment this if you have a LV400 flash */ 3 #endif 4 #define CONFIG_AMD_LV800 1 /* uncomment this if you have a LV800 flash */ 而我们使用的nor flash不是LV400/LV800; 我们使用的芯片是MX29LV160DB,但是uboot里面没有,而有相近的AM29LV040B,这是需要借助数据手册仿写一个 MX29LV160DB特性有: 大
[单片机]
2.3NandFlash的操作
1.NandFlash分类: 2.MLC和SLC对比: 3.信号引脚: Nandflash的读: 页读:页地址 随机读:页地址+列地址 今天是要实现页读:需要页的地址。 我们来看芯片手册K9F2G08U0A.pdf:找到桉叶读的工作原理图: 寄存器NFCONT: 在这个寄存器里的【1】位: 上面就是选择操作nandflash的位: nandflash的statue寄存器: 第四位: 清除RB:第四位为1是,清除RB。 接下来是发送的操作: 寄存器: 发送: NFDATA: 主方法: Make:编译 要实现nandf
[单片机]
2.3<font color='red'>NandFlash</font>的操作
关于stm32的flash的擦除的理解
最近消化 正点原子《STM32不完全手册》—— 第三十一章 FLASH模拟EEPROM实验 时,对判断FLASH是否需要擦除的语句有点不理解,度娘了才知:FLASH擦除后字节存储的数据是0XFF void STMFLASH_Write(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite) { u32 secpos; //扇区地址 u16 secoff; //扇区内偏移地址(16位字计算) u16 secremain; //扇区内剩余地址(16位字计算) u16 i; u32 offaddr; //去掉0X08000000后的地址 if(W
[单片机]
STM32 FLASH擦除、写入以及防止误擦除程序代码
编译环境:(Keil)MDK4.72.10 stm32库版本:STM32F10x_StdPeriph_Driver_3.5.0 一、本文不对FLASH的基础知识做详细的介绍,不懂得地方请查阅有关资料。   对STM32 内部FLASH进行编程操作,需要遵循以下流程:   1、FLASH解锁;   2、清除相关标志位;   3、擦除FLASH(先擦除后写入的原因是为了工业上制作方便,即物理实现方便);   4、写入FLASH;   5、锁定FLASH; 实例: #define FLASH_PAGE_SIZE ((uint16_t)0x400) //如果一页为1K大小 #define WRITE_START_ADDR ((uin
[单片机]
STM32 FLASH<font color='red'>擦除</font>、写入以及防止误<font color='red'>擦除</font>程序代码
S3C6410移植u-boot-2010.3(6)回顾总结
  经过半周的制作,我的OK6410总算是成功在我自己构建的uboot上运行系统了。   在此回顾一下整个过程的大纲。   一、成功编译的开始   1、下载一份未经过处理的uboot   2、精简uboot中不用到的文件,保留相似的芯片6400的板级文件   3、对精简后的uboot进行初次编译,顺便可以检查工具链、uboot的完整性   4、通过创建目录、拷贝、修改的方式,创建出初始的6410的板级文件、Makefile等   5、对自己创建的6410进行初次编译,产生的错误一般是拷贝的文件内容未进行更改,修复后一般能正常编译   二、基本启动信息的修改   1、修改汇编入口代码/cpu/arm1176/start.S,
[单片机]
S3C6410移植u-boot
  1、下载uboot http://ftp.denx.de/pub/u-boot/   2、解压文件 tar xvf u-boot-latest.tar.bz2   3、进入uboot根目录、接着开始修改文件   4、创建板级文件目录 cd /borad/samsung mkdir smdk6410   5、复制smdk6400文件到创建的文件目录 cp smdk6400/* smdk6410/   6、修改目录下的文件名 cd smdk6410 mv smdk6400.c smdk6410.c mv smdk6400_nand_spl.c smdk6410_nand_spl.c   7、修改板级文件内容
[单片机]
u-boot移植(八)---代码修改---存储控制器--MMU
一、MMU介绍 1.1 虚拟地址与物理地址   建立两个应用程序,hello1.c和hello2.c,然后运行:   hello1.c      hello2.c      运行结果如下:      可以看到两个结果打印的地址是一样的,都为 0x601040,这说明两段程序都运行于同一个地址中。我们的死循环程序又保证了两个程序在同时运行。      对于2440来说,如下图:      CPU只管发出地址,读写数据。不管地址是否是物理地址还是虚拟地址。   写程序的时候,链接地址也没有物理地址和虚拟地址。链接地址是CPU看到的,是从CPU的角度来说的。 1.2 虚拟地址转换成物理地址   对与ARM来说,虚拟地址(VA)
[单片机]
u-<font color='red'>boot</font><font color='red'>移植</font>(八)---代码修改---存储控制器--MMU
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved