为什么编译器版本出现在我的ELF可执行文件中?

时间:2011-06-07 09:44:36

标签: c linux gcc elf

我最近使用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

他们真的需要吗?无法减少可执行文件的大小?

6 个答案:

答案 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文件不考虑堆栈使用情况 - 基本上所有自动变量都是未计入的。这些仅仅是其他三个例子。