0

我正在玩 bfd 库 (<bfd.h>),我能够objdump -h通过打印出部分、它们的 vmas、大小等来实现我自己的二进制文件版本。现在,我在实现nm. 我可以使用该bfd库来获取二进制可执行文件的所有不同符号,但是如何使用 asection/asymbol 结构数据获取每个符号的(主要等)vma?这是我打印出每个符号名称的代码:

#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <bfd.h>

int main(int argc, char *argv[])
{
    bfd *ibfd = NULL;
    if (!argv[1])
    {
        printf("Please supply a second argument\n");
        return -1;
    }
    else
    {
        // initialize bfd so we can use it
        bfd_init();
        // open the supplied argument file
        const char *str = argv[1];
        ibfd = bfd_openr(str, "elf64-x86-64");

        // if issue opening
        if (!ibfd)
        {
            bfd_perror("open failure\n");
            return -1;
        }
        // if file isnt elf binary file
        if (!bfd_check_format(ibfd, bfd_object))
        {
            printf("not an object file\n");
            return -1;
        }

        int spaceNeeded = bfd_get_symtab_upper_bound(ibfd);
        if (spaceNeeded < 0)
        {
            return -1;
        }
        else if (spaceNeeded == 0)
        {
            return 1;
        }

        asymbol **symTable = malloc(spaceNeeded);

        long numSyms = bfd_canonicalize_symtab(ibfd, symTable);

        if (numSyms < 0)
            return -1;

        for (int i = 0, count = 0; i < numSyms; i++)
        {
            printf("%s\n", symTable[i]->name);
        }
        bfd_close(ibfd);
    }
    // success code
    return 1;
}
4

1 回答 1

0

nm 使用函数 bfd_symbol_info 来获取符号的虚拟内存地址。您可以阅读该函数的源代码以了解实现。

void
bfd_symbol_info (symbol, ret)
     asymbol *symbol;
     symbol_info *ret;
{
  ret->type = bfd_decode_symclass (symbol);

  if (bfd_is_undefined_symclass (ret->type))
    ret->value = 0;
  else
    ret->value = symbol->value + symbol->section->vma;

  ret->name = symbol->name;
}
于 2020-09-21T23:50:30.067 回答