it-swarm-ru.tech

Как работают числа SO (общий объект))?

Мне известно, что общие объекты в Linux используют "так числа", а именно, что разные версии общего объекта имеют разные расширения, например:

  • example.so.1
  • example.so.2

Я понимаю, что идея состоит в том, чтобы иметь два разных файла, чтобы в системе могли существовать две версии библиотеки (в отличие от "DLL Hell" в Windows). Я хотел бы знать, как это работает на практике? Часто я вижу, что example.so на самом деле является символической ссылкой на example.so.2 где .2 это последняя версия. Как тогда работает приложение в зависимости от более старой версии example.so определить это правильно? Есть ли какие-либо правила относительно того, какие числа нужно использовать? Или это просто соглашение? Является ли это случаем, что, в отличие от Windows, где двоичные файлы программного обеспечения передаются между системами, если в системе имеется более новая версия общего объекта, она автоматически связывается со старой версией при компиляции из исходного кода?

Я подозреваю, что это связано с ldconfig, но я не знаю как.

127
user119

Сами двоичные файлы знают, от какой версии разделяемой библиотеки они зависят, и запрашивают ее специально. Вы можете использовать ldd, чтобы показать зависимости; мои для ls являются:

$ ldd /bin/ls
    linux-gate.so.1 =>  (0xb784e000)
    librt.so.1 => /lib/librt.so.1 (0xb782c000)
    libacl.so.1 => /lib/libacl.so.1 (0xb7824000)
    libc.so.6 => /lib/libc.so.6 (0xb76dc000)
    libpthread.so.0 => /lib/libpthread.so.0 (0xb76c3000)
    /lib/ld-linux.so.2 (0xb784f000)
    libattr.so.1 => /lib/libattr.so.1 (0xb76bd000)

Как вы можете видеть, это указывает, например, на libpthread.so.0, не просто libpthread.so.


Причиной символической ссылки является компоновщик. Когда вы хотите связать с libpthread.so прямо, вы даете gcc флаг -lpthread и ​​добавляет префикс lib и ​​.so суффикс автоматически. Вы не можете сказать это, чтобы добавить на .so.0 суффикс, поэтому символическая ссылка указывает на новейшую версию библиотеки, чтобы облегчить

91
Michael Mrozek

Числа в общих библиотеках являются условным обозначением, используемым в Linux для идентификации API библиотеки. Обычно формат таков:

libFOO.so.MAJOR.MINOR

И, как вы заметили, обычно есть символическая ссылка от libFOO.so к libFOO.so.MAJOR.MINOR. ldconfig отвечает за обновление этой ссылки до последней версии.

Значение MAJOR обычно увеличивается при изменении API (удаляются новые точки входа или изменяются параметры или типы). MINOR обычно увеличивается для выпусков исправлений ошибок или когда новые API вводятся без нарушения существующих API.

Более подробное обсуждение можно найти здесь: Разделение разделяемых библиотек

61
miguel.de.icaza

Разделяемые библиотеки должны иметь версии по следующей схеме:

blah.so.X.Y.Z

где

  • X = обратно несовместимый выпуск ABI
  • Y = обратно совместимый выпуск ABI
  • Z = только внутренние изменения - без изменений в ABI

Обычно вы видите только первую цифру, например hello.so.1 потому что первая цифра - единственное, что необходимо для идентификации "версии" библиотеки, поскольку все остальные цифры обратно совместимы.

ldconfig содержит таблицу того, какие общие библиотеки доступны в системе и где существует путь к этой библиотеке. Вы можете проверить это, запустив:

ldconfig -p

Когда пакет создается для чего-то вроде Red Hat, совместно используемые библиотеки, вызываемые в двоичном файле, будут просматриваться и добавляться как зависимости пакета во время сборки RPM. Поэтому при переходе к установке пакета установщик проверяет, является ли hello.so.1 устанавливается в систему путем проверки ldconfig.

Вы можете увидеть зависимости пакета, выполнив что-то вроде:

rpm -qpR hello.rpm

Эта система (в отличие от Windows) допускает несколько версий hello.so для установки в системе и одновременного использования различными приложениями.

25
ascotan

libNAME.so - это имя файла, используемое компилятором/компоновщиком при первом поиске библиотеки, указанной в -lNAME. Внутри файла общей библиотеки находится поле под названием SONAME. Это поле устанавливается, когда сама библиотека сначала связывается в общий объект (то есть) в процессе сборки. Этот SONAME - это то, что компоновщик хранит в исполняемом файле, в зависимости от того, с каким общим объектом он связан. Обычно SONAME имеет вид libNAME.so.MAJOR и изменяется каждый раз, когда библиотека становится несовместимой с существующими исполняемыми файлами, связанными с ней, и обе основные версии библиотеки могут быть установлены по мере необходимости (хотя для разработки будет указана только одна). as libNAME.so) Кроме того, для поддержки простого обновления между второстепенными версиями библиотеки libNAME.so.MAJOR обычно представляет собой ссылку на файл, подобный libNAME.so.MAJOR.MINOR. Можно установить новую минорную версию, и после ее завершения ссылка на старую минорную версию будет увеличена, чтобы указать на новую минорную версию, немедленно обновляющую все новые исполнения для использования обновленной библиотеки. Также см. Мой ответ на Linux, GNU GCC, ld, сценарии версий и двоичный формат ELF - как это работает?

20
penguin359