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