1. 概述

1.1 基本概念

分散加载是一种实现特定代码快速启动的技术,通过优先加载特定代码到内存,达到缩短从系统开机到特定代码执行的时间。可被应用来实现关键业务的快速启动。

嵌入式系统通过uboot加载flash上的镜像文件到内存并执行,而镜像文件本身可能较大,由于flash读取速度的限制,将镜像全部加载完再执行可能无法满足时间敏感的业务对启动速度的要求。

分散加载的思想是先加载部分镜像并执行,这部分镜像包含了时间敏感的关键业务,从而达到快速启动关键业务的效果。

Huawei LiteOS的分散加载

Huawei LiteOS的分散加载分为两个阶段,第一阶段通过uboot将关键业务部分镜像加载到内存并执行,待这部分业务得到执行后,第二阶段在代码中加载剩余部分镜像到内存继续执分散加载的内部原理图如图2所示,图中的运作顺序可参照图1的流程说明。通过合理布局镜像,第一阶段加载部分镜像的速度会比加载完整镜像快,从而缩短系统启动到关键业务运行的时间。

在IPC Huawei LiteOS版本上,通过应用分散加载技术,实现了1s内从开机启动到录制,超越Linux版本的3s-4.5s。

1.2 运作机制

分散加载的主体思想是将部分时间敏感的业务提前加载执行,具体手段是将与这些业务相关的数据、代码段布局到镜像文件的前端,第一阶段只加载前端这段镜像,达到最短时间内即可运行时间敏感业务的开发指导目的。

在这些业务得到执行之后,第一阶段的代码中调用分散加载接口加载剩余部分镜像,接着运行镜像剩余部分的业务。

分散加载的内部原理图如图2所示,图中的运作顺序可参照图1的流程说明。

分散加载在关键业务第一时间被加载执行之后,再加载非关键业务。

2. 开发指导

2.1 使用场景

分散加载技术应用的典型场景是快速启动对时间敏感的业务。

嵌入式系统中可能存在某些业务对启动时间要求比较高,譬如Huawei LiteOS IPC项目上对从开机到录制预览的时间要求较高,可以利用分散加载技术实现录制预览业务的快速启动。

2.2 功能

Huawei LiteOS系统中的分散加载模块为用户提供如下接口。

功能分类 接口名 描述
分散加载接口 LOS_ScatterLoad 在分散加载阶段的最后调用此接口,从镜像加载剩余非紧急业务

2.3 开发流程

分散加载流程图如下所示。

步骤1 调用接口LOS_ScatterLoad,编写分散加载业务代码

业务代码入口为函数app_init,该函数位于os_adapt.c。在紧急业务代码后调用LOS_ScatterLoad函数进行分散加载,并用#ifndef MAKE_SCATTER_IMAGE、 #endif将该函数后的非紧急业务包围起来,用以编译紧急镜像和全部镜像时作区分,示例代码如下:

void app_init() {
proc_fs_init();
hi_uartdev_init();
system_console_init("/dev/uartdev-0");
LOS_CppSystemInit((unsigned long)&__init_array_start__, (unsigned long)&__init_array_end__,
BEFORE_SCATTER);
LOS_ScatterLoad(0x100000, flash_read, NAND_READ_ALIGN_SIZE);
#ifndef MAKE_SCATTER_IMAGE /* 以下为非紧急业务 */
LOS_CppSystemInit((unsigned long)&__init_array_start__, (unsigned long)&__init_array_end__,
AFTER_SCATTER);
extern unsigned int osShellInit(void);
osShellInit();
rdk_fs_init();
SDK_init();
hi_product_driver_init();
char *apszArgv[3]={"vs_server","./higv.bin","-i"};
vs_server(3, apszArgv);
#endif /* MAKE_SCATTER_IMAGE */
}

os_adapt.c位于Huawei_LiteOS代码包的platform/bsp/hi3516a/os_adapt路径下。

步骤2 配置SCATTER_SRC变量

在根目录下Makefile中配置SCATTER_SRC,将变量定义为调用分散加载函数的业务源文件路径,如下所示,其中LITEOSTOPDIR指代Huawei_LiteOS代码根目录。

SCATTER_SRC := $(LITEOSTOPDIR)/platform/bsp/$(LITEOS_PLATFORM)/os_adapt/os_adapt.c

步骤3 执行make scatter,编译紧急部分镜像

在根目录下执行如下命令,则不会编译#ifndef MAKE_SCATTER_IMAGE以下的业务代码。编译系统将自动调用工具链抽取分散加载最小镜像的符号表并根据该符号表提取分散加载最小镜像的.a库列表。

Huawei_LiteOS$ make scatter

步骤4 执行make,编译全部镜像

  • 在根目录下执行如下命令,则编译全部业务代码。
Huawei_LiteOS$ make

