mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-02-01 03:53:33 +00:00
131 lines
5.1 KiB
C
131 lines
5.1 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/bits/atomic.h"
|
|
#include "libc/calls/calls.h"
|
|
#include "libc/calls/strace.internal.h"
|
|
#include "libc/errno.h"
|
|
#include "libc/intrin/setjmp.internal.h"
|
|
#include "libc/macros.internal.h"
|
|
#include "libc/nexgen32e/threaded.h"
|
|
#include "libc/runtime/internal.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/sysv/consts/clone.h"
|
|
#include "libc/sysv/consts/map.h"
|
|
#include "libc/sysv/consts/prot.h"
|
|
#include "libc/thread/thread.h"
|
|
|
|
STATIC_YOINK("_main_thread_ctor");
|
|
|
|
static cthread_t cthread_allocate(const cthread_attr_t *attr) {
|
|
char *mem;
|
|
size_t size;
|
|
cthread_t td;
|
|
size = ROUNDUP(
|
|
attr->stacksize +
|
|
ROUNDUP((uintptr_t)_tls_size + sizeof(struct cthread_descriptor_t),
|
|
PAGESIZE),
|
|
FRAMESIZE);
|
|
mem = mmap(0, size, PROT_READ | PROT_WRITE, MAP_STACK | MAP_ANONYMOUS, -1, 0);
|
|
if (mem == MAP_FAILED) return 0;
|
|
if (attr->guardsize > PAGESIZE) {
|
|
mprotect(mem, attr->guardsize, PROT_NONE);
|
|
}
|
|
td = (cthread_t)(mem + size - sizeof(struct cthread_descriptor_t));
|
|
td->self = td;
|
|
td->self2 = td;
|
|
td->err = errno;
|
|
td->tid = -1;
|
|
td->stack.bottom = mem;
|
|
td->stack.top = mem + attr->stacksize;
|
|
td->alloc.bottom = mem;
|
|
td->alloc.top = mem + size;
|
|
if (attr->mode & CTHREAD_CREATE_DETACHED) {
|
|
td->state = cthread_detached;
|
|
} else {
|
|
td->state = cthread_started;
|
|
}
|
|
// Initialize TLS with content of .tdata section
|
|
memmove((void *)((intptr_t)td - (intptr_t)_tls_size), _tdata_start,
|
|
(intptr_t)_tdata_size);
|
|
return td;
|
|
}
|
|
|
|
static int cthread_start(void *arg) {
|
|
axdx_t rc;
|
|
void *exitcode;
|
|
cthread_t td = arg;
|
|
if (!(rc = setlongerjmp(td->exiter)).ax) {
|
|
exitcode = td->func(td->arg);
|
|
} else {
|
|
exitcode = (void *)rc.dx;
|
|
}
|
|
td->exitcode = exitcode;
|
|
if (atomic_load(&td->state) & cthread_detached) {
|
|
// we're still using the stack
|
|
// thus we can't munmap it yet
|
|
// kick the can down the road!
|
|
cthread_zombies_add(td);
|
|
}
|
|
atomic_fetch_add(&td->state, cthread_finished);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Creates thread.
|
|
*
|
|
* @param ptd will receive pointer to new thread descriptor
|
|
* @param attr contains special configuration if non-null
|
|
* @param func is thread callback function
|
|
* @param arg is argument supplied to `func`
|
|
* @return 0 on success, or error number on failure
|
|
* @threadsafe
|
|
*/
|
|
int cthread_create(cthread_t *ptd, const cthread_attr_t *attr,
|
|
void *(*func)(void *), void *arg) {
|
|
int rc, tid;
|
|
cthread_t td;
|
|
cthread_attr_t default_attr;
|
|
__threaded = true;
|
|
cthread_zombies_reap();
|
|
cthread_attr_init(&default_attr);
|
|
if ((td = cthread_allocate(attr ? attr : &default_attr))) {
|
|
td->func = func;
|
|
td->arg = arg;
|
|
cthread_attr_destroy(&default_attr);
|
|
tid =
|
|
clone(cthread_start, td->stack.bottom, td->stack.top - td->stack.bottom,
|
|
CLONE_THREAD | CLONE_VM | CLONE_FS | CLONE_FILES | CLONE_SIGHAND |
|
|
CLONE_SETTLS | CLONE_CHILD_SETTID | CLONE_CHILD_CLEARTID,
|
|
td, 0, td, sizeof(struct cthread_descriptor_t), &td->tid);
|
|
if (tid != -1) {
|
|
*ptd = td;
|
|
rc = 0;
|
|
} else {
|
|
rc = errno;
|
|
munmap(td->alloc.bottom, td->alloc.top - td->alloc.bottom);
|
|
}
|
|
} else {
|
|
rc = errno;
|
|
tid = -1;
|
|
}
|
|
STRACE("cthread_create([%d], %p, %p, %p) → %s", tid, attr, func, arg,
|
|
!rc ? "0" : strerrno(rc));
|
|
return rc;
|
|
}
|