好的,这里有一个更长的解释,但要小心。我仍然强烈建议您设置一个 chrooted 环境以匹配嵌入式设备上可用的环境,并在构建过程的最后阶段使用它。
您应该了解动态链接的 ELF 可执行文件是如何加载和执行的。有一种称为运行时链接编辑器(RTLD) 的东西,也称为动态链接器,它负责加载所有必要的动态链接库、修复重定位等。动态链接器的名称/lib/ld-linux.so.2在 32 位 Linux 系统上
glibc2与. 动态链接器与库的耦合非常紧密,通常只能处理该库的匹配版本。它的路径也被链接器硬编码到可执行文件中(通常/lib64/ld-linux-x86-64.so.2glibc2glibc2ld,由编译器隐式调用以进行链接)。您只需执行以下操作即可轻松检查最后一条语句的有效性ldd some_elf_executable- 运行时链接编辑器显示完整路径:
$ ldd some_elf_executable
linux-vdso.so.1 => (0x00007fffab59e000)
libm.so.6 => /lib64/libm.so.6 (0x0000003648400000)
libc.so.6 => /lib64/libc.so.6 (0x0000003648800000)
/lib64/ld-linux-x86-64.so.2 (0x0000003648000000) <--- the RTLD
为了生成一个动态链接的可执行文件,它使用glibc2与系统上安装的版本不同的版本,可执行文件将在其中运行,您应该使用以下一组选项将您的代码链接到ld:
-rpath=/path/to/newer/libs- 这个指示动态链接器/path/to/newer/libs在尝试解析库依赖项时首先搜索。应该与您在嵌入式设备上/path/to/newer/libs复制较新版本的路径相匹配glibc2
-rpath-link=/path/to/newer/libs- 此选项指示链接器(不是动态链接器)/path/to/newer/libs在链接期间解决共享库之间的依赖关系时使用 - 在您的情况下通常不需要
--dynamic-linker=/path/to/newer/libs/ld-linux.so.2- 这个覆盖了嵌入到可执行文件中的 RTLD 的路径
提供这些选项的方式ld通常是通过-WlGCC 的选项。
-rpath=/path/to/newer/libs
变成:
-Wl,-rpath,/path/to/newer/libs
(注意=替换为,)
--dynamic-linker=/path/to/newer/libs/ld-linux.so.2
变成:
-Wl,--dynamic-linker,/path/to/newer/libs/ld-linux.so.2
/lib/ld-linux.so.2您应该从开发系统复制到/path/to/newer/libs/嵌入式设备上。您还应该复制libc.so.6数学库libm.so.6和可执行文件使用的或可能间接加载的所有其他库。请注意,libc.so.6和libm.so.6实际上是指向真实库的符号链接,其名称类似于libc-2.<version>.so. 您应该复制这些库文件并创建适当的符号链接以使每个人都满意。