Env在u-boot中通常有两种存在方式,在永久性存储介质中(flash、NVRAM等),在SDRAM中。可配置不适用env的永久存储方式,但不常用。U-boot在启动时会将存储在永久性存储介质中的env重新定位到RAM中,这样可以快速访问,同时可以通过saveenv将RAM保存到永久性存储介质中。

  1. 相关结构体

env_t定义于include/environment.h中

typedef struct environment_s {

uint32_t crc; /* CRC32 over data bytes */

#ifdef CFG_REDUNDAND_ENVIRONMENT

unsigned char flags; /* active/obsolete flags */

#endif

unsigned char data[ENV_SIZE];

} env_t;

结构说明:

crc是u-boot在保存env的时候加上去的校验头,在第一次启动时一般crc校验会出错,这很正常,因为此时Flash中的数据无效。

data字段保存实际的环境变量。U-boot的env采用name=value”\0”的方式存储,在所有env的最后以“\0\0”表示整个env的结束。新的name=value对总是被添加到env数据块的末尾,当删除一个name=value对时,后面的环境变量将前移,对一个已经存在的环境变量的修改实际上先删除再插入。

env会从flash等存储设备重定位到RAM中,在env的不同实现版本(env_xxx.c)中定义了env_ptr,它指向envRAM中的位置。u-boot在重定位env后对环境变量的操作都是针对env_ptr。

env_embedded.c-----env和uboot存储于同一块区。

env_nand.c------------env存储在nandflash中。

env_dataflash.c --env存储在dataflash中。

env_eeprom.c --env存储在eeprom中。

env_flash.c --env存储在norflash中。

env_ptr指向环境参数区,系统启动时默认的环境参数default_environment[],定义于common/env_common.c中

uchar default_environment[] = {

61 #ifdef CONFIG_BOOTARGS

62 "bootargs=" CONFIG_BOOTARGS "\0"

63 #endif

64 #ifdef CONFIG_BOOTCOMMAND

65 "bootcmd=" CONFIG_BOOTCOMMAND "\0"

66 #endif

......

127 #if defined(CONFIG_PCI_BOOTDELAY) &&
(CONFIG_PCI_BOOTDELAY > 0)

128 "pcidelay=" MK_STR(CONFIG_PCI_BOOTDELAY) "\0"

129 #endif

130 #ifdef CONFIG_EXTRA_ENV_SETTINGS

131 CONFIG_EXTRA_ENV_SETTINGS

132 #endif

133 "\0"

134 };

参数解释如下:

bootfile 定义缺省的下载文件

bootargs 定义传递给Linux内核的命令行参数

bootcmd 定义自动启动时执行的几条命令

serverip 定义tftp服务器端的IP地址

env_t中除了数据之外还包含校验头,u-boot把env_t的数据指针又保存在另外一个地方,这就是gd_t数据结构(不同平台有不同的gd_t),这里以ARM为例仅列出env相关的部分。

typedef struct
global_data

{

……

unsigned long
env_off;

unsigned long
env_addr;

unsigned long
env_valid; /* checksum of environment valid */

……

} gd_t;

<include/asm-arm/global_data.h>

gd_t.env_addr即指向env_ptr->data。

  1. 相关文件

common/env_common.c

供u-boot调用的通用函数接口,它们隐藏了env的不同实现方式,比如dataflash、eeprom、flash等。

common/env_dataflash.c

env存储在dataflash中的实现

common/cmd_nvedit.c

实现u-boot对环境变量的操作命令

environment.c

环境变量以及一些宏定义

env如果存储在flash中还需要flash的支持。

  1. 环境变量操作流程

Env初始化

Start_armboot:lib_arm/board.c

*env_init:env_xxx.c(xxx
= nand|flash|eeprom……)

env_relocate:env_common.c

*env_relocate_spec:env_xxx.c

ENV_IS_EMBEDDED:env是否存在于u-boot
TEXT段中。

CFG_ENV_SIZE:env块的大小。

