1 
2 #undef _GNU_SOURCE
3 #define _GNU_SOURCE 1
4 
5 #include <signal.h>
6 #include <stdio.h>
7 #include <sys/ucontext.h>
8 
9 static char* rip_at_sig = NULL;
10 
int_handler(int signum,siginfo_t * si,void * uc_arg)11 static void int_handler(int signum, siginfo_t *si, void *uc_arg)
12 {
13    ucontext_t *uc = (ucontext_t *)uc_arg;
14    /* Note that uc->uc_mcontext is an embedded struct, not a pointer */
15    mcontext_t *mc = &(uc->uc_mcontext);
16    void *pc = (void*)mc->gregs[REG_RIP];
17    printf("in int_handler, RIP is ...\n");
18    rip_at_sig = pc;
19 }
20 
register_handler(int sig,void * handler)21 static void register_handler(int sig, void *handler)
22 {
23    struct sigaction sa;
24    sa.sa_flags = SA_RESTART | SA_SIGINFO;
25    sigfillset(&sa.sa_mask);
26    sa.sa_sigaction = handler;
27    sigaction(sig, &sa, NULL);
28 }
29 
main(void)30 int main(void) {
31    char *intaddr = NULL;
32    puts("main");
33    register_handler(SIGTRAP, int_handler);
34    asm volatile(
35       "movabsq $zz_int, %%rdx\n"
36       "mov %%rdx, %0\n"
37       "zz_int:\n"
38       "int $3\n"
39       : /* no outputs */
40       : "m" (intaddr) /* input: address of var to store target addr to */
41       : /* clobbers */ "rdx"
42       );
43    /* intaddr is the address of the int 3 insn.  rip_at_sig is the PC
44       after the exception, which should be the next insn along.
45       Hence: */
46    if (intaddr != NULL && rip_at_sig != NULL
47        && rip_at_sig == intaddr+1)
48      printf("PASS\n");
49    else
50      printf("FAIL\n");
51    return 0;
52 }
53