Хоча ви можете встановити LD_LIBRARY_PATH, щоб динамічний лінкер знав, де шукати, є кращі варіанти. Ви можете розмістити свою спільну бібліотеку в одному зі стандартних місць, дивіться /etc/ld.so.conf
(у Linux) та /usr/bin/crle
(на Solaris) список цих місць
Ви можете перейти -R <path>
до посилання під час створення вашого двійкового файлу, який додасть <path>
до списку каталогів, відсканованих для вашої спільної бібліотеки. Ось приклад. По-перше, показ проблеми:
libtest.h:
void hello_world(void);
libtest.c:
#include <stdio.h>
void hello_world(void) {
printf("Hello world, I'm a library!\n");
}
hello.c:
#include "libtest.h"
int main(int argc, char **argv) {
hello_world();
}
Makefile (вкладки потрібно використовувати):
all: hello
hello: libtest.so.0
%.o: %.c
$(CC) $(CFLAGS) -fPIC -c -o $@ $<
libtest.so.0.0.1: libtest.o
$(CC) -shared -Wl,-soname,libtest.so.0 -o libtest.so.0.0.1 libtest.o
libtest.so.0: libtest.so.0.0.1
ln -s $< $@
clean:
rm -f hello libtest.o hello.o libtest.so.0.0.1 libtest.so.0
Давайте запустимо:
$ make
cc -fPIC -c -o libtest.o libtest.c
cc -shared -Wl,-soname,libtest.so.0 -o libtest.so.0.0.1 libtest.o
ln -s libtest.so.0.0.1 libtest.so.0
cc hello.c libtest.so.0 -o hello
$ ./hello
./hello: error while loading shared libraries: libtest.so.0: cannot open shared object file: No such file or directory
Як це виправити? Додайте -R <path>
до прапорців посилання (тут, встановивши LDFLAGS
).
$ make clean
(...)
$ make LDFLAGS="-Wl,-R -Wl,/home/maciej/src/tmp"
(...)
cc -Wl,-R -Wl,/home/maciej/src/tmp hello.c libtest.so.0 -o hello
$ ./hello
Hello world, I'm a library!
Дивлячись на бінарне, ви бачите, що йому потрібно libtest.so.0
:
$ objdump -p hello | grep NEEDED
NEEDED libtest.so.0
NEEDED libc.so.6
Бінарний файл шукає свої бібліотеки, крім стандартних місць, у вказаному каталозі:
$ objdump -p hello | grep RPATH
RPATH /home/maciej/src/tmp
Якщо ви хочете, щоб двійковий файл виглядав у поточному каталозі, ви можете встановити RPATH на $ORIGIN
. Це трохи хитро, тому що вам потрібно переконатися, що знак долара не інтерпретується make. Ось один із способів зробити це:
$ make CFLAGS="-fPIC" LDFLAGS="-Wl,-rpath '-Wl,\$\$ORIGIN'"
$ objdump -p hello | grep RPATH
RPATH $ORIGIN
$ ./hello
Hello world, I'm a library!
echo $LD_LIBRARY_PATH
порожній на моїй машині :(