Подтвердить что ты не робот

Почему я получаю сообщение gcc "undefined reference", пытающееся создать общие объекты?

Почему я получаю ошибку "undefined reference" с помощью gcc?

Я пытаюсь создать общий объект (.so), который экспортирует одну функцию, "external()". Затем я пытаюсь связать с .so, но получим ссылку undefined 'external'. Что я здесь делаю неправильно?

Файл: external.c

int external() {
    return 5;
}

Файл: program.c

int external();
int main(char** argv, int* argc) {
    return external();
}

Команды

$ gcc -fPIC -c external.c
$ gcc -shared -o libexternal.so external.o
$ gcc -L. -lexternal -o program program.c
/tmp/cc3MmhAE.o: In function `main':
program.c:(.text+0x7): undefined reference to `external'
collect2: ld returned 1 exit status

Я могу даже запустить nm и видеть, что .so определяет "внешний":

Команда:

$ nm libexternal.so | grep external
0000040c T external

Что мне здесь не хватает?

4b9b3361

Ответ 1

Последние версии gcc/ld по умолчанию связаны с --as-needed.

Это означает, что если вы пишете -lexternal перед C файлом, библиотека будет автоматически исключена (порядок имеет значение при тестировании, если это так необходимо)

Вы можете исправить это с помощью любого из:

  • gcc -L. -o program program.c -lexternal
  • gcc -L. -Wl,--no-as-needed -lexternal -o program program.c

Последний из них передает --no-as-needed в компоновщик, что приведет к тому, что библиотека все еще будет связана, даже если вы не вызывали из нее external().

Примечание. -Wl,--no-as-needed не применяется глобально ко всему, что связано, оно применяется только к тем, которые следуют за ним в порядке командной строки. Так что -lexternal -Wl,--no-as-needed тоже не сработает. Это означает, что вы можете смешивать и сопоставлять поведение, хотя, например, gcc -L. -Wl,--no-as-needed -lexternal -Wl,--as-needed -o program program.c -lmightneed всегда ссылается на внешнюю, но только на ссылку против потенциального, если один или оба из program.c/libexternal.so заставляют это быть необходимым.