2014-04-16 22:45:42 +02:00
|
|
|
/*
|
2014-02-13 14:18:30 +01:00
|
|
|
* Copyright (C) 2013 Freie Universität Berlin
|
|
|
|
*
|
2014-07-31 19:45:27 +02: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.
|
2014-04-16 22:45:42 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2019-07-06 22:40:07 +02:00
|
|
|
* @ingroup pthread
|
2014-02-13 14:18:30 +01:00
|
|
|
* @{
|
2014-04-16 22:45:42 +02:00
|
|
|
* @file
|
|
|
|
* @brief Thread creation features.
|
|
|
|
* @see [The Open Group Base Specifications Issue 7: pthread.h - threads](http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/pthread.h.html)
|
2014-02-13 14:18:30 +01:00
|
|
|
* @author Christian Mehlis <mehlis@inf.fu-berlin.de>
|
|
|
|
* @author René Kijewski <kijewski@inf.fu-berlin.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2015-05-22 14:31:23 +02:00
|
|
|
#include "cpu_conf.h"
|
2014-02-13 14:18:30 +01:00
|
|
|
#include "irq.h"
|
|
|
|
#include "msg.h"
|
|
|
|
#include "mutex.h"
|
2014-07-29 09:21:11 +02:00
|
|
|
#include "priority_queue.h"
|
2014-02-13 14:18:30 +01:00
|
|
|
#include "thread.h"
|
|
|
|
#include "sched.h"
|
|
|
|
|
|
|
|
#include "pthread.h"
|
|
|
|
|
2016-11-07 19:49:09 +01:00
|
|
|
#ifdef HAVE_MALLOC_H
|
|
|
|
#include <malloc.h>
|
|
|
|
#endif
|
|
|
|
|
2014-03-03 22:12:25 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2019-09-25 15:35:53 +02:00
|
|
|
#ifndef CONFIG_PTHREAD_REAPER_BASE_STACKSIZE
|
|
|
|
#define CONFIG_PTHREAD_REAPER_BASE_STACKSIZE (THREAD_STACKSIZE_IDLE)
|
|
|
|
#endif
|
|
|
|
|
2014-02-13 14:18:30 +01:00
|
|
|
#if ENABLE_DEBUG
|
2019-09-25 15:35:53 +02:00
|
|
|
# define PTHREAD_REAPER_STACKSIZE ((CONFIG_PTHREAD_REAPER_BASE_STACKSIZE) + THREAD_EXTRA_STACKSIZE_PRINTF)
|
2015-04-28 20:02:05 +02:00
|
|
|
# define PTHREAD_STACKSIZE THREAD_STACKSIZE_MAIN
|
2014-02-13 14:18:30 +01:00
|
|
|
#else
|
2019-09-25 15:35:53 +02:00
|
|
|
# define PTHREAD_REAPER_STACKSIZE (CONFIG_PTHREAD_REAPER_BASE_STACKSIZE)
|
2015-04-28 20:02:05 +02:00
|
|
|
# define PTHREAD_STACKSIZE THREAD_STACKSIZE_DEFAULT
|
2014-02-13 14:18:30 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "debug.h"
|
|
|
|
|
2016-04-07 21:31:00 +02:00
|
|
|
typedef enum {
|
2014-02-13 14:18:30 +01:00
|
|
|
PTS_RUNNING,
|
|
|
|
PTS_DETACHED,
|
|
|
|
PTS_ZOMBIE,
|
2016-04-07 21:31:00 +02:00
|
|
|
} pthread_thread_status_t;
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2016-04-07 21:31:00 +02:00
|
|
|
typedef struct {
|
2014-08-06 13:54:11 +02:00
|
|
|
kernel_pid_t thread_pid;
|
2014-02-21 17:32:30 +01:00
|
|
|
|
2016-04-07 21:31:00 +02:00
|
|
|
pthread_thread_status_t status;
|
2014-08-07 15:06:50 +02:00
|
|
|
kernel_pid_t joining_thread;
|
2014-02-13 14:18:30 +01:00
|
|
|
void *returnval;
|
|
|
|
bool should_cancel;
|
|
|
|
|
|
|
|
void *(*start_routine)(void *);
|
|
|
|
void *arg;
|
|
|
|
|
|
|
|
char *stack;
|
2014-03-01 00:03:56 +01:00
|
|
|
|
2014-05-21 17:12:46 +02:00
|
|
|
struct __pthread_tls_datum *tls_head;
|
|
|
|
|
2014-03-01 00:03:56 +01:00
|
|
|
__pthread_cleanup_datum_t *cleanup_top;
|
2014-02-13 14:18:30 +01:00
|
|
|
} pthread_thread_t;
|
|
|
|
|
|
|
|
static pthread_thread_t *volatile pthread_sched_threads[MAXTHREADS];
|
2016-02-29 00:00:28 +01:00
|
|
|
static mutex_t pthread_mutex;
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2014-08-07 15:06:50 +02:00
|
|
|
static volatile kernel_pid_t pthread_reaper_pid = KERNEL_PID_UNDEF;
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2014-02-21 14:03:55 +01:00
|
|
|
static char pthread_reaper_stack[PTHREAD_REAPER_STACKSIZE];
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
static void *pthread_start_routine(void *pt_)
|
2014-02-13 14:18:30 +01:00
|
|
|
{
|
2014-03-04 20:20:01 +01:00
|
|
|
pthread_thread_t *pt = pt_;
|
2014-02-13 14:18:30 +01:00
|
|
|
void *retval = pt->start_routine(pt->arg);
|
|
|
|
pthread_exit(retval);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int insert(pthread_thread_t *pt)
|
|
|
|
{
|
2019-08-21 16:13:28 +02:00
|
|
|
int result = KERNEL_PID_UNDEF;
|
2014-02-13 14:18:30 +01:00
|
|
|
mutex_lock(&pthread_mutex);
|
2014-05-21 17:12:46 +02:00
|
|
|
|
2014-02-13 14:18:30 +01:00
|
|
|
for (int i = 0; i < MAXTHREADS; i++){
|
|
|
|
if (!pthread_sched_threads[i]) {
|
|
|
|
pthread_sched_threads[i] = pt;
|
2014-04-16 17:31:47 +02:00
|
|
|
result = i+1;
|
2014-02-13 14:18:30 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-05-21 17:12:46 +02:00
|
|
|
|
2014-02-13 14:18:30 +01:00
|
|
|
mutex_unlock(&pthread_mutex);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-03-04 20:20:01 +01:00
|
|
|
static void *pthread_reaper(void *arg)
|
2014-02-13 14:18:30 +01:00
|
|
|
{
|
2014-03-04 20:20:01 +01:00
|
|
|
(void) arg;
|
|
|
|
|
2014-02-13 14:18:30 +01:00
|
|
|
while (1) {
|
|
|
|
msg_t m;
|
|
|
|
msg_receive(&m);
|
2016-06-02 20:40:15 +02:00
|
|
|
DEBUG("pthread_reaper(): free(%p)\n", m.content.ptr);
|
2014-02-13 14:18:30 +01:00
|
|
|
free(m.content.ptr);
|
|
|
|
}
|
2014-03-04 20:20:01 +01:00
|
|
|
|
|
|
|
return NULL;
|
2014-02-13 14:18:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_create(pthread_t *newthread, const pthread_attr_t *attr, void *(*start_routine)(void *), void *arg)
|
|
|
|
{
|
|
|
|
pthread_thread_t *pt = calloc(1, sizeof(pthread_thread_t));
|
|
|
|
|
2019-10-19 18:48:08 +02:00
|
|
|
if (pt == NULL) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2014-07-06 22:57:56 +02:00
|
|
|
kernel_pid_t pthread_pid = insert(pt);
|
2014-08-06 11:52:00 +02:00
|
|
|
if (pthread_pid == KERNEL_PID_UNDEF) {
|
2014-02-13 14:18:30 +01:00
|
|
|
free(pt);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
*newthread = pthread_pid;
|
|
|
|
|
|
|
|
pt->status = attr && attr->detached ? PTS_DETACHED : PTS_RUNNING;
|
|
|
|
pt->start_routine = start_routine;
|
|
|
|
pt->arg = arg;
|
|
|
|
|
|
|
|
bool autofree = attr == NULL || attr->ss_sp == NULL || attr->ss_size == 0;
|
|
|
|
size_t stack_size = attr && attr->ss_size > 0 ? attr->ss_size : PTHREAD_STACKSIZE;
|
|
|
|
void *stack = autofree ? malloc(stack_size) : attr->ss_sp;
|
2019-10-19 18:48:08 +02:00
|
|
|
|
|
|
|
if (stack == NULL) {
|
|
|
|
free(pt);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2014-02-13 14:18:30 +01:00
|
|
|
pt->stack = autofree ? stack : NULL;
|
2019-08-23 15:56:41 +02:00
|
|
|
if (autofree && pthread_reaper_pid == KERNEL_PID_UNDEF) {
|
2014-02-13 14:18:30 +01:00
|
|
|
mutex_lock(&pthread_mutex);
|
2019-08-23 15:56:41 +02:00
|
|
|
if (pthread_reaper_pid == KERNEL_PID_UNDEF) {
|
2014-02-13 14:18:30 +01:00
|
|
|
/* volatile pid to overcome problems with double checking */
|
2014-07-06 22:57:56 +02:00
|
|
|
volatile kernel_pid_t pid = thread_create(pthread_reaper_stack,
|
2014-02-13 14:18:30 +01:00
|
|
|
PTHREAD_REAPER_STACKSIZE,
|
|
|
|
0,
|
2015-12-02 12:00:19 +01:00
|
|
|
THREAD_CREATE_STACKTEST,
|
2014-02-13 14:18:30 +01:00
|
|
|
pthread_reaper,
|
2014-03-04 20:20:01 +01:00
|
|
|
NULL,
|
2014-02-13 14:18:30 +01:00
|
|
|
"pthread-reaper");
|
2019-08-23 16:06:58 +02:00
|
|
|
if (!pid_is_valid(pid)) {
|
|
|
|
free(pt->stack);
|
|
|
|
free(pt);
|
|
|
|
pthread_sched_threads[pthread_pid-1] = NULL;
|
|
|
|
mutex_unlock(&pthread_mutex);
|
|
|
|
return -1;
|
|
|
|
}
|
2014-02-13 14:18:30 +01:00
|
|
|
pthread_reaper_pid = pid;
|
|
|
|
}
|
|
|
|
mutex_unlock(&pthread_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
pt->thread_pid = thread_create(stack,
|
|
|
|
stack_size,
|
2015-04-28 20:02:05 +02:00
|
|
|
THREAD_PRIORITY_MAIN,
|
2019-08-23 17:11:19 +02:00
|
|
|
THREAD_CREATE_SLEEPING |
|
2015-12-02 12:00:19 +01:00
|
|
|
THREAD_CREATE_STACKTEST,
|
2014-02-13 14:18:30 +01:00
|
|
|
pthread_start_routine,
|
2014-03-04 20:20:01 +01:00
|
|
|
pt,
|
2014-02-13 14:18:30 +01:00
|
|
|
"pthread");
|
2019-08-21 16:13:28 +02:00
|
|
|
if (!pid_is_valid(pt->thread_pid)) {
|
2014-02-13 14:18:30 +01:00
|
|
|
free(pt->stack);
|
|
|
|
free(pt);
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_sched_threads[pthread_pid-1] = NULL;
|
2014-02-13 14:18:30 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:11:19 +02:00
|
|
|
thread_wakeup(pt->thread_pid);
|
2014-02-13 14:18:30 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pthread_exit(void *retval)
|
|
|
|
{
|
2014-04-17 14:59:00 +02:00
|
|
|
pthread_t self_id = pthread_self();
|
|
|
|
|
|
|
|
if (self_id == 0) {
|
|
|
|
DEBUG("ERROR called pthread_self() returned 0 in \"%s\"!\n", __func__);
|
|
|
|
}
|
2014-04-29 18:29:30 +02:00
|
|
|
else {
|
2014-05-21 17:12:46 +02:00
|
|
|
pthread_thread_t *self = pthread_sched_threads[self_id - 1];
|
2014-04-17 14:59:00 +02:00
|
|
|
|
2014-04-29 18:29:30 +02:00
|
|
|
while (self->cleanup_top) {
|
|
|
|
__pthread_cleanup_datum_t *ct = self->cleanup_top;
|
|
|
|
self->cleanup_top = ct->__next;
|
2014-03-01 00:03:56 +01:00
|
|
|
|
2014-04-29 18:29:30 +02:00
|
|
|
ct->__routine(ct->__arg);
|
|
|
|
}
|
2014-03-01 00:03:56 +01:00
|
|
|
|
2014-05-21 17:12:46 +02:00
|
|
|
/* Prevent linking in pthread_tls.o if no TSS functions were used. */
|
|
|
|
extern void __pthread_keys_exit(int self_id) __attribute__((weak));
|
|
|
|
if (__pthread_keys_exit) {
|
|
|
|
__pthread_keys_exit(self_id);
|
|
|
|
}
|
|
|
|
|
2014-08-06 09:27:23 +02:00
|
|
|
self->thread_pid = KERNEL_PID_UNDEF;
|
2014-04-29 18:29:30 +02:00
|
|
|
DEBUG("pthread_exit(%p), self == %p\n", retval, (void *) self);
|
|
|
|
if (self->status != PTS_DETACHED) {
|
|
|
|
self->returnval = retval;
|
|
|
|
self->status = PTS_ZOMBIE;
|
2014-03-01 00:03:56 +01:00
|
|
|
|
2014-04-29 18:29:30 +02:00
|
|
|
if (self->joining_thread) {
|
|
|
|
/* our thread got an other thread waiting for us */
|
|
|
|
thread_wakeup(self->joining_thread);
|
|
|
|
}
|
|
|
|
}
|
2014-02-13 14:18:30 +01:00
|
|
|
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_disable();
|
2014-04-29 18:29:30 +02:00
|
|
|
if (self->stack) {
|
|
|
|
msg_t m;
|
|
|
|
m.content.ptr = self->stack;
|
|
|
|
msg_send_int(&m, pthread_reaper_pid);
|
2014-02-13 14:18:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sched_task_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_join(pthread_t th, void **thread_return)
|
|
|
|
{
|
2014-04-17 14:59:00 +02:00
|
|
|
if (th < 1 || th > MAXTHREADS) {
|
|
|
|
DEBUG("passed pthread_t th (%d) exceeds bounds of pthread_sched_threads[] in \"%s\"!\n", th, __func__);
|
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_thread_t *other = pthread_sched_threads[th-1];
|
2014-02-13 14:18:30 +01:00
|
|
|
if (!other) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (other->status) {
|
|
|
|
case (PTS_RUNNING):
|
2020-08-23 21:25:54 +02:00
|
|
|
other->joining_thread = thread_getpid();
|
2014-02-13 14:18:30 +01:00
|
|
|
/* go blocked, I'm waking up if other thread exits */
|
|
|
|
thread_sleep();
|
2018-01-16 11:06:49 +01:00
|
|
|
/* falls through */
|
2014-02-13 14:18:30 +01:00
|
|
|
case (PTS_ZOMBIE):
|
|
|
|
if (thread_return) {
|
|
|
|
*thread_return = other->returnval;
|
|
|
|
}
|
|
|
|
free(other);
|
|
|
|
/* we only need to free the pthread layer struct,
|
|
|
|
native thread stack is freed by other */
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_sched_threads[th-1] = NULL;
|
2014-02-13 14:18:30 +01:00
|
|
|
return 0;
|
|
|
|
case (PTS_DETACHED):
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_detach(pthread_t th)
|
|
|
|
{
|
2014-04-17 14:59:00 +02:00
|
|
|
if (th < 1 || th > MAXTHREADS) {
|
|
|
|
DEBUG("passed pthread_t th (%d) exceeds bounds of pthread_sched_threads[] in \"%s\"!\n", th, __func__);
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_thread_t *other = pthread_sched_threads[th-1];
|
2014-02-13 14:18:30 +01:00
|
|
|
if (!other) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (other->status == PTS_ZOMBIE) {
|
|
|
|
free(other);
|
|
|
|
/* we only need to free the pthread layer struct,
|
|
|
|
native thread stack is freed by other */
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_sched_threads[th-1] = NULL;
|
2014-02-13 14:18:30 +01:00
|
|
|
} else {
|
|
|
|
other->status = PTS_DETACHED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_t pthread_self(void)
|
|
|
|
{
|
2014-04-17 14:31:36 +02:00
|
|
|
pthread_t result = 0;
|
2014-02-13 14:18:30 +01:00
|
|
|
mutex_lock(&pthread_mutex);
|
2020-08-23 21:25:54 +02:00
|
|
|
kernel_pid_t pid = thread_getpid(); /* thread_getpid() is volatile */
|
2014-02-13 14:18:30 +01:00
|
|
|
for (int i = 0; i < MAXTHREADS; i++) {
|
|
|
|
if (pthread_sched_threads[i] && pthread_sched_threads[i]->thread_pid == pid) {
|
2014-04-16 17:31:47 +02:00
|
|
|
result = i+1;
|
2014-02-13 14:18:30 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&pthread_mutex);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_cancel(pthread_t th)
|
|
|
|
{
|
2014-04-16 17:31:47 +02:00
|
|
|
pthread_thread_t *other = pthread_sched_threads[th-1];
|
2014-02-13 14:18:30 +01:00
|
|
|
if (!other) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
other->should_cancel = 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_setcancelstate(int state, int *oldstate)
|
|
|
|
{
|
|
|
|
(void) state;
|
|
|
|
(void) oldstate;
|
2014-04-16 22:45:42 +02:00
|
|
|
return -1;
|
2014-02-13 14:18:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int pthread_setcanceltype(int type, int *oldtype)
|
|
|
|
{
|
|
|
|
(void) type;
|
|
|
|
(void) oldtype;
|
2014-04-16 22:45:42 +02:00
|
|
|
return -1;
|
2014-02-13 14:18:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void pthread_testcancel(void)
|
|
|
|
{
|
|
|
|
pthread_t self = pthread_self();
|
2014-04-17 14:59:00 +02:00
|
|
|
|
|
|
|
if (self == 0) {
|
|
|
|
DEBUG("ERROR called pthread_self() returned 0 in \"%s\"!\n", __func__);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-04-16 17:31:47 +02:00
|
|
|
if (pthread_sched_threads[self-1]->should_cancel) {
|
2014-04-16 22:45:42 +02:00
|
|
|
pthread_exit(PTHREAD_CANCELED);
|
2014-02-13 14:18:30 +01:00
|
|
|
}
|
|
|
|
}
|
2014-03-01 00:03:56 +01:00
|
|
|
|
|
|
|
void __pthread_cleanup_push(__pthread_cleanup_datum_t *datum)
|
|
|
|
{
|
2014-04-17 14:59:00 +02:00
|
|
|
pthread_t self_id = pthread_self();
|
|
|
|
|
|
|
|
if (self_id == 0) {
|
|
|
|
DEBUG("ERROR called pthread_self() returned 0 in \"%s\"!\n", __func__);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_thread_t *self = pthread_sched_threads[self_id-1];
|
2014-03-01 00:03:56 +01:00
|
|
|
datum->__next = self->cleanup_top;
|
|
|
|
self->cleanup_top = datum;
|
|
|
|
}
|
|
|
|
|
|
|
|
void __pthread_cleanup_pop(__pthread_cleanup_datum_t *datum, int execute)
|
|
|
|
{
|
2014-04-17 14:59:00 +02:00
|
|
|
pthread_t self_id = pthread_self();
|
|
|
|
|
|
|
|
if (self_id == 0) {
|
|
|
|
DEBUG("ERROR called pthread_self() returned 0 in \"%s\"!\n", __func__);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pthread_thread_t *self = pthread_sched_threads[self_id-1];
|
2014-03-01 00:03:56 +01:00
|
|
|
self->cleanup_top = datum->__next;
|
|
|
|
|
|
|
|
if (execute != 0) {
|
|
|
|
/* "The pthread_cleanup_pop() function shall remove the routine at the
|
|
|
|
* top of the calling thread's cancellation cleanup stack and optionally
|
|
|
|
* invoke it (if execute is non-zero)." */
|
|
|
|
datum->__routine(datum->__arg);
|
|
|
|
}
|
|
|
|
}
|
2014-05-21 17:12:46 +02:00
|
|
|
|
|
|
|
struct __pthread_tls_datum **__pthread_get_tls_head(int self_id)
|
|
|
|
{
|
|
|
|
pthread_thread_t *self = pthread_sched_threads[self_id-1];
|
|
|
|
return self ? &self->tls_head : NULL;
|
|
|
|
}
|