编译后,命令行界面会返回紧急镜像大小信息,如下图所示。

  • 编译完成后,检查镜像段的排布,如果镜像中生成了分散加载相关的段则表明分散加载的镜像生成成功。进入系统镜像生成目录(hi3516a平台的镜像生成目录为out/hi3516a,其他类推),可以看到生成的系统镜像vs_server文件,执行命令readelf -S vs_server打开该文件,结果如下图所示。显示了与分散加载相关的段信息(包括段的名称、起始地址及偏移大小)。其中.fast_rodata为分散加载镜像的只读数据段, .fast_text为代码段, .fast_data为数据段

查看分散加载链接脚本.text段,新增了scatter.o(.text),如下图所示,实现了将分散加载的快速启动部分代码相关符号归拢到一个同一个段中。

分散加载链接脚本路径:Huawei_LiteOS/tools/scripts/ld/scatter.ld

步骤5 执行tftp 0x82000000 vs_server.bin;nand erase 0x100000 0x700000;nand write 0x82000000 0x100000 0x700000;,将全部镜像烧写到Flash

进入串口工具界面,输入如下命令,将全部镜像烧写到Flash的0x100000地址位。

tftp 0x82000000 vs_server.bin;nand erase 0x100000 0x700000;nand write 0x82000000 0x100000 0x700000;

其中, vs_server.bin为系统镜像文件名,先将其烧写到内存中一段高地址位0x82000000。然后烧写到Flash,起始地址为0x100000,烧写长度为0x700000,即烧写的镜像文件大小不能超过7M,跟据实际镜像大小调整数值。

步骤6 执行nand read 0x80008000 0x100000 0x4E0000; go 0x80008000;,加载紧急业务

执行如下命令,从Flash的0x100000地址处读取长度为0x4E0000的镜像,加载紧急业务到0x80008000。

nand read 0x80008000 0x100000 0x4E0000; go 0x80008000;

步骤7 系统自动重启

系统自动重启,在0x80008000地址处加载镜像。

3. 注意事项

  • 分散加载第一阶段拷贝过少或者拷贝偏移地址没有根据存储介质的差异进行对齐都会导致系统异常,因此使用时要按照编译最后给出的大小进行uboot加载镜像。
  • 用户需保证提取的库文件列表是支持关键业务运行的超集,否则会导致分散加载第一阶段中的代码访问到第二阶段中的代码或数据,从而导致系统异常。
  • 分散加载使用中可能存在这样一种场景:一个变量在第一阶段中运行后值被修改,但是在第二阶段加载运行之后,该变量值又成为一个未初始化的值。这种场景的原因是该变量在第一阶段中使用到,但是并没有被归拢到第一阶段中,所以在第一阶段修改之后,第二阶段加载进内存后该变量值又被覆盖成未初始化的值。解决的方法是将该变量归拢到第一阶段中,确保第一阶段使用到的数据都在快速启动段中

4. 常见问题汇总

本节介绍使用分散加载技术遇到的主要问题和解决方法。

  • 缺少.O文件
arm-hisiv300-linux-ld: cannot find libscatter.O
make: *** [vs_server] Error 1

这个问题出现的原因是修改了链接脚本后,没有对应生成.O文件,解决的方法是生成对应的.O文件并且放到目标目录下

  • 符号未定义
/usr1/xxxxx/gerrit_code/modify-debug/liteos_ipc/out/lib/libar6003.a(ar6000_drv.o): In
function `ar6000_avail_ev':
/usr1/xxxxx/gerrit_code/modify-debug/liteos_ipc/vendor/ar6k3_wifi/AR6003/host/qca/source/
ar6000_drv.c:1553: undefined reference to `wireless_init_event'
/usr1/xxxxx/gerrit_code/modify-debug/liteos_ipc/out/lib/libar6003.a(drv_config.o): In
function `ar6000_tkip_micerr_event':
/usr1/xxxxx/gerrit_code/modify-debug/liteos_ipc/vendor/ar6k3_wifi/AR6003/host/qca/source/
drv_config.c:1856: undefined reference to `wireless_send_event'
make: *** [vs_server] Error 1

这个问题的出现是比较常见的,可能是裁剪过程中在修改链接脚本的时候,将一些必要的.a文件也删除了,这时需要用grep指令在out/lib目录下搜索未定义的变量,找出都存在于哪些.a文件中,将未添加的.a文件添加到链接脚本中。

  • 分散加载进指令异常。

    通过查看系统异常时pc的位置是否超出分散加载第一阶段的范围,如果是则应该是第一阶段库文件列表涵盖不全,导致有符号未被归拢到第一阶段的代码、数据段中,需要结合系统镜像反汇编文件定位到异常pc所在函数名,找到该函数定义所在的库,将该库添加到库列表中。

