转:https://blog.csdn.net/weixin_34355715/article/details/85751477

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,它指向env在RAM中的位置。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中setenv和saveenv分析的更多相关文章

  1. u-boot中nandflash初始化流程分析(转)

    u-boot中nandflash初始化流程分析(转) 原文地址http://zhuairlunjj.blog.163.com/blog/static/80050945201092011249136/ ...

  2. U-boot中SPL功能和源码流程分析

    在U-boot目录下,有个比较重要的目录就是SPL的,SPL到底是什么呢?为什么要用它呢? SPL(Secondary programloader)是uboot第一阶段执行的代码.主要负责搬移uboo ...

  3. tiny4412 串口驱动分析一 --- u-boot中的串口驱动

    作者:彭东林 邮箱:pengdonglin137@163.com 开发板:tiny4412ADK+S700 4GB Flash 主机:Wind7 64位 虚拟机:Vmware+Ubuntu12_04 ...

  4. 分析uboot中 make xxx_config过程

    make xxx_config实质上就是调用了 首先看MKCONFIG: [注意]SRCTREE=源文件下的目录 之后的语句: @$(MKCONFIG) $(@:_config=) arm arm92 ...

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

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

  6. u-boot中分区和内核MTD分区关系

    一.u-boot中环境变量与uImage中MTD的分区关系 分区只是内核的概念,就是说A-B地址放内核,C-D地址放文件系统,(也就是规定哪个地址区间放内核或者文件系统)等等. 一般我们只需要分3-4 ...

  7. U-Boot 启动过程和源码分析(第二阶段)-main_loop分析

    1> main_loop  common/main.c /******************************************************************** ...

  8. uboot-tiny4412启动流程(下)----如何将自己的裸板测试程序加入uboot中启动测试

    今天在工作上搞了一天高通的芯片uboot程序,目的是希望将一个裸板的程序移植到uboot中,并且开机让它运行.这个芯片是NXP4330,目前是高通的一个芯片,基于ARM-contexA9架构,那么就跟 ...

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

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

随机推荐

  1. LINUX 安装 VMware-Tools(附常见问题)

    mkdir /cdrom #创建光驱挂载目录 mount /dev/cdrom /cdrom # 加载光驱 tar -zxvf /cdro/VMwareTools**********.tar.gz ~ ...

  2. [LeetCode] 849. Maximize Distance to Closest Person 最大化最近人的距离

    In a row of seats, 1 represents a person sitting in that seat, and 0 represents that the seat is emp ...

  3. 2019年Java面试题基础系列228道,题目汇总,可以先看会多少

    Java面试题(一) 1.面向对象的特征有哪些方面? 2.访问修饰符 public,private,protected,以及不写(默认)时的区别? 3.String 是最基本的数据类型吗? 4.flo ...

  4. Andrew Ng机器学习课程10补充

    Andrew Ng机器学习课程10补充 VC dimension 讲到了如果通过最小化训练误差,使用一个具有d个参数的hypothesis class进行学习,为了学习好,一般需要参数d的线性关系个训 ...

  5. JIRA问题状态已关闭,但是解决结果还是未解决

    自己设置的工作流,状态和解决结果是没有关联的,这时候我们要配置关联关系 1.如下,状态时已关闭,但是解决结果是未解决 . 2.解决方法: 2.1设置-问题-工作流,找到目前在使用的工作流,点击编辑 3 ...

  6. 带你一步一步搭建TypeScript环境

    今天继续来更新,本篇文章我们讲环境搭建,主要分享一些环境搭建的学习资源及安装步骤,解决一些安装时可能会出现的问题.下面就让我们一起进入学习第一步,搭建TypeScript环境:一. 环境搭建1.1. ...

  7. Python学习路线图【转载】

    文章来源:https://blog.csdn.net/u014044812/article/details/88079011

  8. 【HC89S003F4开发板】 1环境搭建

    HC89S003F4开发板环境搭建 一.概述 芯圣电子做活动,一个开发板只用一块钱,买过来玩玩.︿( ̄︶ ̄)︿ 全套资料可以在论坛或qq群里下载.总之先安装个环境先. 二.安装Keil C51 作为增 ...

  9. 组件上使用v-model

    组件上使用v-model <input v-model="searchText"> 等价于 <input v-bind:value="searchTex ...

  10. Netty服务端创建流程及组件职责

    public class NettyServer { public static void main(String[] args) throws InterruptedException { NioE ...