答网友问:嵌入式Linux执行程序提示Not found的解答

若如初见. 提交于 2020-02-26 18:06:05

某日,网友不会飞的骆驼问了我一个问题。在嵌入式Linux系统中执行./a.out时,提示找不到,信息如下:

$ ./a.out 
-sh: ./a.out: not found
1
2
找了点资料,帮解决了。

问题重现
我在板子上重现了问题。的确如上所述。
使用file命令查看a.out属性,信息如下:

file ./a.out 
./a.out: ELF 32-bit LSB executable, ARM, EABI5 version 1 (SYSV), dynamically linked, interpreter /lib/ld-linux-armhf.so.3, for GNU/Linux 2.6.32, BuildID[sha1]=89b9b01541d4f84bfba73ce649cdd2982bb3840e, stripped
1
2
从信息上看,a.out是32位ARM系统程序,动态链接,解析器为/lib/ld-linux-armhf.so.3。但是,在板子上查看该文件:

$ ls /lib/ld-linux*
/lib/ld-linux.so.3
1
2
可以看到只有ld-linux.so.3,找不到前面提到的ld-linux-armhf.so.3,所以执行程序时提示not found。
造成这个问题的原因是:板子上的系统所用的链接器版本,与编译程序的交叉编译器的链接器版本不一致。先试试将交叉编译器的链接器拷贝到板子系统中,再次执行:

$ ./a.out 
./a.out: /lib/libstdc++.so.6: version `GLIBCXX_3.4.21' not found (required by ./a.out)
1
2
提示信息变化了,但还是出错,是因为c++库的版本号不对应。即a.out使用的版本是GLIBCXX_3.4.21,但系统上的c++动态库没有这个版本号。在板子上查找这个库,信息如下:

$ ll  /lib/libstdc\+\+.so.6*
lrwxrwxrwx    1 root     root            19 Jan 21  2017 /lib/libstdc++.so.6 -> libstdc++.so.6.0.16
-rwxr-xr-x    1 root     root       1044441 Oct 15 09:57 /lib/libstdc++.so.6.0.16
1
2
3
从中得到,c++库版本为6.0.16(姑且这么认为)。
再在交叉编译器目录中查找,已精简的信息如下:

$ ll ./sysroots/usr/lib/libstdc++*
./sysroots/usr/lib/libstdc++.so -> libstdc++.so.6.0.21*
lrwxrwxrwx 1 latelee latelee      19 Jan 14 08:23 ./sysroots/usr/lib/libstdc++.so.6 -> libstdc++.so.6.0.21*
-rwxr-xr-x 1 latelee latelee 1240644 Jun 20  2018 ./sysroots/usr/lib/libstdc++.so.6.0.21*
1
2
3
4
可以看到,c++库的版本为6.0.21。这样,更能印证版本不匹配的问题了。

解决办法
确认清楚板子系统使用哪个版本的编译器,再使用这个版本编译程序源码即可。在实践中,u-boot、kernel、busybox以及应用层,必须统一使用一套相同的编译器,否则,会出现各种问题。
(注:也不一定要必须,但是,统一一个版本的话,将减少很多不必要的麻烦)

其它扩展
之前曾经也遇到过此类问题,写在此,作为一个扩展知识点。查看交叉编译器版本,提示信息如下:

latelee@ubuntu:~$ arm-linux-gcc --version
-bash: /home/latelee/bin/gcc-4.6.2-glibc-2.13-linaro-multilib-2011.12/fsl-linaro-toolchain/bin/arm-linux-gcc: No such file or directory
1
2
用file命令查看文件属性如下:

/home/latelee/bin/gcc-4.6.2-glibc-2.13-linaro-multilib-2011.12/fsl-linaro-toolchain/bin/arm-fsl-linux-gnueabi-gcc: ELF 32-bit LSB executable, Intel 80386, version 1 (SYSV), dynamically linked, interpreter /lib/ld-linux.so.2, for GNU/Linux 2.6.0, stripped
1
可以看到解析器为/lib/ld-linux.so.2,但这个文件并不存在,所以无法找到。
这个问题很典型,是因为在64位系统中执行32位程序造成的(除交叉编译器外,其它程序亦然)。如何解决?安装32位库即可,命令如下:

sudo apt-get install -y lib32ncurses5 lib32z1
1
安装完毕后,系统上有/lib/ld-linux.so.2文件。重新执行前面的命令即可显示正常的版本号。
————————————————
版权声明:本文为CSDN博主「李迟」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/subfate/article/details/86490887

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!