/** * Native CPU irq.h implementation * * Copyright (C) 2013 Ludwig Ortmann * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. * * @ingroup native_cpu * @ingroup irq * @{ * @file * @author Ludwig Ortmann */ #include #include #include #include #ifdef HAVE_VALGRIND_H #include #define VALGRIND_DEBUG DEBUG #elif defined(HAVE_VALGRIND_VALGRIND_H) #include #define VALGRIND_DEBUG DEBUG #else #define VALGRIND_STACK_REGISTER(...) #define VALGRIND_DEBUG(...) #endif // __USE_GNU for gregs[REG_EIP] access under Linux #define __USE_GNU #include #undef __USE_GNU #include "irq.h" #include "cpu.h" #include "lpm.h" #include "native_internal.h" #define ENABLE_DEBUG (0) #include "debug.h" volatile int native_interrupts_enabled; volatile int _native_in_isr; volatile int _native_in_syscall; static sigset_t _native_sig_set, _native_sig_set_dint; char __isr_stack[SIGSTKSZ]; ucontext_t native_isr_context; ucontext_t *_native_cur_ctx, *_native_isr_ctx; volatile unsigned int _native_saved_eip; volatile int _native_sigpend; int _sig_pipefd[2]; struct int_handler_t { void (*func)(void); }; static struct int_handler_t native_irq_handlers[255]; char sigalt_stk[SIGSTKSZ]; void print_thread_sigmask(ucontext_t *cp) { sigset_t *p = &cp->uc_sigmask; if (sigemptyset(p) == -1) { err(EXIT_FAILURE, "print_thread_sigmask: sigemptyset"); } for (int i = 1; i < (NSIG); i++) { if (native_irq_handlers[i].func != NULL) { printf("%s: %s\n", strsignal(i), (sigismember(&_native_sig_set, i) ? "blocked" : "unblocked") ); } if (sigismember(p, i)) { printf("%s: pending\n", strsignal(i)); } } } void print_sigmasks(void) { ucontext_t *p; //tcb_t *cb = NULL; for (int i = 0; i < MAXTHREADS; i++) { if (sched_threads[i] != NULL) { printf("%s:\n", sched_threads[i]->name); //print_thread_sigmask(sched_threads[i]->sp); p = (ucontext_t *)(sched_threads[i]->stack_start); print_thread_sigmask(p); puts(""); } } } void native_print_signals(void) { sigset_t p, q; puts("native signals:\n"); if (sigemptyset(&p) == -1) { err(EXIT_FAILURE, "native_print_signals: sigemptyset"); } if (sigpending(&p) == -1) { err(EXIT_FAILURE, "native_print_signals: sigpending"); } if (sigprocmask(SIG_SETMASK, NULL, &q) == -1) { err(EXIT_FAILURE, "native_print_signals(): sigprocmask"); } for (int i = 1; i < (NSIG); i++) { if (native_irq_handlers[i].func != NULL || i == SIGUSR1) { printf("%s: %s in active thread\n", strsignal(i), (sigismember(&_native_sig_set, i) ? "blocked" : "unblocked") ); } if (sigismember(&p, i)) { printf("%s: pending\n", strsignal(i)); } if (sigismember(&q, i)) { printf("%s: blocked in this context\n", strsignal(i)); } } } /** * block signals */ unsigned disableIRQ(void) { unsigned int prev_state; _native_syscall_enter(); DEBUG("disableIRQ()\n"); if (_native_in_isr == 1) { DEBUG("disableIRQ + _native_in_isr\n"); } if (sigprocmask(SIG_SETMASK, &_native_sig_set_dint, NULL) == -1) { err(EXIT_FAILURE, "disableIRQ(): sigprocmask()"); } prev_state = native_interrupts_enabled; native_interrupts_enabled = 0; DEBUG("disableIRQ(): return\n"); _native_syscall_leave(); return prev_state; } /** * unblock signals */ unsigned enableIRQ(void) { unsigned int prev_state; if (_native_in_isr == 1) { #if DEVELHELP real_write(STDERR_FILENO, "enableIRQ + _native_in_isr\n", 27); #else DEBUG("enableIRQ + _native_in_isr\n"); #endif } _native_syscall_enter(); DEBUG("enableIRQ()\n"); if (sigprocmask(SIG_SETMASK, &_native_sig_set, NULL) == -1) { err(EXIT_FAILURE, "enableIRQ(): sigprocmask()"); } prev_state = native_interrupts_enabled; native_interrupts_enabled = 1; _native_syscall_leave(); DEBUG("enableIRQ(): return\n"); return prev_state; } void restoreIRQ(unsigned state) { DEBUG("restoreIRQ()\n"); if (state == 1) { enableIRQ(); } else { disableIRQ(); } return; } int inISR(void) { DEBUG("inISR(): %i\n", _native_in_isr); return _native_in_isr; } void dINT(void) { disableIRQ(); } void eINT(void) { enableIRQ(); } int _native_popsig(void) { int nread, nleft, i; int sig; nleft = sizeof(int); i = 0; while ((nleft > 0) && ((nread = real_read(_sig_pipefd[0], ((uint8_t*)&sig) + i, nleft)) != -1)) { i += nread; nleft -= nread; } if (nread == -1) { err(EXIT_FAILURE, "_native_popsig(): real_read()"); } return sig; } /** * call signal handlers, * restore user context */ void native_irq_handler(void) { DEBUG("\n\n\t\tnative_irq_handler\n\n"); while (_native_sigpend > 0) { int sig = _native_popsig(); _native_sigpend--; if (native_irq_handlers[sig].func != NULL) { DEBUG("calling interrupt handler for %i\n", sig); native_irq_handlers[sig].func(); } else if (sig == SIGUSR1) { DEBUG("ignoring SIGUSR1\n"); } else { errx(EXIT_FAILURE, "XXX: no handler for signal %i\nXXX: this should not have happened!\n", sig); } } DEBUG("native_irq_handler(): return\n"); cpu_switch_context_exit(); } void isr_set_sigmask(ucontext_t *ctx) { ctx->uc_sigmask = _native_sig_set_dint; } /** * save signal, return to _native_sig_leave_tramp if possible */ void native_isr_entry(int sig, siginfo_t *info, void *context) { (void) info; /* unused at the moment */ //printf("\n\033[33m\n\t\tnative_isr_entry(%i)\n\n\033[0m", sig); /* save the signal */ if (real_write(_sig_pipefd[1], &sig, sizeof(int)) == -1) { err(EXIT_FAILURE, "native_isr_entry(): real_write()"); } _native_sigpend++; //real_write(STDOUT_FILENO, "sigpend\n", 8); if (context == NULL) { errx(EXIT_FAILURE, "native_isr_entry: context is null - unhandled"); } if (sched_active_thread == NULL) { _native_in_isr++; warnx("native_isr_entry: sched_active_thread is null - unhandled"); _native_in_isr--; return; } /* XXX: Workaround safety check - whenever this happens it really * indicates a bug in disableIRQ */ if (native_interrupts_enabled == 0) { //printf("interrupts are off, but I caught a signal.\n"); return; } if (_native_in_isr != 0) { //real_write(STDOUT_FILENO, "interrupts in ISR!!\n", 20); return; } if (_native_in_syscall != 0) { DEBUG("\n\n\t\treturn to syscall\n\n"); return; } native_isr_context.uc_stack.ss_sp = __isr_stack; native_isr_context.uc_stack.ss_size = SIGSTKSZ; native_isr_context.uc_stack.ss_flags = 0; makecontext(&native_isr_context, native_irq_handler, 0); _native_cur_ctx = (ucontext_t *)sched_active_thread->sp; DEBUG("\n\n\t\treturn to _native_sig_leave_tramp\n\n"); /* disable interrupts in context */ isr_set_sigmask((ucontext_t *)context); _native_in_isr = 1; #ifdef __MACH__ _native_saved_eip = ((ucontext_t *)context)->uc_mcontext->__ss.__eip; ((ucontext_t *)context)->uc_mcontext->__ss.__eip = (unsigned int)&_native_sig_leave_tramp; #elif defined(__FreeBSD__) _native_saved_eip = ((struct sigcontext *)context)->sc_eip; ((struct sigcontext *)context)->sc_eip = (unsigned int)&_native_sig_leave_tramp; #else //printf("\n\033[31mEIP:\t%p\ngo switching\n\n\033[0m", (void*)((ucontext_t *)context)->uc_mcontext.gregs[REG_EIP]); _native_saved_eip = ((ucontext_t *)context)->uc_mcontext.gregs[REG_EIP]; ((ucontext_t *)context)->uc_mcontext.gregs[REG_EIP] = (unsigned int)&_native_sig_leave_tramp; #endif } /** * register signal/interrupt handler for signal sig * * TODO: use appropriate data structure for signal * handlers. */ int register_interrupt(int sig, void (*handler)(void)) { DEBUG("register_interrupt()\n"); _native_syscall_enter(); if (sigdelset(&_native_sig_set, sig)) { err(EXIT_FAILURE, "register_interrupt: sigdelset"); } native_irq_handlers[sig].func = handler; /* set current dINT sigmask for all signals */ struct sigaction sa; sa.sa_sigaction = native_isr_entry; sa.sa_mask = _native_sig_set_dint; sa.sa_flags = SA_RESTART | SA_SIGINFO | SA_ONSTACK; for (int i = 0; i < 255; i++) { if (native_irq_handlers[i].func != NULL) { if (sigaction(sig, &sa, NULL)) { err(EXIT_FAILURE, "register_interrupt: sigaction"); } } } _native_syscall_leave(); return 0; } /** * empty signal mask */ int unregister_interrupt(int sig) { DEBUG("XXX: unregister_interrupt()\n"); _native_syscall_enter(); if (sigaddset(&_native_sig_set, sig) == -1) { err(EXIT_FAILURE, "unregister_interrupt: sigaddset"); } native_irq_handlers[sig].func = NULL; /* reset signal handler for sig */ struct sigaction sa; sa.sa_handler = SIG_IGN; /* there may be late signals, so we need to ignore those */ sa.sa_mask = _native_sig_set_dint; sa.sa_flags = SA_RESTART | SA_SIGINFO | SA_ONSTACK; if (sigaction(sig, &sa, NULL)) { err(EXIT_FAILURE, "unregister_interrupt: sigaction"); } /* change sigmask for remaining signal handlers */ sa.sa_sigaction = native_isr_entry; for (int i = 0; i < 255; i++) { if (native_irq_handlers[i].func != NULL) { if (sigaction(sig, &sa, NULL)) { err(EXIT_FAILURE, "register_interrupt: sigaction"); } } } _native_syscall_leave(); return 0; } void shutdown(int sig, siginfo_t *info, void *context) { (void)sig; (void)info; (void)context; lpm_set(LPM_OFF); } /** * register internal signal handler, * initalize local variables * * TODO: see register_interrupt */ void native_interrupt_init(void) { struct sigaction sa; DEBUG("XXX: native_interrupt_init()\n"); VALGRIND_STACK_REGISTER(__isr_stack, __isr_stack + sizeof(__isr_stack)); VALGRIND_DEBUG("VALGRIND_STACK_REGISTER(%p, %p)\n", __isr_stack, (void*)((int)__isr_stack + sizeof(__isr_stack))); native_interrupts_enabled = 1; _native_sigpend = 0; for (int i = 0; i < 255; i++) { native_irq_handlers[i].func = NULL; } sa.sa_sigaction = native_isr_entry; if (sigfillset(&sa.sa_mask) == -1) { err(EXIT_FAILURE, "native_interrupt_init: sigfillset"); } sa.sa_flags = SA_RESTART | SA_SIGINFO | SA_ONSTACK; /* get current process interrupt masks */ if (sigprocmask(SIG_SETMASK, NULL, &_native_sig_set) == -1) { err(EXIT_FAILURE, "native_interrupt_init(): sigprocmask"); } /* we need to disable all signals during our signal handler as it * can not cope with interrupted signals ... */ if (sigfillset(&_native_sig_set_dint) == -1) { err(EXIT_FAILURE, "native_isr_entry(): sigfillset"); } /* SIGUSR1 is intended for debugging purposes and shall always be * enabled */ if (sigdelset(&_native_sig_set, SIGUSR1) == -1) { err(EXIT_FAILURE, "native_interrupt_init: sigdelset"); } if (sigdelset(&_native_sig_set_dint, SIGUSR1) == -1) { err(EXIT_FAILURE, "native_interrupt_init: sigdelset"); } /* SIGUSR1 is handled like a regular interrupt */ if (sigaction(SIGUSR1, &sa, NULL)) { err(EXIT_FAILURE, "native_interrupt_init: sigaction"); } if (getcontext(&native_isr_context) == -1) { err(EXIT_FAILURE, "native_isr_entry(): getcontext()"); } native_isr_context.uc_stack.ss_sp = __isr_stack; native_isr_context.uc_stack.ss_size = SIGSTKSZ; native_isr_context.uc_stack.ss_flags = 0; _native_isr_ctx = &native_isr_context; static stack_t sigstk; sigstk.ss_sp = sigalt_stk; sigstk.ss_size = SIGSTKSZ; sigstk.ss_flags = 0; if (sigaltstack(&sigstk, NULL) < 0) { err(EXIT_FAILURE, "main: sigaltstack"); } makecontext(&native_isr_context, native_irq_handler, 0); _native_in_syscall = 0; if (real_pipe(_sig_pipefd) == -1) { err(EXIT_FAILURE, "native_interrupt_init(): pipe()"); } /* allow for ctrl+c to shut down gracefully always */ //register_interrupt(SIGINT, shutdown); sa.sa_sigaction = shutdown; if (sigdelset(&_native_sig_set, SIGINT) == -1) { err(EXIT_FAILURE, "native_interrupt_init: sigdelset"); } if (sigdelset(&_native_sig_set_dint, SIGINT) == -1) { err(EXIT_FAILURE, "native_interrupt_init: sigdelset"); } if (sigaction(SIGINT, &sa, NULL)) { err(EXIT_FAILURE, "native_interrupt_init: sigaction"); } puts("RIOT native interrupts/signals initialized."); } /** @} */