/* Check that a fault signal handler gets the expected info */ #include <signal.h> #include <stdio.h> #include <stdlib.h> #include <fcntl.h> #include <setjmp.h> #include <unistd.h> struct test { void (*test)(void); int sig; int code; }; static const struct test *curr_test; static jmp_buf escape; static int testsig(int sig, int want) { if (sig != want) { fprintf(stderr, " FAIL: expected signal %d, not %d\n", want, sig); return 0; } return 1; } static int testcode(int code, int want) { if (code != want) { fprintf(stderr, " FAIL: expected si_code==%d, not %d\n", want, code); return 0; } return 1; } static void handler(int sig, siginfo_t *si, void *uc) { int ok = 1; ok = ok && testsig(sig, curr_test->sig); ok = ok && testcode(si->si_code, curr_test->code); if (ok) fprintf(stderr, " PASS\n"); siglongjmp(escape, ok + 1); } static void test1(void) { __asm__ volatile("li $t0, 0x80000000\n\t" "move $t1, $t0\n\t" "add $a0, $t0, $t1\n\t" : : : "t0", "t1", "a0", "cc", "memory"); } static void test2() { __asm__ volatile("li $t0, 0x7fffffff\n\t" "addi $a0, $t0, 0x7fff\n\t" : : : "t0", "a0", "cc", "memory"); } static void test3(void) { __asm__ volatile("li $t0, 0xffff0000\n\t" "li $t1, 0x7fffffff\n\t" "sub $a0, $t0, $t1\n\t" : : : "t0", "t1", "a0", "cc", "memory"); } int main() { int i; static const int sigs[] = { SIGFPE }; struct sigaction sa; sa.sa_sigaction = handler; sa.sa_flags = SA_SIGINFO; sigfillset(&sa.sa_mask); for(i = 0; i < sizeof(sigs)/sizeof(*sigs); i++) sigaction(sigs[i], &sa, NULL); const struct test tests[] = { #define T(n, sig, code) { test##n, sig, code } T(1, SIGFPE, FPE_INTOVF), T(2, SIGFPE, FPE_INTOVF), T(3, SIGFPE, FPE_INTOVF), #undef T }; for(i = 0; i < sizeof(tests)/sizeof(*tests); i++) { curr_test = &tests[i]; if (sigsetjmp(escape, 1) == 0) { fprintf(stderr, "Test %d: ", i+1); tests[i].test(); fprintf(stderr, " FAIL: no fault, or handler returned\n"); } } return 0; }