动态库大小比静态库大和链接对象大小的总和,怎么来的?

时间:2015-03-12 17:14:09

标签: opencv gcc android-ndk shared-libraries static-libraries

[请参阅编辑,似乎额外的大小来自于在链接时添加的调试符号,但之所以发生这种情况仍然不清楚!]

我正在交叉编译OpenCV 2.4.11 Ubuntu x86 64bit - > armeabi。

我正在使用https://developer.android.com/tools/sdk/ndk/index.html中提供的工具链,选择4.9编译器。

当我编译动态库时,它们比静态库大得多更大。例子:

 3793082 Mar 12 17:21 libopencv_core.a
 6131716 Mar 12 17:29 libopencv_core.so
  446060 Mar 12 17:22 libopencv_highgui.a
 5510352 Mar 12 17:30 libopencv_highgui.so
 3477794 Mar 12 17:21 libopencv_imgproc.a
 5325504 Mar 12 17:29 libopencv_imgproc.so
   38004 Mar 12 17:19 libopencv_info.so
  844990 Mar 12 17:21 libopencv_ml.a
 3827136 Mar 12 17:29 libopencv_ml.so
  747744 Mar 12 17:22 libopencv_objdetect.a
 2370188 Mar 12 17:30 libopencv_objdetect.so
  405920 Mar 12 17:22 libopencv_video.a
 2196268 Mar 12 17:30 libopencv_video.so

对于静态库,大小或多或少与对象文件的总大小相对应。 core和highgui的示例。

du -chs `find -iname \*.o|grep opencv_core.dir`
[...]
3,5M    total

du -chs `find -iname \*.o|grep opencv_highgui.dir`
[...]
352K    total

如果我使用make或ninja构建,也会发生同样的情况。

编译器标志在构建时只有一点点差别,但如果我检查为静态和动态构建生成的目标文件,它们的大小完全相同。这是我用来生成这样一个列表的命令:

ls -s `find -iname \*.o`|grep core

所以,我想,它必须在链接阶段。我看了一下build.ninja文件的差异,这些是仅针对共享版本的行:

LINK_FLAGS = -llog -Wl,--fix-cortex-a8 -Wl,--no-undefined -Wl,--gc-sections -Wl,-z,noexecstack -Wl,-z,relro -Wl,-z,now
LINK_LIBRARIES = lib/libopencv_features2d.so -ldl -lm -llog -ltbb lib/libopencv_flann.so lib/libopencv_highgui.so lib/libopencv_imgproc.so lib/libopencv_core.so

我认为链接的附加库(dl,m,log,tbb)不会影响最终大小,因为它们都比我发现的差异小得多。此外,我开始验证和登录只有.so可用,而对于tbb(100Kb)我有共享和静态版本。顺便说一句,我试图建立没有tbb。

为了100%确定,我采用了链接目标文件的实际命令行,删除了-no-undefined option,然后删除了所有其他选项和链接库。文件大小没有变化,除了删除导致文件大小增加的-Wl,--gc-sections (它是一些垃圾收集选项)。

那么,剩下的唯一选择是...链接器错误?!?有没有人知道发生了什么?

其他一些信息:

编译器详细信息:

 ./arm-linux-androideabi-gcc -v
Using built-in specs.
COLLECT_GCC=./arm-linux-androideabi-gcc
COLLECT_LTO_WRAPPER=/opt/toolchain-arm17/bin/../libexec/gcc/arm-linux-androideabi/4.9/lto-wrapper
Target: arm-linux-androideabi
Configured with: /s/ndk-toolchain/src/build/../gcc/gcc-4.9/configure --prefix=/tmp/ndk-andrewhsieh/build/toolchain/prefix --target=arm-linux-androideabi --host=x86_64-linux-gnu --build=x86_64-linux-gnu --with-gnu-as --with-gnu-ld --enable-languages=c,c++ --with-gmp=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-mpfr=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-mpc=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-cloog=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-isl=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-ppl=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --disable-ppl-version-check --disable-cloog-version-check --disable-isl-version-check --enable-cloog-backend=isl --with-host-libstdcxx='-static-libgcc -Wl,-Bstatic,-lstdc++,-Bdynamic -lm' --disable-libssp --enable-threads --disable-nls --disable-libmudflap --disable-libgomp --disable-libstdc__-v3 --disable-sjlj-exceptions --disable-shared --disable-tls --disable-libitm --with-float=soft --with-fpu=vfp --with-arch=armv5te --enable-target-optspace --enable-initfini-array --disable-nls --prefix=/tmp/ndk-andrewhsieh/build/toolchain/prefix --with-sysroot=/tmp/ndk-andrewhsieh/build/toolchain/prefix/sysroot --with-binutils-version=2.24 --with-mpfr-version=3.1.1 --with-mpc-version=1.0.1 --with-gmp-version=5.0.5 --with-gcc-version=4.9 --with-gdb-version=7.6 --with-python=/usr/local/google/home/andrewhsieh/mydroid/ndk/prebuilt/linux-x86_64/bin/python-config.sh --with-gxx-include-dir=/tmp/ndk-andrewhsieh/build/toolchain/prefix/include/c++/4.9 --with-bugurl=http://source.android.com/source/report-bugs.html --enable-languages=c,c++ --disable-bootstrap --enable-plugins --enable-libgomp --disable-libsanitizer --enable-gold --enable-graphite=yes --with-cloog-version=0.18.0 --with-isl-version=0.11.1 --enable-eh-frame-hdr-for-static --with-arch=armv5te --program-transform-name='s&^&arm-linux-androideabi-&' --enable-gold=default
Thread model: posix
gcc version 4.9 20140827 (prerelease) (GCC) 

