1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/cpu/native/native_cpu.c

146 lines
3.2 KiB
C
Raw Normal View History

2013-03-06 10:29:49 +01:00
/**
* Native CPU kernel_intern.h and sched.h implementation
*
* in-process preemptive context switching utilizes POSIX ucontexts.
* (ucontext provides for architecture independent stack handling)
*
2013-03-06 10:29:49 +01:00
* Copyright (C) 2013 Ludwig Ortmann
*
* This file subject to the terms and conditions of the GNU Lesser General
* Public License. See the file LICENSE in the top level directory for more
* details.
2013-03-06 10:29:49 +01:00
*
* @ingroup native_cpu
2013-03-06 10:29:49 +01:00
* @{
* @file
* @author Ludwig Ortmann <ludwig.ortmann@fu-berlin.de>
*/
2013-03-06 01:08:15 +01:00
#include <stdio.h>
2013-05-15 17:45:43 +02:00
#ifdef __MACH__
#define _XOPEN_SOURCE
#endif
2013-03-06 01:08:15 +01:00
#include <ucontext.h>
2013-05-15 17:45:43 +02:00
#ifdef __MACH__
#undef _XOPEN_SOURCE
#endif
2013-03-06 01:08:15 +01:00
#include <err.h>
#include "kernel_internal.h"
2013-03-06 01:08:15 +01:00
#include "sched.h"
#include "cpu.h"
#include "cpu-conf.h"
#include "debug.h"
extern volatile tcb_t *active_thread;
static ucontext_t end_context;
static char __end_stack[SIGSTKSZ];
2013-03-06 01:08:15 +01:00
2013-08-08 11:08:33 +02:00
#ifdef MODULE_UART0
fd_set _native_rfds;
#endif
2013-03-06 01:08:15 +01:00
/**
* TODO: implement
*/
void thread_print_stack(void)
{
DEBUG("XXX: thread_print_stack()\n");
return;
}
char *thread_stack_init(void (*task_func)(void), void *stack_start, int stacksize)
2013-03-06 01:08:15 +01:00
{
unsigned int *stk;
ucontext_t *p;
DEBUG("thread_stack_init()\n");
stk = stack_start;
#ifdef NATIVESPONTOP
p = (ucontext_t *)stk;
stk += sizeof(ucontext_t) / sizeof(void *);
2013-03-06 01:08:15 +01:00
stacksize -= sizeof(ucontext_t);
#else
p = (ucontext_t *)(stk + ((stacksize - sizeof(ucontext_t)) / sizeof(void *)));
2013-03-06 01:08:15 +01:00
stacksize -= sizeof(ucontext_t);
#endif
if (getcontext(p) == -1) {
2013-03-06 01:08:15 +01:00
err(1, "thread_stack_init(): getcontext()");
}
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) {
2013-03-06 01:08:15 +01:00
err(1, "thread_stack_init(): sigemptyset()");
}
makecontext(p, task_func, 0);
return (char *) p;
}
void cpu_switch_context_exit(void)
{
2013-04-15 20:08:46 +02:00
ucontext_t *ctx;
2013-03-06 01:08:15 +01:00
DEBUG("XXX: cpu_switch_context_exit()\n");
//active_thread = sched_threads[0];
sched_run();
DEBUG("XXX: cpu_switch_context_exit(): calling setcontext(%s)\n\n", active_thread->name);
ctx = (ucontext_t *)(active_thread->sp);
eINT(); // XXX: workaround for bug (?) in sched_task_exit
if (setcontext(ctx) == -1) {
2013-03-06 01:08:15 +01:00
err(1, "cpu_switch_context_exit(): setcontext():");
}
}
void thread_yield()
{
2013-04-15 20:08:46 +02:00
/**
* XXX: check whether it is advisable to switch context for sched_run()
*/
ucontext_t *oc, *nc;
2013-03-06 01:08:15 +01:00
DEBUG("thread_yield()\n");
oc = (ucontext_t *)(active_thread->sp);
2013-04-15 20:08:46 +02:00
2013-03-06 01:08:15 +01:00
sched_run();
nc = (ucontext_t *)(active_thread->sp);
if (nc != oc) {
2013-04-15 20:08:46 +02:00
DEBUG("thread_yield(): calling swapcontext(%s)\n\n", active_thread->name);
if (swapcontext(oc, nc) == -1) {
2013-04-15 20:08:46 +02:00
err(1, "thread_yield(): swapcontext()");
}
}
else {
DEBUG("thread_yield(): old = new, returning to context (%s)\n\n", active_thread->name);
2013-03-06 01:08:15 +01:00
}
}
void native_cpu_init()
{
if (getcontext(&end_context) == -1) {
err(1, "end_context(): 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);
DEBUG("RIOT native cpu initialized.");
2013-03-06 01:08:15 +01:00
}
/** @} */