Ale*_*Bar 11 c c++ linux shared-libraries dlopen
我可以在使用时以编程方式从共享库(仅限Linux)获取所有函数名称的列表dl_open()吗?
我想要这样的东西:
std::vector<std::string> list_all_functions(void *dl) {
//... what can I do here?
}
int main() {
void * dl = dl_open("./mylib.so", RTLD_NOW);
auto functions = list_all_functions(dl);
//...
dl_close(dl);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
示例库(mylib.so)
标题(.h):
extern "C" {
int sum (int a, int b);
}
Run Code Online (Sandbox Code Playgroud)
来源(.c):
int sum (int a, int b) { return a + b; }
Run Code Online (Sandbox Code Playgroud)
我知道的肮脏的黑客:使用nm或objdump实用
Max*_*sov 10
更新 | 长话短说:
我实际上找到了更短的方法:
auto library = dlopen("/path/to/lib.so", RTLD_LAZY | RTLD_GLOBAL);
const char * libname = "lib.so";
struct link_map * map = nullptr;
dlinfo(library, RTLD_DI_LINKMAP, &map);
Elf64_Sym * symtab = nullptr;
char * strtab = nullptr;
int symentries = 0;
for (auto section = map->l_ld; section->d_tag != DT_NULL; ++section)
{
if (section->d_tag == DT_SYMTAB)
{
symtab = (Elf64_Sym *)section->d_un.d_ptr;
}
if (section->d_tag == DT_STRTAB)
{
strtab = (char*)section->d_un.d_ptr;
}
if (section->d_tag == DT_SYMENT)
{
symentries = section->d_un.d_val;
}
}
int size = strtab - (char *)symtab;
for (int k = 0; k < size / symentries; ++k)
{
auto sym = &symtab[k];
// If sym is function
if (ELF64_ST_TYPE(symtab[k].st_info) == STT_FUNC)
{
//str is name of each symbol
auto str = &strtab[sym->st_name];
printf("%s\n", str);
}
}
Run Code Online (Sandbox Code Playgroud)
老的
我相信作者不再需要这个,但也许有人需要实际的代码,这就是(基于之前的答案)
首先,我们需要一个回调dl_iterate_phdr():
static int callback(struct dl_phdr_info *info, size_t size, void *data)
{
// data is copy of 2nd arg in dl_iterate_phdr
// you can use it for your lib name as I did
const char * libname = (const char *)data;
// if current elf's name contains your lib
if (strstr(info->dlpi_name, libname))
{
printf("loaded %s from: %s\n", libname, info->dlpi_name);
for (int j = 0; j < info->dlpi_phnum; j++)
{
// we need to save dyanmic section since it contains symbolic table
if (info->dlpi_phdr[j].p_type == PT_DYNAMIC)
{
Elf64_Sym * symtab = nullptr;
char * strtab = nullptr;
int symentries = 0;
auto dyn = (Elf64_Dyn *)(info->dlpi_addr + info->dlpi_phdr[j].p_vaddr);
for (int k = 0; k < info->dlpi_phdr[j].p_memsz / sizeof(Elf64_Dyn); ++k)
{
if (dyn[k].d_tag == DT_SYMTAB)
{
symtab = (Elf64_Sym *)dyn[k].d_un.d_ptr;
}
if (dyn[k].d_tag == DT_STRTAB)
{
strtab = (char*)dyn[k].d_un.d_ptr;
}
if (dyn[k].d_tag == DT_SYMENT)
{
symentries = dyn[k].d_un.d_val;
}
}
int size = strtab - (char *)symtab;
// for each string in table
for (int k = 0; k < size / symentries; ++k)
{
auto sym = &symtab[k];
auto str = &strtab[sym->st_name];
printf("%s\n", str);
}
break;
}
}
}
return 0;
}
Run Code Online (Sandbox Code Playgroud)
接下来,我们调用dl_iterate_phdr():
int main()
{
auto library = dlopen("/path/to/library.so", RTLD_LAZY | RTLD_GLOBAL);
const char * libname = "library.so";
dl_iterate_phdr(callback, (void*)libname);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
如果您需要将这些名称存储在某个地方,您可以将指针传递给容器,使用强制转换恢复它并写入其中。
对于我的示例库:
#include "simple_lib.h"
#include <cstdio>
void __attribute__ ((constructor)) initLibrary(void)
{
printf("Library is initialized\n");
}
void __attribute__ ((destructor)) cleanUpLibrary(void)
{
printf("Library is exited\n");
}
void make_number()
{
printf("1\n");
}
Run Code Online (Sandbox Code Playgroud)
打印这个:
Library is initialized
_ITM_deregisterTMCloneTable
puts
__gmon_start__
_ITM_registerTMCloneTable
__cxa_finalize
_Z11initLibraryv
make_number
_Z14cleanUpLibraryv
Library is exited
Run Code Online (Sandbox Code Playgroud)
没有 libc 函数可以做到这一点。但是,您可以自己编写一个(或从 readelf 等工具复制/粘贴代码)。
在 Linux 上,dlopen()返回一个结构的地址link_map,该结构有一个名为 的成员,l_addr该成员指向已加载共享对象的基地址(假设您的系统没有随机化共享库放置,并且您的库尚未预先链接)。
在 Linux 上,查找基地址( 的地址)的一种方法是在ing 库之后Elf*_Ehdr使用。dl_iterate_phdr()dlopen()
有了 ELF 头,您应该能够迭代导出符号的列表(动态符号表),方法是首先定位 ofElf*_Phdr类型PT_DYNAMIC,然后定位DT_SYMTAB,DT_STRTAB条目,并迭代动态符号表中的所有符号。用来/usr/include/elf.h指导你。
此外,您可以使用libelf,我个人对此不太了解。
但是,请注意,您将获得已定义函数的列表,但您不知道如何调用它们。
| 归档时间: |
|
| 查看次数: |
1332 次 |
| 最近记录: |