如何用 ld 将 C++ 目标文件链接起来

47
我正在尝试使用ld而不是g++链接C ++的输出。 我这样做只是为了学习如何做,而不是为了实际目的,请不要建议仅使用g ++来完成它。
查看此问题,当该人运行ld命令时会得到相同的错误。
$ ld test.o -o test.out
ld: warning: cannot find entry symbol _start; defaulting to 00000000004000e8
test.o: In function `main':
test.cpp:(.text+0x1c): undefined reference to `strcasecmp'
test.cpp:(.text+0x23): undefined reference to `std::cout'
test.cpp:(.text+0x28): undefined reference to `std::ostream::operator<<(int)'
test.cpp:(.text+0x2d): undefined reference to `std::basic_ostream<char, std::char_traits<char> >& std::endl<char, std::char_traits<char> >(std::basic_ostream<char, std::char_traits<char> >&)'
test.cpp:(.text+0x35): undefined reference to `std::ostream::operator<<(std::ostream& (*)(std::ostream&))'
test.o: In function `__static_initialization_and_destruction_0(int, int)':
test.cpp:(.text+0x75): undefined reference to `std::ios_base::Init::Init()'
test.cpp:(.text+0x7a): undefined reference to `__dso_handle'
test.cpp:(.text+0x84): undefined reference to `std::ios_base::Init::~Init()'
test.cpp:(.text+0x89): undefined reference to `__cxa_atexit'
ld: test.out: hidden symbol `__dso_handle' isn't defined
ld: final link failed: Bad value

链接帖子中的答案表明将C++库作为链接器参数添加将解决问题,因此我尝试了。
ld test.o -o test.out -llibstd++

他们建议使用的是这个,我也尝试了很多其他的库名称,比如libstdc++或者stdc++。但总是会出现一个看起来像这样的错误:

ld: cannot find -llibstd++

我做错了什么,如何使用ld链接我的目标文件?

-lstdc++ 是正确的,但你可能还需要使用 -L dir 告诉链接器在哪里找到该库。 - Jonathan Wakely
16
我要向 @gsingh2011 再次大喊一声,感谢他提出了如何做这件事的问题。这并不是一个幼稚或愚蠢的请求。即使你在现实生活中永远不会这样做,知道你可以这样做是理解工具链工作方式的关键练习(我相信)。 - Kristopher Micinski
2个回答

53

如果你在运行g++时加上-v参数,会显示它使用的链接命令。这里是一个简单的示例程序:

#include <iostream>

int main(void)
{
    std::cout << "Hello, world!" << std::endl;
    return 0;
}

运行 g++ -v -o example example.cpp 的输出结果:

Using built-in specs.
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.4.4-14ubuntu5.1' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.4 --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.4.5 (Ubuntu/Linaro 4.4.4-14ubuntu5.1) 
COLLECT_GCC_OPTIONS='-v' '-o' 'example' '-shared-libgcc' '-mtune=generic'
 /usr/lib/gcc/x86_64-linux-gnu/4.4.5/cc1plus -quiet -v -D_GNU_SOURCE example.cpp -D_FORTIFY_SOURCE=2 -quiet -dumpbase example.cpp -mtune=generic -auxbase example -version -fstack-protector -o /tmp/ccV8qjvd.s
ignoring nonexistent directory "/usr/local/include/x86_64-linux-gnu"
ignoring nonexistent directory "/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../x86_64-linux-gnu/include"
ignoring nonexistent directory "/usr/include/x86_64-linux-gnu"
#include "..." search starts here:
#include <...> search starts here:
 /usr/include/c++/4.4
 /usr/include/c++/4.4/x86_64-linux-gnu
 /usr/include/c++/4.4/backward
 /usr/local/include
 /usr/lib/gcc/x86_64-linux-gnu/4.4.5/include
 /usr/lib/gcc/x86_64-linux-gnu/4.4.5/include-fixed
 /usr/include
End of search list.
GNU C++ (Ubuntu/Linaro 4.4.4-14ubuntu5.1) version 4.4.5 (x86_64-linux-gnu)
    compiled by GNU C version 4.4.5, GMP version 4.3.2, MPFR version 3.0.0-p3.
GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
Compiler executable checksum: d92fbc2d715a3b7e0f4133f0c40053e4
COLLECT_GCC_OPTIONS='-v' '-o' 'example' '-shared-libgcc' '-mtune=generic'
 as -V -Qy -o /tmp/ccGHR0pc.o /tmp/ccV8qjvd.s
GNU assembler version 2.20.51 (x86_64-linux-gnu) using BFD version (GNU Binutils for Ubuntu) 2.20.51-system.20100908
COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/:/lib/../lib/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../:/lib/:/usr/lib/:/usr/lib/x86_64-linux-gnu/
COLLECT_GCC_OPTIONS='-v' '-o' 'example' '-shared-libgcc' '-mtune=generic'
 /usr/lib/gcc/x86_64-linux-gnu/4.4.5/collect2 --build-id --eh-frame-hdr -m elf_x86_64 --hash-style=gnu -dynamic-linker /lib64/ld-linux-x86-64.so.2 -o example -z relro /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crt1.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crti.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtbegin.o -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5 -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5 -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib -L/lib/../lib -L/usr/lib/../lib -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../.. -L/usr/lib/x86_64-linux-gnu /tmp/ccGHR0pc.o -lstdc++ -lm -lgcc_s -lgcc -lc -lgcc_s -lgcc /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtend.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crtn.o

