Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/c/65.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C 使用bfd从跟踪中恢复位置信息_C_Linux_Stack Trace_Debug Symbols_Bfd - Fatal编程技术网

C 使用bfd从跟踪中恢复位置信息

C 使用bfd从跟踪中恢复位置信息,c,linux,stack-trace,debug-symbols,bfd,C,Linux,Stack Trace,Debug Symbols,Bfd,我正在使用libld()中的dladdr来获取函数调用的跟踪。这里是一个单一跟踪元素的最小示例: #include<iostream> #include <dlfcn.h> // link with -ldl -rdynamic a void f(){ void **frame = static_cast<void **>(__builtin_frame_address(0)); void **bp = static_cast<void

我正在使用
libld
()中的
dladdr
来获取函数调用的跟踪。这里是一个单一跟踪元素的最小示例:

#include<iostream>
#include <dlfcn.h> // link with -ldl -rdynamic a

void f(){
    void **frame = static_cast<void **>(__builtin_frame_address(0));
    void **bp = static_cast<void **>(*frame);
    void *ip = frame[1];
    Dl_info info;
    dladdr(ip, &info);
    std::cout << info.dli_sname << " " << info.dli_fname << " " << info.dli_saddr << std::endl;

    ip = bp[1];
    bp = static_cast<void**>(bp[0]);
    dladdr(ip, &info);
    std::cout << info.dli_sname << " " << info.dli_fname << " " << info.dli_saddr << std::endl;
}

int main(){
    f();
}
也就是说,
Dl_info
具有跟踪的函数名、它所属的编译文件以及一些地址(
0x7f…
),在手册页中描述为“命名符号的确切地址”

地址包含源文件位置的信息(从中调用函数)。事实上,借助一些实用工具,我可以获得以下信息:

$ addr2line -e a.out
/home/user/test.cpp:34
(给出源文件中定义的
main
的确切行)。只要程序是用
-g
选项编译的,这就可以工作

现在我想要的是从程序中提取这些信息。据推测,这在BFD库中是可能的。

这是我的尝试,基于以下BFD示例:

1) 首先,我必须定义一个函数
find\u addr\u sect
,该函数稍后将由
bfd\u map\u over\u sections
(通过指针)调用

不幸的是,我被困在这里,因为
bfd\u map\u over\u部分
调用没有任何作用。我以错误的方式使用了
bfd\u map\u over\u sections
,为什么

对不起,使用C++,这是一个C问题。它缩短了我的大部分代码,我对它更习惯了


编辑:我添加了这一行,我可以确认问题的一个线索是节数为零

unsigned int numSections = -1;
numSections =  bfd_count_sections(abfd);
std::cout << "num sections " << numSections << std::endl; // gives "0"
unsigned int numSections=-1;
numSections=bfd\u count\u段(abfd);

std::cout我寻找了更多的示例,似乎我遗漏了两件事,在打开后调用函数
bfd\u check\u format
,并且在
bfd\u data
结构中填充和传递地址信息

...
    bfd *abfd = bfd_openr(info.dli_fname, NULL); assert(abfd);
//  char **matching;
//  bfd_data data;// = (bfd_data *)obj;

    if (!bfd_check_format (abfd, bfd_object)){
        bfd_close (abfd);       assert(0);
    }
...
稍后,
bfd_data
变量用作
find_addr_sect
的输入和输出。所以

...
bfd_data data;// = (bfd_data *)obj;
data.pc = (bfd_hostptr_t)info.dli_saddr;
data.found = FALSE;
bfd_map_over_sections(abfd, find_addr_sect, &data);
...

现在它工作了。

我寻找了更多的例子,似乎我遗漏了两件事,在打开后调用函数
bfd\u check\u format
,并在
bfd\u data
结构中填充和传递地址信息

...
    bfd *abfd = bfd_openr(info.dli_fname, NULL); assert(abfd);
//  char **matching;
//  bfd_data data;// = (bfd_data *)obj;

    if (!bfd_check_format (abfd, bfd_object)){
        bfd_close (abfd);       assert(0);
    }
...
稍后,
bfd_data
变量用作
find_addr_sect
的输入和输出。所以

...
bfd_data data;// = (bfd_data *)obj;
data.pc = (bfd_hostptr_t)info.dli_saddr;
data.found = FALSE;
bfd_map_over_sections(abfd, find_addr_sect, &data);
...
现在它开始工作了

...
bfd_data data;// = (bfd_data *)obj;
data.pc = (bfd_hostptr_t)info.dli_saddr;
data.found = FALSE;
bfd_map_over_sections(abfd, find_addr_sect, &data);
...