Как узнать, что все символы экспортируются из общего объекта?

131

У меня есть общий объект (dll). Как мне узнать, что из этого экспортируются все символы?

Chappar
источник
1
Экспортируются все символы в объекте - даже «внутренние» функции. Вам просто нужно объявить их компилятору, чтобы они были готовы для компоновщика. Обычно это делается с помощью файла заголовка, как сказал ниже Райан Фокс.
Крис Лутц,
6
Крис Лутц ошибается: не все символы экспортируются из перемещаемых объектных файлов, не говоря уже о разделяемых библиотеках.
Занятый россиянин

Ответы:

218

У вас есть «общий объект» (обычно это общая библиотека в AIX), разделяемая библиотека UNIX или Windows DLL? Это все разные вещи, и ваш вопрос объединяет их все :-(

  • Для общего объекта AIX используйте dump -Tv /path/to/foo.o.
  • Для разделяемой библиотеки ELF используйте readelf -Ws /path/to/libfoo.soили (если у вас GNU nm) nm -D /path/to/libfoo.so.
  • Для разделяемой библиотеки UNIX, отличной от ELF, укажите, какая UNIX вас интересует.
  • Для Windows DLL используйте dumpbin /EXPORTS foo.dll.
Занятые русские
источник
7
В GNU / Linux нет такой утилиты «dumpbin». И вопрос помечен как linux.
Hi-Angel
3
Очень полезно, хорошо иметь такой обзор. nmтакже работает на MacOSX, кроме -Dопции. Или brew install binutilsиспользуйте версию GNU через gnm. Для GNU nm, --demangleтакже полезно. Также gobjdump.
Альберт
На самом деле, вы можете работать как с общими библиотеками, DLL, так и с объектами-файлами из одной утилиты, см. Этот ответ .
Hi-Angel
Вопрос помечен, linuxпоэтому я думаю, что можно с уверенностью сказать, что у @chappar есть общая библиотека Linux.
jww
Я полагаю, что во время выполнения нет API для этого, верно? Я обнаружил, что в Windows у вас есть GetProcAddress (), но вы не можете использовать его, не выполнив фактически библиотеку (что очень опасно, если у родительского приложения слишком много прав доступа).
Пабло Ариэль
23

objdump - еще один хороший вариант для Linux.

smcameron
источник
Также доступно на AIX
pitseeker
17

Если это файл DLL Windows и ваша ОС Linux, используйте winedump :

$ winedump -j export pcre.dll

Contents of pcre.dll: 229888 bytes

Exports table:

  Name:            pcre.dll
  Characteristics: 00000000
  TimeDateStamp:   53BBA519 Tue Jul  8 10:00:25 2014
  Version:         0.00
  Ordinal base:    1
  # of functions:  31
  # of Names:      31
Addresses of functions: 000375C8
Addresses of name ordinals: 000376C0
Addresses of names: 00037644

  Entry Pt  Ordn  Name
  0001FDA0     1 pcre_assign_jit_stack
  000380B8     2 pcre_callout
  00009030     3 pcre_compile
...
Бьорн Линдквист
источник
12

На * nix проверьте nm. В windows используйте программу Dependency Walker

AllDayCpp
источник
2
В частности, nm --defined-only -g something.soбудет печатать символы, которые определены в библиотеке и внешние символы, что, вероятно, и нужно OP.
Дэвид Грейсон
8

см человек нм

GNU nm перечисляет символы из объектных файлов objfile .... Если объектные файлы не указаны в качестве аргументов, nm принимает файл a.out.
VolkerK
источник
8
btw: для общих объектов вам понадобится динамический параметр -D / -. например, nm -D libmagic.so
VolkerK 06
8

Использование: nm --demangle <libname>.so

codebin
источник
2
nm: /usr/lib/i386-linux-gnu/libtemplates_parser.so.11.6: no symbols, readelfили -Dфлаг работает.
Янус Троелсен
5

Кроссплатформенный способ (не только сам кроссплатформенный, но также работающий, по крайней мере, с обоими *.soи *.dll) использует фреймворк обратного проектирования radare2 . Например:

$ rabin2 -s glew32.dll | head -n 5 
[Symbols]
vaddr=0x62afda8d paddr=0x0005ba8d ord=000 fwd=NONE sz=0 bind=GLOBAL type=FUNC name=glew32.dll___GLEW_3DFX_multisample
vaddr=0x62afda8e paddr=0x0005ba8e ord=001 fwd=NONE sz=0 bind=GLOBAL type=FUNC name=glew32.dll___GLEW_3DFX_tbuffer
vaddr=0x62afda8f paddr=0x0005ba8f ord=002 fwd=NONE sz=0 bind=GLOBAL type=FUNC name=glew32.dll___GLEW_3DFX_texture_compression_FXT1
vaddr=0x62afdab8 paddr=0x0005bab8 ord=003 fwd=NONE sz=0 bind=GLOBAL type=FUNC name=glew32.dll___GLEW_AMD_blend_minmax_factor

В качестве бонуса rabin2распознает искажение имени C ++, например (а также с .soфайлом) :

$ rabin2 -s /usr/lib/libabw-0.1.so.1.0.1 | head -n 5
[Symbols]
vaddr=0x00027590 paddr=0x00027590 ord=124 fwd=NONE sz=430 bind=GLOBAL type=FUNC name=libabw::AbiDocument::isFileFormatSupported
vaddr=0x0000a730 paddr=0x0000a730 ord=125 fwd=NONE sz=58 bind=UNKNOWN type=FUNC name=boost::exception::~exception
vaddr=0x00232680 paddr=0x00032680 ord=126 fwd=NONE sz=16 bind=UNKNOWN type=OBJECT name=typeinfoforboost::exception_detail::clone_base
vaddr=0x00027740 paddr=0x00027740 ord=127 fwd=NONE sz=235 bind=GLOBAL type=FUNC name=libabw::AbiDocument::parse

Также работает с объектными файлами:

$ g++ test.cpp -c -o a.o
$ rabin2 -s a.o | head -n 5
Warning: Cannot initialize program headers
Warning: Cannot initialize dynamic strings
Warning: Cannot initialize dynamic section
[Symbols]
vaddr=0x08000149 paddr=0x00000149 ord=006 fwd=NONE sz=1 bind=LOCAL type=OBJECT name=std::piecewise_construct
vaddr=0x08000149 paddr=0x00000149 ord=007 fwd=NONE sz=1 bind=LOCAL type=OBJECT name=std::__ioinit
vaddr=0x080000eb paddr=0x000000eb ord=017 fwd=NONE sz=73 bind=LOCAL type=FUNC name=__static_initialization_and_destruction_0
vaddr=0x08000134 paddr=0x00000134 ord=018 fwd=NONE sz=21 bind=LOCAL type=FUNC name=_GLOBAL__sub_I__Z4funcP6Animal
Привет, ангел
источник
1

Вы можете использовать gnu objdump. objdump -p your.dll, Затем перейдите к .edataсодержимому раздела, и вы найдете экспортированные функции под [Ordinal/Name Pointer] Table.

Дон Ф
источник
0

Обычно у вас также есть файл заголовка, который вы включаете в свой код для доступа к символам.

Райан Фокс
источник