summaryrefslogtreecommitdiff
path: root/tests/Ltest-init-local-signal.c
blob: 4bde218f3bb2cfc556f0dc48b2c9841b1dc4da96 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#include "libunwind.h"
#include <sys/types.h>
#include <unistd.h>
#include <string.h>

#include <stdlib.h>

#include <signal.h>
#include <stdio.h>
#include <assert.h>

int stepper(unw_cursor_t* c) {
  int steps = 0;
  int ret = 1;
  while (ret) {

    ret = unw_step(c);
    if (!ret) {
      break;
    }
    steps++;
  }
  return steps;
}

/* Verify that we can step from both ucontext, and from getcontext()
 * roughly the same.  This tests that the IP from ucontext is used
 * correctly (see impl of unw_init_local2) */
void handler(int num, siginfo_t* info, void* ucontext) {
  unw_cursor_t c;
  unw_context_t context;
  unw_getcontext(&context);
  int ret = unw_init_local2(&c, ucontext, UNW_INIT_SIGNAL_FRAME);
  assert(!ret);
  int ucontext_steps = stepper(&c);

  ret = unw_init_local(&c, &context);
  (void)ret;
  assert(!ret);
  int getcontext_steps = stepper(&c);
  if (ucontext_steps == getcontext_steps - 2) {
    exit(0);
  }
  printf("unw_getcontext steps was %i, ucontext steps was %i, should be %i\n",
	 getcontext_steps, ucontext_steps, getcontext_steps - 2);
  exit(-1);
}

int foo(volatile int* f);

int main(){
  struct sigaction a;
  memset(&a, 0, sizeof(struct sigaction));
  a.sa_sigaction = &handler;
  a.sa_flags = SA_SIGINFO;
  sigaction(SIGSEGV, &a, NULL);

  foo(NULL);
  return 0;
}