移植u-boot2012.04.1 -》2440 (二)norflash 识别

    在上一篇文章中,我们实现了新建单板,时钟 sdram 等一系列初始化工作,串口已经能正确输出打印信息,但是有错误信息。


    现在,我们就来解决问题。搜索“Flash:”,或者往下看一下代码,不难发现 arch\arm\lib\board.c -》board_init_r 函数中有一句。

#if !defined(CONFIG_SYS_NO_FLASH)
	puts("Flash: ");

	flash_size = flash_init();
    flash_init 就是 norflash 的初始化函数了,下面来简单分析一下 flash_init 函数 norflash 的识别过程。

    drivers\mtd\cfi_flash.c -> flash_init

if (!flash_detect_legacy(cfi_flash_bank_addr(i), i))
			flash_get_size(cfi_flash_bank_addr(i), i);
board_flash_get_legacy(base, banknum, info)
<span style="white-space:pre">	</span>portwidth = FLASH_CFI_16BIT;
<span style="white-space:pre">	</span>chipwidth = FLASH_CFI_BY16;
<span style="white-space:pre">	</span>interface = FLASH_CFI_X16;
if (board_flash_get_legacy(base, banknum, info)) {
		// 如果 info 中没有指定厂商,我们假设它是下面两种之一,分别进行尝试
		if (!info->vendor) {
			int modes[] = {
				CFI_CMDSET_AMD_STANDARD,
				CFI_CMDSET_INTEL_STANDARD
			};
			int i;
                        // 尝试两种厂商,循环两次,第一次是 CFI_CMDSET_AMD_STANDARD
			for (i = 0; i < sizeof(modes) / sizeof(modes[0]); i++) {
				info->vendor = modes[i];
				info->start[0] =
					(ulong)map_physmem(base,
							   info->portwidth,
							   MAP_NOCACHE);
				if (info->portwidth == FLASH_CFI_8BIT
					&& info->interface == FLASH_CFI_X8X16) {
					info->addr_unlock1 = 0x2AAA;
					info->addr_unlock2 = 0x5555;
				} else {  // 执行这个分支,norflash 位宽16bit
					info->addr_unlock1 = 0x5555;
					info->addr_unlock2 = 0x2AAA;
				}
				flash_read_jedec_ids(info);// 读芯片ID
				debug("JEDEC PROBE: ID %x %x %x\n",
						info->manufacturer_id,
						info->device_id,
						info->device_id2);
				if (jedec_flash_match(info, info->start[0])) //匹配flash芯片
					break;
				else
					unmap_physmem((void *)info->start[0],
						      MAP_NOCACHE);
			}
		}
                ...
	}
    首先引起我注意的是 info->addr_unlock1 和 info->addr_unlock1 ,因为它和我们 norflash 芯片手册中的两个值很相似。

    如果没猜错的话,unlock1 就对应于Cycles 1 、3、5 ,unlock2 对应于 Cycles 2、4、6,由于我们的norflash位宽是16bit->word,那么我们这里应该分别修改为 info->addr_unlock1 = 0x555;info->addr_unlock2 = 0x2AA;
    在 flash_read_jedec_ids(info);执行完之后,下面会debug出它读取出的信息.
    根据上图芯片手册,正确的话我们的厂家 ID = 01 ,device id = 0x22C4 .
    由于 debug 默认是不开启的,我们需要修改宏,配置输出 debug 信息,在 include\common.h
    增加 #define DEBUG 开启 debug 输出,然后 make 烧写!
    我们可以看到,已经能够正确识别出我们的 norflash id ,有些同学可能会奇怪下面又一次识别的是啥,那个好像不对,还记得前边 for 循环么?默认的第一个厂家识别匹配不成功的话,会尝试第二次识别匹配,我们识别是成功了,但是还不知道匹配行不行呢,解决了匹配问题,第二次识别就不会执行了。
int jedec_flash_match(flash_info_t *info, ulong base)
{
	int ret = 0;
	int i;
	ulong mask = 0xFFFF;
	if (info->chipwidth == 1)
		mask = 0xFF;

	for (i = 0; i < ARRAY_SIZE(jedec_table); i++) {
		if ((jedec_table[i].mfr_id & mask) == (info->manufacturer_id & mask) &&
		    (jedec_table[i].dev_id & mask) == (info->device_id & mask)) {
			fill_info(info, &jedec_table[i], base);
			ret = 1;
			break;
		}
	}
	return ret;
}
    这个函数的功能能一目了然,取出 jecec_table 数组中的每一项,和我们识别出来的厂家ID 设备ID 进行比较,如果比较成功,将 jecec_table 中的信息填充到 info 中去,那么我们未匹配成功十有八九就是 jecec_table 中没有我们 norflash 的信息咯。
    打开 jecec_table 确实没有与我们匹配的项目,因此照葫芦画瓢添加一个。drivers\mtd\Jedec_flash.c 
	{	//增加
		.mfr_id		= (u16)0x01,
		.dev_id		= 0x2249,
		.name		= "S29AL016J",
		.uaddr		= {
			[1] = MTD_UADDR_0x0555_0x02AA /* x16 */
		},
		.DevSize		= SIZE_2MiB,
		.CmdSet			= CFI_CMDSET_AMD_LEGACY,
		.NumEraseRegions	= 4,
		.regions		= {
			ERASEINFO(0x04000, 1),
			ERASEINFO(0x02000, 2),
			ERASEINFO(0x08000, 1),
			ERASEINFO(0x10000, 31),
		}
	},
    uaddr :和我们前边修改的保持一致,先发0x555后发0x2AA
    cmdset:不清楚什么意思,但是数据中的值大多都是这一个 0xfff0,先试试不行再说
    NumEraseRegions:几种不同的 sector ,与.regions 对应

    我这款 norflash :
        sector0          :16 * 1024 bytes (0x4000) 1个
        sector1-sector2  : 8 * 1024 bytes (0x2000) 2个
        sector3          :32 * 1024 bytes (0x8000) 1个
        sector4-sector34 :64 * 1024 bytes (0x8000)31个
    .regions 显然就是上边的信息了,ok添加完毕 make 烧写。

    已经能够正确识别 norflash 并且成功进入控制台了,print 能够正确打印环境变量,有一个nor的错误信息,

ERROR:too many flash sectors ,搜索 too many flash sectors

    include\configs\smdk2440.h 中定义了

        #define CONFIG_SYS_MAX_FLASH_SECT     (19)

   修改为 #define CONFIG_SYS_MAX_FLASH_SECT     (35) //我的sector是35个

    修改完成之后,norflash 部分算是没有问题了,nandflash 的问题在下一篇文章中解决。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值