我最近使用gcc:
在Debian Linux下编译了一个简单的hello world C程序gcc -mtune=native -march=native -m32 -s -Wunused -O2 -o hello hello.c
文件大小为2980字节。我在十六进制编辑器中打开它,我看到以下几行:
GCC: (Debian 4.4.5-8) 4.4.5 GCC: (Debian 4.4.5-10) 4.4.5 .shstrtab .interp .note.ABI-tag .note.gnu.build-id .gnu.hash .dynsym .dynstr .gnu.version .gnu.version_r .rel.dyn .rel.plt .init .text .fini .rodata .eh_frame .ctors .dtors .jcr .dynamic .got .got.plt data.data .bss .comment
他们真的需要吗?无法减少可执行文件的大小?
答案 0 :(得分:9)
这是在ELF二进制文件的评论部分。你可以把它剥离出来:
$ gcc -m32 -O2 -s -o t t.c
$ ls -l t
-rwxr-xr-x 1 me users 5488 Jun 7 11:58 t
$ readelf -p .comment t
String dump of section '.comment':
[ 0] GCC: (Gentoo 4.5.1-r1 p1.4, pie-0.4.5) 4.5.1
[ 2d] GCC: (Gentoo 4.5.2 p1.1, pie-0.4.5) 4.5.2
$ strip -R .comment t
$ readelf -p .comment t
readelf: Warning: Section '.comment' was not dumped because it does not exist!
$ ls -l t
-rwxr-xr-x 1 me users 5352 Jun 7 11:58 t
但收益微不足道,不确定是否值得。
答案 1 :(得分:9)
使用-Qn来避免这种情况。
aa$ touch hello.c
aa$ gcc -c hello.c
aa$ objdump -s hello.o
hello.o: file format elf32-i386
Contents of section .comment:
0000 00474343 3a202844 65626961 6e20342e .GCC: (Debian 4.
0010 372e322d 35292034 2e372e32 00 7.2-5) 4.7.2.
aa$ gcc -Qn -c hello.c
aa$ objdump -s hello.o
hello.o: file format elf32-i386
aa$ gcc -v
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/i486-linux-gnu/4.7/lto-wrapper
Target: i486-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Debian 4.7.2-5' --with-bugurl=file:///usr/share/doc/gcc-4.7/README.Bugs --enable-languages=c,c++,go,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.7 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.7 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --enable-plugin --enable-objc-gc --enable-targets=all --with-arch-32=i586 --with-tune=generic --enable-checking=release --build=i486-linux-gnu --host=i486-linux-gnu --target=i486-linux-gnu
Thread model: posix
gcc version 4.7.2 (Debian 4.7.2-5)
aa$
答案 2 :(得分:5)
这是在一个没有加载到内存中的注释部分(并注意ELF文件通常使用填充,以便内存映射它们将保持正确的对齐)。如果你想摆脱这些不需要的部分,请参阅各种objcopy选项并找出:
objcopy --remove-section .comment a.o b.o
答案 3 :(得分:3)
我自己也有同样的问题,但使用MinGW的GCC实现 - 剥离可执行文件并传递-Qn选项什么都没做,我无法删除" .comment"因为没有一部分。
为了阻止包含此信息的编译器,无论可执行文件中包含哪些部分,都可以将 -fno-ident 参数传递给编译器和链接器:
没有参数(strings -a [filename]):
!This program cannot be run in DOS mode.
.text
0`.rdata
0@.idata
GCC: (tdm64-2) 4.8.1
使用参数:
!This program cannot be run in DOS mode.
.text
0`.idata
答案 4 :(得分:2)
如果你不想要它,你似乎可以'只是'剥离它;请参阅此页面以了解详细信息。
http://timelessname.com/elfbin/
请注意,页面(当然)也会使用程序集,您可能不想这样做,但是应用了一般要点
答案 5 :(得分:1)
您可以使用链接描述文件通知加载器哪些部分包含在您的输出中。您可以使用objdump命令查看文件中包含的部分。你已经注意到精灵中存在一些“垃圾” - 垃圾直到你希望你拥有它。
但请注意,elf可执行文件的大小并不表示在内存中实现的图像内存占用。很多“垃圾”不在内存映像中,图像可以调用sbreak和/或mmap来获取更多内存,elf文件不考虑堆栈使用情况 - 基本上所有自动变量都是未计入的。这些仅仅是其他三个例子。