liteos分散加载(十四)的更多相关文章

  1. liteos动态加载(十三)

    1. 概述 1.1 基本概念 动态加载是一种程序加载技术. 静态链接是在链接阶段将程序各模块文件链接成一个完整的可执行文件,运行时作为整体一次性加载进内存.动态加载允许用户将程序各模块编译成独立的文件 ...

  2. XAML加载的四种方式

    XAML加载与编译可以分为四种: 仅使用代码进行WPF程序的生成 使用代码和未编译的标记 使用代码和编译过的BAML 1.只是用代码进行窗体的生成:优点是可以随意定制应用程序,缺点是没有可视化编辑窗口 ...

  3. ExtJs基础知识总结:自定义弹窗和ComboBox自动联想加载(四)

    概述 Extjs弹窗可以分为消息弹窗.对话框,这些弹窗的方式ExtJs自带的Ext.Msg.alert就已经可以满足简单消息提示,但是相对复杂的提示,比如如何将Ext.grid.Panel的控件显示嵌 ...

  4. 深入java虚拟机学习 -- 类的加载机制(四)

    类加载的命名空间 每个类加载器都有自己的命名空间,命名空间由所有以此加载器为初始类加载器的类组成,不同命名空间的两个类是不可见的,但只要得到类所对应的Class对象的refrence(反射),还是可以 ...

  5. Storyboard中ViewController加载的四种方式

    这个总结来自于<Programming iOS 10>一书: 1.storyboard的初始化ViewController,通过方法instantiateInitialViewContro ...

  6. mybatis源码解析之Configuration加载(四)

    概述 上一篇文章,我们主要讲了datasource的相关内容,那么<environments>标签下的内容就看的差不多了,今天就来看一下在拿到transationManager和datas ...

  7. 基于FBX SDK的FBX模型解析与加载 -(四)

    8. 骨骼蒙皮动画 骨骼蒙皮动画是当前游戏引擎中最常用的一种动画方式,关于其基本原理网络上的资料较多,关于到涉及的其它较复杂操作,如插值.融合等在这里也就先不再讨论了,而且其实现方式也与具体引擎的动作 ...

  8. ERP存储过程的调用和树形菜单的加载(四)

    引用:DAL:System.Data.SqlClient;System.Data; namespace CommTool { public class SqlComm { /// <summar ...

  9. Keil sct分散加载文件

    官方说明:http://www.keil.com/support/man/docs/armlink/armlink_pge1401393372646.htm

随机推荐

  1. Linux find 用法和参数

    Linux中find常见用法 ·find   path   -option   [   -print ]   [ -exec   -ok   command ]   {} \; find命令的参数: ...

  2. TypeScript 学习笔记(四)

    函数: 1.函数是一组一起执行一个任务的语句 2.我们可以把一段可复用的代码放到一起组成函数,从而提高效率 3.函数声明(通过关键字 function 来声明)告诉编译器函数的名称.返回类型和参数 4 ...

  3. 如何使用压缩的方式将Windows下的zip压缩包上传到Linux系统

      我们可以使用在Windows下压缩文件夹,然后到Linux系统下解压缩的方式,完成整个上传工作. 第一步:在Windows系统下,将整个文件夹压缩成zip后缀的压缩包 方法一:

  4. How to recover a skipped tablespace after an incomplete recovery? (Doc ID 1561645.1)

    How to recover a skipped tablespace after an incomplete recovery? (Doc ID 1561645.1) APPLIES TO: Ora ...

  5. linux下关闭selinux

    找到 /etc/sysconfig/selinux 文件 修改 SELINUX=enable 使之 SELINUX=disable 重启 reboot

  6. YUM命令总结

    1.关于YUM源 Yum 全称为 Yellow dog Updater Modified,它是一个在线的软件安装命令. 能够从指定的服务器自动下载RPM包并且安装,可以自动处理依赖性关系,并且一次安装 ...

  7. 【转载】C++编译过程

    C++编译过程 C++ 编译过程在介绍编译器之前,先简单地说一下 C++ 的编译过程,以便理解编译器的工作.编译(compiling)并不意味着只创建仅仅一个可执行文件.创建一个可执行文件是一个多级过 ...

  8. RPM常用命令总结

    安装 rpm -ivh package_name (package_name指的是RPM包的文件名) 查询 1.查询是否安装,及安装版本 rpm -q 已安装的软件名(ex:rpm -q docker ...

  9. Java学习笔记(6)---正则表达式,方法

    1.正则表达式: a.定义: 正则表达式定义了字符串的模式. 正则表达式可以用来搜索.编辑或处理文本. 正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别. 在Java,一个字符串其实就是一 ...

  10. python简单处理excel方法

    python自带xlrd和xlwt模块用来处理excel,但总觉得xlwt模块用着别扭,于是按自己的习惯重新封装了一个 # coding=utf- import xlrd # 读模块 import x ...