哇,真是一团糟。幸运的是,链接行是最后一行,所以您可以很容易地看到发生了什么。

正如您在下面的评论中注意到的那样,前端正在使用collect2而不是ld。幸运的是,collect2只是ld的别名。这里是一个使用它的示例:

首先让我们生成一个对象文件:

$ ls
example.cpp
$ c++ -c example.cpp
$ ls
example.cpp example.o

然后我们将使用前端将其链接以查看链接线:

$ c++ -v -o example example.o
Using built-in specs.
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.4.4-14ubuntu5.1' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.4 --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.4.5 (Ubuntu/Linaro 4.4.4-14ubuntu5.1) 
COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/:/lib/../lib/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../:/lib/:/usr/lib/:/usr/lib/x86_64-linux-gnu/
COLLECT_GCC_OPTIONS='-v' '-o' 'example' '-shared-libgcc' '-mtune=generic'
 /usr/lib/gcc/x86_64-linux-gnu/4.4.5/collect2 --build-id --eh-frame-hdr -m elf_x86_64 --hash-style=gnu -dynamic-linker /lib64/ld-linux-x86-64.so.2 -o example -z relro /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crt1.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crti.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtbegin.o -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5 -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5 -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib -L/lib/../lib -L/usr/lib/../lib -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../.. -L/usr/lib/x86_64-linux-gnu example.o -lstdc++ -lm -lgcc_s -lgcc -lc -lgcc_s -lgcc /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtend.o /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crtn.o

然后丢掉这个二进制文件, 并进行链接(通常我会直接复制/粘贴该行代码,但为了更易读性,我使用了多行方式,采用了\):

$ ls
example  example.cpp  example.o
$ rm example
$ ls
example.cpp example.o
$ ld                                                              \
> --build-id                                                      \
> --eh-frame-hdr                                                  \
> -m elf_x86_64                                                   \
> --hash-style=gnu                                                \
> -dynamic-linker                                                 \
> /lib64/ld-linux-x86-64.so.2                                     \
> -o example                                                      \
> -z relro                                                        \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crt1.o      \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crti.o      \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtbegin.o                  \
> -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5                           \
> -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5                           \
> -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib           \
> -L/lib/../lib                                                   \
> -L/usr/lib/../lib                                               \
> -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../..                  \
> -L/usr/lib/x86_64-linux-gnu                                     \
> example.o                                                       \
> -lstdc++                                                        \
> -lm                                                             \
> -lgcc_s                                                         \
> -lgcc                                                           \
> -lc                                                             \
> -lgcc_s                                                         \
> -lgcc                                                           \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtend.o                    \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crtn.o

最后,运行它!

$ ls
example example.cpp example.o
$ ./example 
Hello, world!

经过一些实验,您可能可以通过删除一些参数来显著缩短该链接行。以下是我想出的最小设置:

你可以通过删除一些参数来显着缩短链接行。以下是我经过一些实验后找到的最小设置:

$ ld                                                              \
> -dynamic-linker                                                 \
> /lib64/ld-linux-x86-64.so.2                                     \
> -o example                                                      \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crt1.o      \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crti.o      \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtbegin.o                  \
> example.o                                                       \
> -L/usr/lib/gcc/x86_64-linux-gnu/4.4.5                           \
> -lstdc++                                                        \
> -lc                                                             \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/crtend.o                    \
> /usr/lib/gcc/x86_64-linux-gnu/4.4.5/../../../../lib/crtn.o

当然,这组标志和库将取决于您的程序使用的库函数和语言特性。


1
我能不能使用ld代替collect2?它们有什么区别? - gsgx
4
@gsingh2011,你可以使用ld替代g++,但如果没有非常充分的理由,你不应该这样做。g++使用的命令会随着每个版本的更新而变化,如果你不更新ld命令以匹配新版本,可能会遇到许多麻烦。 - Employed Russian
@CarlNorum 这正是解决方案,非常漂亮的解释。还要注意,甚至可以使用-###,它的功能类似于-v。 - touchStone
@AngelusMortis,应该是指-lstdc ++标记。 - Carl Norum
@AngelusMortis,可能不行。你几乎肯定也需要运行时文件。上面的列表是我为这个特定测试想出的最小集合。你可以进行自己的实验来比较。为什么不像应该的那样使用编译器驱动程序进行链接呢? - Carl Norum
显示剩余2条评论

0

我成功地编译了这样的代码

ld -s hello.o crt2.o -o hello.exe libstdc++.a libgcc.a libmingw32.a libmingwex.a libmsvcrt.a libkernel32.a

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接