我还试图看看尝试另一个版本的链接器会发生什么,但是没有大小的改变

arm-linux-androideabi-g++.exe -v
Using built-in specs.
COLLECT_GCC=arm-linux-androideabi-g++.exe
COLLECT_LTO_WRAPPER=lto-wrapper.exe
Target: arm-linux-androideabi
Configured with: /s/ndk-toolchain/src/build/../gcc/gcc-4.8/configure --prefix=/tmp/ndk-andrewhsieh/build/toolchain/prefix --target=arm-linux-androideabi --host=x86_64-pc-mingw32msvc --build=x86_64-lin
ux-gnu --with-gnu-as --with-gnu-ld --enable-languages=c,c++ --with-gmp=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-mpfr=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-mpc=/tmp/n
dk-andrewhsieh/build/toolchain/temp-install --with-cloog=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-isl=/tmp/ndk-andrewhsieh/build/toolchain/temp-install --with-ppl=/tmp/ndk-andrewhsieh/
build/toolchain/temp-install --disable-ppl-version-check --disable-cloog-version-check --disable-isl-version-check --enable-cloog-backend=isl --with-host-libstdcxx='-static-libgcc -Wl,-Bstatic,-lstdc+
+,-Bdynamic -lm' --disable-libssp --enable-threads --disable-nls --disable-libmudflap --disable-libgomp --disable-libstdc__-v3 --disable-sjlj-exceptions --disable-shared --disable-tls --disable-libitm
 --with-float=soft --with-fpu=vfp --with-arch=armv5te --enable-target-optspace --enable-initfini-array --disable-nls --prefix=/tmp/ndk-andrewhsieh/build/toolchain/prefix --with-sysroot=/tmp/ndk-andrew
hsieh/build/toolchain/prefix/sysroot --with-binutils-version=2.24 --with-mpfr-version=3.1.1 --with-mpc-version=1.0.1 --with-gmp-version=5.0.5 --with-gcc-version=4.8 --with-gdb-version=7.6 --with-pytho
n=/usr/local/google/home/andrewhsieh/mydroid/ndk/prebuilt/windows-x86_64/bin/python-config.sh --with-gxx-include-dir=/tmp/ndk-andrewhsieh/build/toolchain/prefix/include/c++/4.8 --with-bugurl=http://so
urce.android.com/source/report-bugs.html --enable-languages=c,c++ --disable-bootstrap --enable-plugins --enable-libgomp --disable-libsanitizer --enable-gold --enable-graphite=yes --with-cloog-version=
0.18.0 --with-isl-version=0.11.1 --enable-eh-frame-hdr-for-static --with-arch=armv5te --program-transform-name='s&^&arm-linux-androideabi-&' --enable-gold=default
Thread model: posix
gcc version 4.8 (GCC)

修改
正如MarcB所建议的,我试图strip图书馆。结果令人惊讶(对我来说:))

$ arm-linux-androideabi/bin/strip -g libopencv_core.so -o libopencv_core_stripped.so
$ ls -la *core*
6293308 Mar 13 10:40 libopencv_core.so
3224840 Mar 13 12:18 libopencv_core_stripped.so

如果没有-g(甚至是-g0)编译的目标文件,那么所有这些调试符号都出现了?

注意:像这样剥离的库似乎功能齐全。剥离/未剥离库的nm -D输出是相同的,nm输出只有几行(比12000中的50行少)。

为了确定,我还尝试在链接之前删除对象,但是他们的文件没有改变(增加一点点),并且链接"剥离"目标文件生成一个与之前大小相同的库。

1 个答案:

答案 0 :(得分:2)

那些不是 debug 符号。它们是常规的链接符号。

共享库可能有两组符号:一组用于链接,另一组用于动态加载。 strip删除第一种符号。 您无法链接剥离的共享库,但您可以在运行时正常加载它(例如,如果您使用dlopen,或者链接到库然后将其删除)。

在运行nm yourlib.so之前和之后,请参阅nm -D yourlib.sostrip

更正可以链接剥离的库。关于这两种符号表的一个很好的解释是here