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

186 lines
4.2 KiB
C

/**
* thread functions
*
* Copyright (C) 2010 Freie Universität Berlin
*
* This file subject to the terms and conditions of the GNU General Public
* License. See the file LICENSE in the top level directory for more details.
*
* @ingroup kernel
* @{
* @file
* @author Kaspar Schleiser <kaspar.schleiser@fu-berlin.de>
* @}
*/
#include <errno.h>
#include <malloc.h>
#include <stdio.h>
#include "thread.h"
#include "kernel.h"
//#define ENABLE_DEBUG
#include "debug.h"
#include "kernel_intern.h"
#include "bitarithm.h"
#include "hwtimer.h"
#include "sched.h"
inline int thread_getpid() {
return active_thread->pid;
}
unsigned int thread_getstatus(int pid) {
if (sched_threads[pid]==NULL)
return STATUS_NOT_FOUND;
return sched_threads[pid]->status;
}
void thread_sleep() {
if ( inISR()) return;
dINT();
sched_set_status((tcb*)active_thread, STATUS_SLEEPING);
thread_yield();
}
int thread_wakeup(int pid) {
int isr = inISR();
if (! isr) dINT();
int result = sched_threads[pid]->status;
if (result == STATUS_SLEEPING) {
sched_set_status((tcb*)sched_threads[pid], STATUS_RUNNING);
if (!isr) {
eINT();
thread_yield();
} else {
sched_context_switch_request = 1;
}
return 0;
} else {
if (!isr) eINT();
return STATUS_NOT_FOUND;
}
}
int thread_measure_stack_usage(char* stack) {
unsigned int* stackp = (unsigned int*)stack;
/* assumption that the comparison fails before or after end of stack */
while( *stackp == (unsigned int)stackp )
stackp++;
int space = (unsigned int)stackp - (unsigned int)stack;
return space;
}
int thread_create(int stacksize, char priority, int flags, void (*function) (void), const char* name)
{
/* stacksize must be a multitude of 4 for alignment and stacktest */
// assert( ((stacksize & 0x03) == 0) && (stacksize > 0) );
// TODO: shall we autoalign the stack?
// stacksize += 4-(~(stacksize & 0x0003));
if (priority >= SCHED_PRIO_LEVELS) {
return -EINVAL;
}
tcb *pd = (tcb*)malloc(sizeof(tcb));
if ( pd == NULL) {
DEBUG("thread_create(): out of memory\n");
return -ENOMEM;
}
char *stack = (char*)malloc(stacksize);
if (stack==NULL)
{
DEBUG("thread_create(): out of memory\n");
free (pd);
return -ENOMEM;
}
if (flags & CREATE_STACKTEST) {
/* assign each int of the stack the value of it's address */
unsigned int *stackmax = (unsigned int*) ((char*)stack + stacksize);
unsigned int* stackp = (unsigned int*)stack;
while(stackp < stackmax) {
*stackp = (unsigned int)stackp;
stackp++;
}
} else {
/* create stack guard */
*stack = (unsigned int)stack;
}
if (! inISR()) {
dINT();
}
int pid = 0;
while (pid < MAXTHREADS) {
if (sched_threads[pid] == NULL) {
sched_threads[pid] = pd;
pd->pid = pid;
break;
}
pid++;
}
if (pid == MAXTHREADS) {
DEBUG("thread_create(): too many threads!\n");
free (pd);
free (stack);
if (! inISR()) {
eINT();
}
return -EOVERFLOW;
}
pd->sp = thread_stack_init(function,stack+stacksize);
pd->stack_start = stack;
pd->stack_size = stacksize;
pd->priority = priority;
pd->status = 0;
pd->name = name;
pd->wait_data = NULL;
pd->msg_queue.data = 0;
pd->msg_queue.priority = 0;
pd->msg_queue.next = NULL;
pd->rq_entry.data = (unsigned int) pd;
pd->rq_entry.next = NULL;
pd->rq_entry.prev = NULL;
num_tasks++;
DEBUG("Created thread %s. PID: %u. Priority: %u.\n", name, pd->pid, priority);
if (flags & CREATE_SLEEPING) {
sched_set_status(pd, STATUS_SLEEPING);
} else {
sched_set_status(pd, STATUS_PENDING);
if (!(flags & CREATE_WOUT_YIELD)) {
if (! inISR()) {
eINT();
thread_yield();
} else {
sched_context_switch_request = 1;
}
}
}
if (!inISR() && active_thread!=NULL) {
eINT();
}
return pid;
}