1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/native/native_cpu.c

249 lines
6.7 KiB
C
Raw Normal View History

2015-09-29 13:41:33 +02:00
/*
* Copyright (C) 2016 Kaspar Schleiser <kaspar@schleiser.de>
* 2013 Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.de>
2013-03-06 10:29:49 +01:00
*
* 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.
2015-09-29 13:41:33 +02:00
*/
/**
2018-06-01 12:17:51 +02:00
* @ingroup cpu_native
2013-03-06 10:29:49 +01:00
* @{
2015-09-29 13:41:33 +02:00
*
2013-03-06 10:29:49 +01:00
* @file
2015-09-29 13:41:33 +02:00
* @brief Native CPU kernel_intern.h and sched.h implementation
*
* in-process preemptive context switching utilizes POSIX ucontexts.
* (ucontext provides for architecture independent stack handling)
*
* @author Ludwig Knüpfer <ludwig.knuepfer@fu-berlin.de>
2015-09-29 13:41:33 +02:00
* @author Kaspar Schleiser <kaspar@schleiser.de>
2013-03-06 10:29:49 +01:00
*/
2014-02-14 16:18:40 +01:00
2013-03-06 01:08:15 +01:00
#include <stdio.h>
2014-02-14 16:18:40 +01:00
#include <unistd.h>
2014-11-28 12:14:54 +01:00
#include <stdlib.h>
#define __USE_GNU
#include <signal.h>
#undef __USE_GNU
2013-03-06 01:08:15 +01:00
#include <ucontext.h>
#include <err.h>
2013-09-30 14:07:10 +02:00
#ifdef HAVE_VALGRIND_H
#include <valgrind.h>
#define VALGRIND_DEBUG DEBUG
#elif defined(HAVE_VALGRIND_VALGRIND_H)
#include <valgrind/valgrind.h>
#define VALGRIND_DEBUG DEBUG
#else
#define VALGRIND_STACK_REGISTER(...)
#define VALGRIND_DEBUG(...)
#endif
#include <stdlib.h>
2014-02-14 16:18:40 +01:00
#include "irq.h"
2013-03-06 01:08:15 +01:00
#include "sched.h"
#include "cpu.h"
#include "cpu_conf.h"
#ifdef MODULE_NETDEV_TAP
#include "netdev_tap.h"
extern netdev_tap_t netdev_tap;
#endif
2013-09-30 14:07:10 +02:00
#include "native_internal.h"
2013-09-30 14:07:10 +02:00
#define ENABLE_DEBUG (0)
2013-03-06 01:08:15 +01:00
#include "debug.h"
ucontext_t end_context;
char __end_stack[SIGSTKSZ];
2013-03-06 01:08:15 +01:00
2014-11-28 12:14:54 +01:00
/**
* make the new context assign `_native_in_isr = 0` before resuming
*/
static void _native_mod_ctx_leave_sigh(ucontext_t *ctx)
{
#ifdef __MACH__
_native_saved_eip = ((ucontext_t *)ctx)->uc_mcontext->__ss.__eip;
((ucontext_t *)ctx)->uc_mcontext->__ss.__eip = (unsigned int)&_native_sig_leave_handler;
#elif defined(__FreeBSD__)
_native_saved_eip = ((struct sigcontext *)ctx)->sc_eip;
((struct sigcontext *)ctx)->sc_eip = (unsigned int)&_native_sig_leave_handler;
#else /* Linux */
#if defined(__arm__)
_native_saved_eip = ((ucontext_t *)ctx)->uc_mcontext.arm_pc;
((ucontext_t *)ctx)->uc_mcontext.arm_pc = (unsigned int)&_native_sig_leave_handler;
#else /* Linux/x86 */
_native_saved_eip = ctx->uc_mcontext.gregs[REG_EIP];
ctx->uc_mcontext.gregs[REG_EIP] = (unsigned int)&_native_sig_leave_handler;
#endif
#endif
}
2013-03-06 01:08:15 +01:00
/**
* TODO: implement
*/
void thread_print_stack(void)
{
DEBUG("thread_print_stack\n");
2013-03-06 01:08:15 +01:00
return;
}
/* This function calculates the ISR_usage */
2017-10-20 17:26:10 +02:00
int thread_isr_stack_usage(void)
{
/* TODO */
return -1;
}
2014-03-04 20:20:01 +01:00
char *thread_stack_init(thread_task_func_t task_func, void *arg, void *stack_start, int stacksize)
2013-03-06 01:08:15 +01:00
{
2015-07-14 11:15:10 +02:00
char *stk;
2013-03-06 01:08:15 +01:00
ucontext_t *p;
2014-07-25 08:17:06 +02:00
VALGRIND_STACK_REGISTER(stack_start, (char *) stack_start + stacksize);
VALGRIND_DEBUG("VALGRIND_STACK_REGISTER(%p, %p)\n",
stack_start, (void*)((int)stack_start + stacksize));
2013-09-30 14:07:10 +02:00
DEBUG("thread_stack_init\n");
2013-03-06 01:08:15 +01:00
2015-07-14 11:15:10 +02:00
stk = stack_start;
2013-03-06 01:08:15 +01:00
p = (ucontext_t *)(stk + (stacksize - sizeof(ucontext_t)));
2013-03-06 01:08:15 +01:00
stacksize -= sizeof(ucontext_t);
if (getcontext(p) == -1) {
err(EXIT_FAILURE, "thread_stack_init: getcontext");
2013-03-06 01:08:15 +01:00
}
p->uc_stack.ss_sp = stk;
p->uc_stack.ss_size = stacksize;
p->uc_stack.ss_flags = 0;
p->uc_link = &end_context;
if (sigemptyset(&(p->uc_sigmask)) == -1) {
err(EXIT_FAILURE, "thread_stack_init: sigemptyset");
2013-03-06 01:08:15 +01:00
}
2014-03-04 20:20:01 +01:00
makecontext(p, (void (*)(void)) task_func, 1, arg);
2013-03-06 01:08:15 +01:00
return (char *) p;
}
void isr_cpu_switch_context_exit(void)
2013-03-06 01:08:15 +01:00
{
2013-04-15 20:08:46 +02:00
ucontext_t *ctx;
DEBUG("isr_cpu_switch_context_exit\n");
if ((sched_context_switch_request == 1) || (sched_active_thread == NULL)) {
2013-09-30 15:45:47 +02:00
sched_run();
}
2013-03-06 01:08:15 +01:00
DEBUG("isr_cpu_switch_context_exit: calling setcontext(%" PRIkernel_pid ")\n\n", sched_active_pid);
ctx = (ucontext_t *)(sched_active_thread->sp);
2013-09-30 15:45:47 +02:00
native_interrupts_enabled = 1;
2014-11-28 12:14:54 +01:00
_native_mod_ctx_leave_sigh(ctx);
if (setcontext(ctx) == -1) {
err(EXIT_FAILURE, "isr_cpu_switch_context_exit: setcontext");
2013-03-06 01:08:15 +01:00
}
2014-01-29 10:54:52 +01:00
errx(EXIT_FAILURE, "2 this should have never been reached!!");
2013-03-06 01:08:15 +01:00
}
void cpu_switch_context_exit(void)
2013-03-06 01:08:15 +01:00
{
#ifdef NATIVE_AUTO_EXIT
if (sched_num_threads <= 1) {
DEBUG("cpu_switch_context_exit: last task has ended. exiting.\n");
real_exit(EXIT_SUCCESS);
}
#endif
if (_native_in_isr == 0) {
irq_disable();
_native_in_isr = 1;
native_isr_context.uc_stack.ss_sp = __isr_stack;
2014-11-28 11:47:42 +01:00
native_isr_context.uc_stack.ss_size = sizeof(__isr_stack);
native_isr_context.uc_stack.ss_flags = 0;
makecontext(&native_isr_context, isr_cpu_switch_context_exit, 0);
if (setcontext(&native_isr_context) == -1) {
2014-11-28 11:47:42 +01:00
err(EXIT_FAILURE, "cpu_switch_context_exit: setcontext");
}
2014-01-29 10:54:52 +01:00
errx(EXIT_FAILURE, "1 this should have never been reached!!");
}
else {
isr_cpu_switch_context_exit();
}
2014-01-29 10:54:52 +01:00
errx(EXIT_FAILURE, "3 this should have never been reached!!");
}
2013-03-06 01:08:15 +01:00
void isr_thread_yield(void)
{
DEBUG("isr_thread_yield\n");
2013-04-15 20:08:46 +02:00
2014-11-28 12:14:54 +01:00
if (_native_sigpend > 0) {
DEBUG("isr_thread_yield(): handling signals\n\n");
native_irq_handler();
}
2013-03-06 01:08:15 +01:00
sched_run();
ucontext_t *ctx = (ucontext_t *)(sched_active_thread->sp);
DEBUG("isr_thread_yield: switching to(%" PRIkernel_pid ")\n\n", sched_active_pid);
2013-03-06 01:08:15 +01:00
native_interrupts_enabled = 1;
2014-11-28 12:14:54 +01:00
_native_mod_ctx_leave_sigh(ctx);
if (setcontext(ctx) == -1) {
err(EXIT_FAILURE, "isr_thread_yield: setcontext");
}
}
void thread_yield_higher(void)
{
if (_native_in_isr == 0) {
2014-11-28 11:47:42 +01:00
ucontext_t *ctx = (ucontext_t *)(sched_active_thread->sp);
_native_in_isr = 1;
if (!native_interrupts_enabled) {
warnx("thread_yield_higher: interrupts are disabled - this should not be");
}
irq_disable();
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, isr_thread_yield, 0);
if (swapcontext(ctx, &native_isr_context) == -1) {
err(EXIT_FAILURE, "thread_yield_higher: swapcontext");
2013-04-15 20:08:46 +02:00
}
irq_enable();
2013-04-15 20:08:46 +02:00
}
else {
sched_context_switch_request = 1;
2013-03-06 01:08:15 +01:00
}
}
void native_cpu_init(void)
2013-03-06 01:08:15 +01:00
{
if (getcontext(&end_context) == -1) {
err(EXIT_FAILURE, "native_cpu_init: getcontext");
2013-03-06 01:08:15 +01:00
}
end_context.uc_stack.ss_sp = __end_stack;
end_context.uc_stack.ss_size = SIGSTKSZ;
end_context.uc_stack.ss_flags = 0;
makecontext(&end_context, sched_task_exit, 0);
2013-09-30 14:07:10 +02:00
VALGRIND_STACK_REGISTER(__end_stack, __end_stack + sizeof(__end_stack));
VALGRIND_DEBUG("VALGRIND_STACK_REGISTER(%p, %p)\n",
(void*)__end_stack, (void*)((int)__end_stack + sizeof(__end_stack)));
2014-01-22 18:23:10 +01:00
DEBUG("RIOT native cpu initialized.\n");
2013-03-06 01:08:15 +01:00
}
/** @} */