在C中列出所有的function/符号?

假定main.c使用来自main.c声明的共享库和函数的符号

是否有一个很好的,优雅的方式来在运行时打印所有可用的函数名称或符号的列表?

这应该是可能的,因为数据在.code段,但需要parsing它。

Solutions Collecting From Web of "在C中列出所有的function/符号?"

由于我有相同的需要在运行时检索所有加载的符号名称,所以我根据R ..的答案做了一些研究。 所以这里是一个ELF格式的linux共享库的详细解决方案,它可以和我的gcc 4.3.4一起工作,但是也可以用更新的版本。

我主要使用以下来源来开发此解决方案:

  • ELF手册页
  • 一些示例代码 (在搜索“dl_iterate_phdr”时找到它)

这是我的代码。 我用自我解释变量名称,并添加了详细的评论,使其可以理解。 如果有什么错误或遗漏,请让我知道…(编辑:我刚刚意识到,问题是C和我的代码是C ++。但是,如果你离开矢量和字符串它应该为C工作)

 #include <link.h> #include <string> #include <vector> using namespace std; /* Callback for dl_iterate_phdr. * Is called by dl_iterate_phdr for every loaded shared lib until something * else than 0 is returned by one call of this function. */ int retrieve_symbolnames(struct dl_phdr_info* info, size_t info_size, void* symbol_names_vector) { /* ElfW is a macro that creates proper typenames for the used system architecture * (eg on a 32 bit system, ElfW(Dyn*) becomes "Elf32_Dyn*") */ ElfW(Dyn*) dyn; ElfW(Sym*) sym; ElfW(Word*) hash; char* strtab = 0; char* sym_name = 0; ElfW(Word) sym_cnt = 0; /* the void pointer (3rd argument) should be a pointer to a vector<string> * in this example -> cast it to make it usable */ vector<string>* symbol_names = reinterpret_cast<vector<string>*>(symbol_names_vector); /* Iterate over all headers of the current shared lib * (first call is for the executable itself) */ for (size_t header_index = 0; header_index < info->dlpi_phnum; header_index++) { /* Further processing is only needed if the dynamic section is reached */ if (info->dlpi_phdr[header_index].p_type == PT_DYNAMIC) { /* Get a pointer to the first entry of the dynamic section. * It's address is the shared lib's address + the virtual address */ dyn = (ElfW(Dyn)*)(info->dlpi_addr + info->dlpi_phdr[header_index].p_vaddr); /* Iterate over all entries of the dynamic section until the * end of the symbol table is reached. This is indicated by * an entry with d_tag == DT_NULL. * * Only the following entries need to be processed to find the * symbol names: * - DT_HASH -> second word of the hash is the number of symbols * - DT_STRTAB -> pointer to the beginning of a string table that * contains the symbol names * - DT_SYMTAB -> pointer to the beginning of the symbols table */ while(dyn->d_tag != DT_NULL) { if (dyn->d_tag == DT_HASH) { /* Get a pointer to the hash */ hash = (ElfW(Word*))dyn->d_un.d_ptr; /* The 2nd word is the number of symbols */ sym_cnt = hash[1]; } else if (dyn->d_tag == DT_STRTAB) { /* Get the pointer to the string table */ strtab = (char*)dyn->d_un.d_ptr; } else if (dyn->d_tag == DT_SYMTAB) { /* Get the pointer to the first entry of the symbol table */ sym = (ElfW(Sym*))dyn->d_un.d_ptr; /* Iterate over the symbol table */ for (ElfW(Word) sym_index = 0; sym_index < sym_cnt; sym_index++) { /* get the name of the i-th symbol. * This is located at the address of st_name * relative to the beginning of the string table. */ sym_name = &strtab[sym[sym_index].st_name]; symbol_names->push_back(string(sym_name)); } } /* move pointer to the next entry */ dyn++; } } } /* Returning something != 0 stops further iterations, * since only the first entry, which is the executable itself, is needed * 1 is returned after processing the first entry. * * If the symbols of all loaded dynamic libs shall be found, * the return value has to be changed to 0. */ return 1; } int main() { vector<string> symbolNames; dl_iterate_phdr(retrieve_symbolnames, &symbolNames); return 0; } 

在动态链接的基于ELF的系统上,可能有一个函数dl_iterate_phdr可用。 如果是这样,它可以用来收集每个加载的共享库文件的信息,并且获得的信息足以检查符号表。 这个过程基本上是:

  1. 从传递给你的dl_phdr_info结构中获取程序头的地址。
  2. 使用PT_DYNAMIC程序标题查找模块的_DYNAMIC表。
  3. 使用DT_SYMTABDT_STRTABDT_HASH条目查找符号列表。 只需要DT_HASH来获取符号表的长度,因为它似乎没有存储在其他地方。

你需要的类型应该都在<elf.h><link.h>

这不是特定的C语言,而是操作系统和二进制格式,以及(用于调试符号和未加载的C ++符号名称)甚至编译器特定的问题。 没有通用的方式,也没有真正优雅的方式。

最便携和面向未来的方式可能是运行外部程序,例如POSIX中的nm 。 在Linux中找到的GNU版本可能有一堆扩展,如果你的目标是可移植性和面向未来,你应该避免这种扩展。

它的输出应该保持稳定,即使二进制格式改变,它也会得到更新并继续工作。 只要用正确的开关运行它,捕获它的输出(可能通过运行popen来避免临时文件)并解析它。

它应该是dl_iterate_phdr(retrieve_symbolnames, &symbolNames);