CFG_ENV_IS_IN_NAND:env块是否存在Nand
Flash中。

CFG_ENV_OFFSET:env块在Flash中偏移地址。

  1. env_init

实现env的第一次初始化,对于nand
env(非embedded方式):

env_nand.c:env_int

gd->env_addr =
(ulong)&default_environment[0];

gd-env_valid = 1;

  1. env_relocate

env_common.c:env_relocate

DEBUGE(“%s[%d] offset
= 0x%lx\n”, __FUNCTION__, __LINE__, gd->reloc_off);

env_ptr = (env_t
*)malloc(CFG_ENV_SIZE);

DEBUGE(“%s[%d]
malloced ENV at %p\n”, __FUNCTION__, __LINE__, env_ptr);

env_relocate_spec();

gd->env_addr =
(ulong)&(env_ptr->data);

  1. env_relocate_spec

size_t total;

ret =
readenv(CFG_ENV_OFFSET, (u_char *) env_ptr);

//
nand_read(&nand_info[0], CFG_ENV_OFFSET, &total, (u_char
*)env_ptr);

if(ret || total!=
CFG_ENV_SIZE)

return use_default();

if(crc32(0,
env_ptr->data, ENV_SIZE) != env_ptr->crc)

return use_default();

env_relocate_spec的意图就是调用nand_read将环境变量从CFG_ENV_OFFSET处读出,环境变量的大小为CFG_ENV_SIZE(注意CFG_ENV_OFFSET和CFG_ENV_SIZE要和nandflash的块/页边界对齐。读出数据后再调用crc32对env_ptr->data进行校验并与保存在env_ptr->crc的校验码对比,确认数据是否出错。从这里可以看出在系统第一次启动时,Nand
Flash里面没有存储任何环境变量,crc校验肯定出错,当我们保存环境变量后,接下来在启动板子u-boot就不会再报crc32出错了。

  1. saveenv

env_nand.c:saveenv

其调用nand_erase和nand_write进行nand
flash的erase、write。nand_write和nand_erase是nand驱动建构。

env在内存中位置不定。env在内存中的空间是由malloc分配的(env_common.c中的env_relocate()),因此其在内存中的位置是一直变化的。每次系统启动时,env在内存中位置可能都不一样。

  1. 环境变量优化

由于u-boot代码通常达到100KB左右,且必须从地址0处开始,按照这样的分配方式(Nandflash结构),我们必须为env分配一块64KB的sector,而实际使用到的可能只是其中几百个字节!u-boot还会为env在RAM中保持一块同样大小的空间,这就造成ROM和RAM空间不必要的浪费。

为了尽可能减少资源浪费,同时保证系统的健壮性,我们可以把env放置在flash中容量最小的sector里。这样,env嵌入(embed)到u-boot的代码段。在common/environment.h中会比较env和monitor的范围,如果确定env位于monitor内,则定义ENV_IS_EMBEDDED。

uboot中变量env(收集)的更多相关文章

  1. Linux系统——访问U-BOOT环境变量

    Linux系统下访问U-BOOT环境变量 移植过U-BOOT的人,都知道:在U-BOOT中存有ENV.但U-BOOT在引导内核启动之后,U-BOOT的生命周期就结束了.那么启动LINUX内核之后,U- ...

  2. u-boot中环境变量的实现

    转载:http://blog.chinaunix.net/uid-28236237-id-3867041.html U-boot中通过环境参数保存一些配置,这些配置可以通过修改环境参数.保存环境参数. ...

  3. 在linux系统中通过fw_printenv查看和设置u-boot中的环境变量

    uboot下可以通过命令访问(printenv)和修改环境变量(setenv),但是如果需要在Linux系统下访问这些数据该怎么办呢?其实uboot早就帮我们想好了.  1.编译fw_printenv ...

  4. 在uboot中加入cmd_run命令,运行环境变量

    在学习uboot的过程中会经常烧录程序,每次都要敲一些下载指令.这样是不是很麻烦,有什么办法能快速的烧写呢.很简单,将需要敲击的指令编译到uboot中,以环境变量的形式存在.但是环境变量很好加,如何运 ...

  5. U-Boot中的filesize环境变量

    U-Boot中的环境命令可以使用$(filesize)来确定刚下载(传输)得到的文件大小. 因为使用类似tftp命令传输文件后,会自动更新filesize环境变量.如:setenv updateroo ...

  6. u-boot中filesize环境变量【转载】

    转载地址:https://blog.csdn.net/fzs333/article/details/48518559 U-Boot中的环境命令可以使用$(filesize)来确定刚下载(传输)得到的文 ...

  7. vue中利用.env文件存储全局环境变量,以及配置vue启动和打包命令

    目录 1,前言 2,.env文件的作用 3,配置.env文件 4,配置启动命令 5,获取.env中的全局变量 5,实际用处 1,前言 分享一下vue项目中利用.env文件存储全局环境变量,以及利于项目 ...

  8. uboot环境变量实现分析

    u-boot的环境变量用来存储一些经常使用的参数变量,uboot希望将环境变量存储在静态存储器中(如nand nor eeprom mmc). 其中有一些也是大家经常使用,有一些是使用人员自己定义的, ...

  9. 在Linux里读取UBOOT环境变量

    转载:http://falloutmx.blog.163.com/blog/static/39236020201211145010154/ 可以通过mtd方式读取,也可以用ioremap方式.不过这些 ...

随机推荐

  1. Oracle JDBC配置

    Oracle数据库是收费的,Oracle的JDBC驱动在Maven仓库中虽有其名,却无法下载到jar包. 这就需要手动下载Oracle JDBC驱动并将其安装到本地Maven仓库中. 如果从官网下载, ...

  2. HTML:关于HTML的Doctype和严格模式与混杂模式

     DOCTYPE标签是一种标准通用标记语言的文档类型声明,它的目的是要告诉标准通用标记语言解析器,它应该使用什么样的文档类型定义(DTD)来解析文档. Doctype可声明三种DTD类型,分别表示严格 ...

  3. XP系统下建立WIFI热点让手机、电脑能上网

    http://wenku.baidu.com/view/372c5b1fa300a6c30c229f42.html 这里记录xp系统下建立共享无线网络连接,若是支持手机设备上的话,网络适配器必须是wi ...

  4. Quartz2之入门示例

    环境:XP+Myeclipse6.5+JDK1.6 quartz官网:http://www.quartz-scheduler.org/ 参考资料 1 Quartz任务调度快速入门 http://www ...

  5. OGG_Oracle GoldenGate简介(概念)

    2014-03-01 Created By BaoXinjian

  6. setContentView(R.layout.activity_main)无法正常引用

    今天在写Android代码的过程中,编译器一直报错,错误出在这一行代码: setContentView(R.layout.activity_main) 提示信息是: activity_main can ...

  7. select函数的并发限制和 poll 函数应用举例

    一.用select实现的并发服务器,能达到的并发数,受两方面限制 1.一个进程能打开的最大文件描述符限制.这可以通过调整内核参数.可以通过ulimit -n来调整或者使用setrlimit函数设置,  ...

  8. 删除vs中最近的项目的方法

    Microsoft Visual Studio中可以自行设置显示多少个最近打开的项目,但有些时候会建个项目做测试,用完了就删了,却总显示在“文件”-“最近的项目”菜单中以及“起始页”-“打开现有项目” ...

  9. Nginx(三):日志文件管理

    一.Nginx日志描述 通过访问日志,你可以得到用户地域来源.跳转来源.使用终端.某个URL访问量等相关信息: 通过错误日志,你可以得到系统某个服务或server的性能瓶颈等.因此,将日志好好利用,你 ...

  10. ev3dev:c语言开发lego ev3主机

    有了ev3dev,真是一且皆有可能啊,最近,看到一个好人,居然做了一个ev3的c库. 激动中... 学习有目标了...,赶紧记录一下. https://github.com/theZiz/ev3c