7

之前在http://sourceware.org/ml/gdb/2007-06/msg00360.html中提到过。
但似乎没有人真正实施过这种想法。
实现这一点有什么障碍吗?

我的要求如下:

  1. 能够插入任何 elf 二进制可执行文件(例如,使用 LD_PRELOAD)
  2. 二进制文件可能是多线程可执行文件
  3. 二进制文件可能链接到包含 main 函数的库
  4. 这应该适用于 x86 以外的各种 cpu 架构(至少 MIPS、ARM、PPC)

所以如果已经有这样的解决方案,我想要一个链接,但如果还没有,我想知道为什么它还没有作为轮子实现。
可能只是没有人不需要它......但我认为这对于准备作为标准非常有用。

除了将代码放在一起之外,还需要任何技术或政治问题。

4

1 回答 1

8

好像不太难。

$ ./a.out
在 0x400966 处捕获信号:分段错误
分段故障
$ GDB_COMM=:1024 ./a.out
在 0x400966 处捕获信号:分段错误
随附的; PID = 2369
监听 1024 端口
$ gdb ./a.out
从 /home/me/a.out 读取符号...完成。
(gdb) 远程目标:1024
使用 :1024 进行远程调试
#define _XOPEN_SOURCE 500
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
static char *gdb_comm;
static void segv_handler(int sig, siginfo_t *si, void *uc) {
    pid_t child;
    char msg[84], pid[20];
    char *const argv[] = {"gdbserver", gdb_comm, "--attach", pid, NULL};
    sprintf(msg, "Caught signal at %p", si->si_addr);
    psignal(si->si_signo, msg);
    if (gdb_comm && *gdb_comm) {
        switch ((child = fork())) {
        case 0:
            sprintf(pid, "%ld", (long)getppid());
            execvp(argv[0], argv);
            perror("Failed to start gdbserver");
            _exit(-1);
        case -1:
            perror("failed to fork");
        default:
            waitpid(child, NULL, 0);
            break;
        }
    }
}
int main(int argc, char **argv) {
    static struct sigaction segv_action = {
        .sa_sigaction = segv_handler,
        .sa_flags = SA_RESETHAND | SA_SIGINFO,
    };
    gdb_comm = getenv("GDB_COMM");
    sigaction(SIGILL, &segv_action, NULL);
    sigaction(SIGFPE, &segv_action, NULL);
    sigaction(SIGSEGV, &segv_action, NULL);
    sigaction(SIGBUS, &segv_action, NULL);
    *(int *)main = 0;
    return 0;
}
于 2009-12-29T20:10:28.917 回答