DLADDR(3) | Руководство программиста Linux | DLADDR(3) |
dladdr, dladdr1 - транслирует адрес в информацию о символе
#define _GNU_SOURCE #include <dlfcn.h>
int dladdr(void *addr, Dl_info *info);
int dladdr1(void *addr, Dl_info *info, void **extra_info, int flags);
Компонуется при указании параметра -ldl.
Функция dladdr() определяет, в каком из загруженных общих объектов приложения расположен адрес, указанный в addr. Если объект находится, то dladdr() возвращает информацию об общем объекте и символе, который относится к адресу addr. Данная информация возвращается в виде структуры Dl_info:
typedef struct { const char *dli_fname; /* путь к общему объекту, который содержит адрес */ void *dli_fbase; /* начальный адрес, по которому расположен общий объект */ const char *dli_sname; /* имя символа, к которому относится адрес addr */ void *dli_saddr; /* точный адрес символа с именем dli_sname */ } Dl_info;
Если символ, соответствующий адресу addr, не найден, то dli_sname и dli_saddr устанавливаются в NULL.
Функция dladdr1() подобна dladdr(), но возвращает дополнительную информацию в аргументе extra_info. Возвращаемая информация зависит от значения, указанного в flags, которое может быть одним из:
struct link_map { ElfW(Addr) l_addr; /* разница между адресом в файле ELF и адресом в памяти */ char *l_name; /* абсолютный путь, где был найден объект */ ElfW(Dyn) *l_ld; /* динамический раздел общего объекта */ struct link_map *l_next, *l_prev; /* цепочка загруженных объектов */ /* дополнительные поля, зависящие от реализации */ };
typedef struct { Elf64_Word st_name; /* имя символа */ unsigned char st_info; /* тип символа и привязка */ unsigned char st_other; /* видимость символа */ Elf64_Section st_shndx; /* номер раздела */ Elf64_Addr st_value; /* значение символа */ Elf64_Xword st_size; /* размер символа */ } Elf64_Sym;
Значение | Описание |
STT_NOTYPE | тип символа не указан |
STT_OBJECT | символ является объектом данных |
STT_FUNC | символ является объектом кода |
STT_SECTION | символ, связанный с разделом |
STT_FILE | имя символа является именем файла |
STT_COMMON | символ является объектом общих данных |
STT_TLS | символ является объектом локальных данных нити |
STT_GNU_IFUNC | символ является объектом скрытого кода |
Значение | Описание |
STB_LOCAL | локальный символ |
STB_GLOBAL | глобальный символ |
STB_WEAK | символ со слабой привязкой |
STB_GNU_UNIQUE | уникальный символ |
Значение | Описание |
STV_DEFAULT | правила видимости символов по умолчанию |
STV_INTERNAL | скрытый класс, зависящий от процессора |
STV_HIDDEN | Символ недоступен в других модулях |
STV_PROTECTED | невыгружаемый, не экспортируется |
При успешном выполнении эти функции возвращают ненулевое значение. Если адрес, указанный в addr, мог бы совпасть с общим объектом, но не с символом в общем объекте, то полям info->dli_sname и info->dli_saddr присваивается значение NULL.
Если адрес, указанный addr, мог бы совпасть с общим объектом, то эти функции возвращают 0. В этом случае сообщение об ошибке не доступно через dlerror(3).
Функция dladdr() есть в glibc 2.0 и новее. Функция dladdr1() впервые появилась в glibc 2.3.3.
Описание терминов данного раздела смотрите в attributes(7).
Интерфейс | Атрибут | Значение |
dladdr(), dladdr1() | Безвредность в нитях | MT-Safe |
Данные функции являются нестандартными расширениями GNU, которые также присутствуют в Solaris.
Иногда указатели на функции, передаваемые в dladdr(), могут вас удивить. На некоторых архитектурах (в частности, i386- и x86-64), dli_fname и dli_fbase могут указывать на объект, из которого вызывалась функция dladdr(), даже если функция, использовавшаяся как аргумент, должна быть из динамически скомпонованной библиотеки.
Проблема в том, что указатель на функцию по прежнему ищется во время компиляции, но всего лишь указывает на раздел plt (таблицу компоновки процедур) первоначального объекта (которая размещает вызов после запроса динамического компоновщика на поиск символа). Чтобы обойти это, вы можете попробовать скомпилировать независимый от размещения код: в этом случае компилятор больше не сможет подготовить указатель во время компиляции и gcc(1) создаст код, который просто загрузит конечный адрес символа из got (глобальной таблицы смещений) при запуске до передачи его в dladdr().
dl_iterate_phdr(3), dlinfo(3), dlopen(3), dlsym(3), ld.so(8)
2017-09-15 | Linux |