Нахождение dylib версия с помощью dlopen

Существует ли способ найти версию dylib использованием его пути? Я ищу что-то, что принимает те же аргументы как dlopen. Я посмотрел на NSVersionOfRunTimeLibrary, но от моего чтения документации похоже, что это получает версию текущего dylib, не ту, определенную в пути.

Спасибо

10
задан adk 14 February 2010 в 07:33
поделиться

2 ответа

Запустите otool -L на нем, и он покажет свою фактическую версию. Я выбрал libSystem.B, так как он имеет разную версию в SDK 10.4 и 10.5:

$ otool -L /Developer/SDKs/MacOSX10.4u.sdk/usr/lib/libSystem.B.dylib
/Developer/SDKs/MacOSX10.4u.sdk/usr/lib/libSystem.B.dylib:
    /usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 88.3.11)
    /usr/lib/system/libmathCommon.A.dylib (compatibility version 1.0.0, current version 220.0.0)
$ otool -L /Developer/SDKs/MacOSX10.5.sdk/usr/lib/libSystem.B.dylib 
/Developer/SDKs/MacOSX10.5.sdk/usr/lib/libSystem.B.dylib:
    /usr/lib/libSystem.B.dylib (compatibility version 1.0.0, current version 111.1.4)
    /usr/lib/system/libmathCommon.A.dylib (compatibility version 1.0.0, current version 292.4.0)

(посмотрите, как в первом случае версия 88.3.11, а во втором 111.1.4). Этот пример также показывает, что не все библиотеки являются символическими ссылками на файлы с номером версии:

$ ll /Developer/SDKs/MacOSX10.*.sdk/usr/lib/libSystem.B.dylib
-rwxr-xr-x  1 root  wheel   749K May 15  2009 /Developer/SDKs/MacOSX10.4u.sdk/usr/lib/libSystem.B.dylib
-rwxr-xr-x  1 root  wheel   670K May 15  2009 /Developer/SDKs/MacOSX10.5.sdk/usr/lib/libSystem.B.dylib
-rwxr-xr-x  1 root  wheel   901K Sep 25 00:21 /Developer/SDKs/MacOSX10.6.sdk/usr/lib/libSystem.B.dylib

Здесь файлы не имеют номера версии в своем имени.

EDIT: второе решение - использовать NSVersionOfRunTimeLibrary в тестовой программе, в которой вы принудительно загружаете библиотеку, которую хотите проверить. Создайте программу libversion из следующего исходного текста на Си:

#include <stdio.h>
#include <mach-o/dyld.h>

int main (int argc, char **argv)
{
  printf ("%x\n", NSVersionOfRunTimeLibrary (argv[1]));
  return 0;
}

Затем вызовите ее следующим образом:

$ DYLD_INSERT_LIBRARIES=/usr/lib/libpam.2.dylib ./a.out libpam.2.dylib
30000

(здесь номер версии выводится в шестнадцатеричном виде, но вы можете адаптировать его под свои нужды)

.
14
ответ дан 3 December 2019 в 20:04
поделиться

Вы можете посмотреть исходный код NSVersionOfRunTimeLibrary здесь: http://www.opensource.apple.com/source/dyld/dyld-132.13/src/dyldAPIsInLibSystem.cpp

На основе этого вы можете создать свою собственную версию, которая заменяет if(names_match(install_name, libraryName) == TRUE) на if(strcmp(_dyld_get_image_name(i), libraryName) == 0). Это исправит проблему того, что в оригинале ожидалось имя библиотеки без полного пути, а в отредактированной версии ожидается полный путь, но поиск все равно будет производиться в загруженных dylibs.

#include <mach-o/dyld.h>
int32_t
library_version(const char* libraryName)
{
    unsigned long i, j, n;
    struct load_command *load_commands, *lc;
    struct dylib_command *dl;
    const struct mach_header *mh;

    n = _dyld_image_count();
    for(i = 0; i < n; i++){
        mh = _dyld_get_image_header(i);
        if(mh->filetype != MH_DYLIB)
        continue;
        load_commands = (struct load_command *)
#if __LP64__
                ((char *)mh + sizeof(struct mach_header_64));
#else
                ((char *)mh + sizeof(struct mach_header));
#endif
        lc = load_commands;
        for(j = 0; j < mh->ncmds; j++){
        if(lc->cmd == LC_ID_DYLIB){
            dl = (struct dylib_command *)lc;
            if(strcmp(_dyld_get_image_name(i), libraryName) == 0)
            return(dl->dylib.current_version);
        }
        lc = (struct load_command *)((char *)lc + lc->cmdsize);
        }
    }
    return(-1);
}
5
ответ дан 3 December 2019 в 20:04
поделиться
Другие вопросы по тегам:

Похожие вопросы: