很早很早以前就想在嵌入式系统上实现动态模块加载的功能了,期间走了些弯路,直到最近,才完整地在嵌入式系统上实现动态模块加载。
=== 动态模块加载的好处 ===
动态模块加载的好处很多,例如,当你升级一个系统的时候,可以只升级一个模块,而不必升级整个系统。你可以把不同的模块放在不同的介质上,并实施不同等级的保护,例如BIOS部分进行写保护。
有些系统允许用户进行二次开发,这个时候几乎一定是需要动态加载功能的,因为你不希望用户需要链接整个系统才能够进行二次开发,而且你可能希望支持多个用户模块,彼此不相互依赖,彼此不干扰。
=== Background ===
一般来说,C的编译器编译出来的代码,由以下几个重要的部分:
.code: 代码段
.data: 有初值的数据段
.bss: 无初值的数据段
通常还有.rodata,是只读的数据段,在嵌入式系统中经常可以合并到.code段中.
注: .code, .data 和.bss这些段的命名不同的编译器可能会有不同。
由于不同段在实际运行的时候可能会被加载到不同的介质,例如.code和.rodata可以放在NOR FLASH上而.data,.bss放入RAM中,或者要满足所谓的scatter loading,因此编译器会努力使段可以自由移动。
但是要做到这一点,并不容易。
在代码段中运行的指令,要获取数据段中的数据,方法有:
a) 通过当前PC值+偏移量
b) 通过绝对地址
c) 通过中间寄存器,寄存器里面:
c.1) 存放绝对地址
c.2) 偏移量
方法b通常只在CISC中存在,许多RISC机器由于指令长度受限制,并不存在方法b。
因此,从这里可以看出,要做到各段可以自由移动,有几种方法:
1) 保留一个寄存器专门用于指示数据段的起始地址
2) 运行前修改指令
3) 保留一小块数据段和代码段的相对位置不变,此片数据段作为指向实际数据段的入口表, 运行前修改此表。
方法1和方法3通常会结合起来一起用,动态链接库就是用了这种技术。
方发2是一种通用的方法,实际上连接器就是这样生成可执行文件的。
=== ARM AIF ===
ARM公司的编译器有一项特殊的功能,即可以产生一种可自我重定位的可执行文件,即AIF格式。
在AIF文件中,包含了一个AIF头和一小段由编译器产生的重定位代码。运行AIF格式的文件,只需要告诉它起始地址,这段重定位代码就会负责修改余下的一些必要的信息达到重定位目的。目前还没有充分的公开的文档解释AIF内部的详细工作机制。
在我过去的一些项目中,AIF工作的很好,但是运行时外部无法获取AIF文件的更多信息,例如你无法去调用AIF映像中的某一个函数,因为你不知道它的地址。另外,AIF的执行映像中,.data必须紧跟在.code之后,对于想重定位到FLASH中执行的嵌入式系统就行不通了。
=== ELF ===
ELF文件是最常见的目标文件格式,它可能有很多扩展名,例如.o,.so,或者最终的可执行文件也是ELF。
ELF有几种:
* 可重定位
* 可动态链接
* 可执行
* 可执行+可重定位
可执行的ELF如果没有可重定位信息,那就只能靠虚拟内存系统来支持它运行。但是对于许多嵌入式系统,有可能连MMU都不具备,因此我们只关心可重定位的ELF。(可动态链接ELF实际上也是可重定位的一种,附加很多额外信息)
有关ELF的详细信息,请参阅:http://www.skyfree.org/linux/references/ELF_Format.pdf
=== ELF Loader ===
我花了不少时间寻找小型的ELF Loader实现,但是真正适合嵌入式系统的却不多。
+Contiki OS:
在Contiki OS里面,有一个很有趣的ELF Loader实现,嗯,其实Contiki OS有很多有意思的东西
+ucLinux:
ucLinux也是一个很有意思的例子。由于ucLinux没有启用虚拟内存系统,因此它在加载可执行文件的时候,就要进行重定位。为了加速重定位和减小ELF文件的体积,ucLinux提供了特殊的工具链,在产生ELF之前进行部分的“预重定位”,最后ELF中只需要携带很小体积的重定位信息。
+其它RTOS:
其它嵌入式OS,如VxWorks也实现了ELF Loader, eCos的ELF Loader看起来尚未完整。
+Linux Kernel:
哦,差点忘了一个最重要的,Linux Kernel。
Linux Kernel的模块是可以通过insmod动态地加入内核。虽然Linux的用户空间程序运行在虚拟内存中,整个内核的空间确只有一个。一些奉行micro kernel的人批评Linux的这种方式,但是一个单一空间的内核运行效率却是最高的。
在2.6内核中,模块重定位工作不再由insmod来完成,而是由内核来做所有的重定位工作。实现代码在:kernel/module.c中。
剥去那些处理特殊section的代码,Linux内核模块加载部分的代码其实是非常简单明了的,而且Linux支持数十种架构意味着你几乎不要担心架构移植的问题。
=== 结论 ===
在嵌入式系统中实现动态模块加载的技术是成熟的,可靠的,可以借鉴的开发源码的实现例子也有不少。一个参考数据: 我最近在一个嵌入式RTOS上实现的ELF Loader,运行在ARM7 CPU上,从NAND FLASH中加载一个400K左右的ELF,耗时大约0.5秒。
=== THE END ===
分享到:
相关推荐
这涉及到Linux内核空间加载ELF文件的过程,以及在程序运行过程中符号的动态解析。这些过程是在指定的实验平台上进行的,该平台包括了Ubuntu 7.04操作系统、Linux内核版本2.6.20、gcc编译器版本4.1.2、glibc版本2.5、...
动态模块加载的好处有很多,例如,当你升级一个系统的时候,可以只升级一个模块,而不必升级整个系统。你可以把不同的模块放在不同的介质上,并实施不同等级的保护,例如 BIOS 部分进行写保护。有些系统允许用户进行...
动态加载是一种程序执行时才将代码加载到内存的技术,它使得程序可以在运行时根据需要加载不同的模块或库,提高了程序的灵活性和效率。 位置无关代码是一种编程技术,允许代码在内存中的任意位置被加载和执行,这是...
2. **使用loadModuleAt()**:通过指定目标模块加载到特定地址,确保所有必要的函数都在跳转范围内,从而避免重定位错误。 #### 结论 VxWorks中的动态加载应用技术,为嵌入式系统开发提供了极大的灵活性和效率。...
《链接器与加载器》是一份详尽的文档,深入探讨了计算机科学中两个至关重要的概念——链接器(linker)和加载器(loader),这两个组件在软件开发和执行过程中扮演着核心角色。以下是从该文档中提炼出的关键知识点,旨在...
9. **共享对象与插件**:ELF 文件还支持动态加载的共享对象(.so 文件),它们可以在程序运行时被加载和卸载,常用于实现模块化和插件机制。 10. **调试信息**:ELF 文件还可以包含调试信息,如DWARF 格式,这些...
链接器和加载器的协同工作使得我们可以编写模块化的程序,每个模块可以独立编译,然后在链接阶段整合。这样的过程大大提高了开发效率,同时也使得共享库成为可能,允许多个程序共享同一份代码,节省系统资源。 在...
共享库(动态链接库)允许程序在运行时动态加载库函数,这在提高程序模块化的同时也节省了内存资源。 #### 10. 重定位技术 重定位技术是指程序在被加载到内存时,需要根据实际的内存地址重新调整程序中某些引用(如...
串口加载简单且易于实现,支持ELF32和S-record等格式的程序文件,但在汽车应用中,由于其速度限制,可能导致下载速度慢和调试维护不便。因此,研究如何优化CAN总线的Bootloader加载机制,提高数据传输速度和稳定性,...
EFI不仅提供了更加丰富的启动选项和服务,而且其模块化的设计也使得系统的启动变得更加灵活和高效。在这个背景下,开发针对EFI环境的应用程序变得尤为重要。 **为什么要在Linux环境下开发EFI应用程序?** - **历史...
其中,串口加载简单便捷,支持ELF32和S-record等格式的程序,但速度较慢,不适用于快速的整车控制器调试和维护。 总结来说,基于MPC5605的Bootloader软件CAN模块的研究聚焦于如何优化嵌入式系统的启动过程,提升...
在Android平台上,so(共享对象)文件的加载是一个涉及到Java层和Native层的复杂过程。本文将深入探讨Android中so文件的加载机制,分析从so文件被载入到最终链接的全部过程。 首先,Android在Java层提供了加载so...
6. 动态加载和插件系统:高级的加载器支持在程序运行时动态加载模块(如DLL或动态加载库),这在实现插件系统或代码热更新时非常有用。 7. 装载异常处理:加载器还需要处理各种异常情况,比如找不到依赖的库文件、...