2013-01-04 26 views
22

Próbuję połączyć wyjście C++ za pomocą ld, a nie g ++. Robię to tylko po to, aby nauczyć się, jak to zrobić, nie w celach praktycznych, więc proszę nie sugerujcie tego po prostu g ++.Jak powiązać pliki obiektów C++ z ld

Patrząc na this question osoba dostaje ten sam błąd po uruchomieniu komendy 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 

odpowiedzi w połączonej postu sugerują, że dodanie ++ biblioteki C jako argument łącznikowej będzie rozwiązać ten problem, tak próbowałem

ld test.o -o test.out -llibstd++ 

który jest co sugeruje, i ja też próbowałem wiele innych nazw, takich jak biblioteki libstdC++ lub STDC++. Ale zawsze otrzymam komunikat o błędzie, który wygląda tak, jakby był nieprawidłowy i jak mogę połączyć moje pliki obiektów za pomocą ld?

+0

Nie jest praktyczne robić to ... nigdy. – rubenvb

+9

@rubenvb Tak, i już powiedziałem, że nie używam go do celów praktycznych: "Robię to tylko po to, aby nauczyć się tego robić, nie w celach praktycznych". Czy to prawda? – gsingh2011

+0

Tak, byłem spadkobiercą. To jest złe.Jedyny czas, w którym powinieneś używać Ld, to kiedy tworzysz swój własny system operacyjny, w którym to przypadku prawdopodobnie sam napisałbyś wszystkie biblioteki systemowe. Jest to tak ekstremalnie system operacyjny i zależne od konfiguracji, że nawet nie jest zabawne. – rubenvb

Odpowiedz

22

Jeśli używasz g++ z flagą -v, zobaczysz linię łącza, której używa. Oto prosty przykład programu:

#include <iostream> 

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

a wyjście z systemem 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 

Wow, co za bałagan. Dogodnie linia łącza jest ostatnia, więc możesz łatwo zobaczyć, co się dzieje.

Jak zauważyłeś w swoim komentarzu poniżej, front-end używa collect2 zamiast ld. Na szczęście collect2 to tylko alias dla ld. Oto przykład używając go:

Zacznijmy wygenerować plik obiektu:

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

Następnie użyjemy front-end, aby je połączyć, aby zobaczyć linię link:

$ 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 

Następnie wyrzuć plik binarny i połącz się (normalnie, po prostu skopiowałbym/wkleiłam linię, ale żeby ułatwić czytanie, zrobiłem to w sposób wielowierszowy z s):

$ 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 

Wreszcie, uruchom!

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

Prawdopodobnie można znacząco skrócić tę linię łączącą, usuwając niektóre argumenty. Oto minimalny zestaw wpadłem po pewnym eksperymentów:

$ 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 

Zestaw flag i bibliotek będzie oczywiście zależeć od funkcji i język Biblioteka dysponuje program korzysta.

+0

Nie mogę użyć ld zamiast collect2? Co za różnica? – gsingh2011

+0

@ gsingh2011 - 'collect2' * to *' ld'. Sprawdź mój dodany przykład. –

+4

@ gsingh2011 Ty * możesz * używać 'ld' zamiast' g ++ ', ale ty ** nie powinieneś ** robić tego bez bardzo dobrego powodu. Polecenie, którego używa 'g ++ ', może zmienić się z jednej wersji' g ++ 'na następną, a jeśli nie zaktualizujesz swojego polecenia' ld', możesz mieć wiele nieprzyjemnych niespodzianek. –