mirror of
https://github.com/python/cpython.git
synced 2025-10-24 18:33:49 +00:00

Joining a thread now ensures the underlying OS thread has exited. This is required for safer fork() in multi-threaded processes. --------- Co-authored-by: blurb-it[bot] <43283697+blurb-it[bot]@users.noreply.github.com>
191 lines
3.2 KiB
C
191 lines
3.2 KiB
C
#include "cpython/pthread_stubs.h"
|
|
|
|
// mutex
|
|
int
|
|
pthread_mutex_init(pthread_mutex_t *restrict mutex,
|
|
const pthread_mutexattr_t *restrict attr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_mutex_destroy(pthread_mutex_t *mutex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_mutex_trylock(pthread_mutex_t *mutex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_mutex_lock(pthread_mutex_t *mutex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_mutex_unlock(pthread_mutex_t *mutex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// condition
|
|
int
|
|
pthread_cond_init(pthread_cond_t *restrict cond,
|
|
const pthread_condattr_t *restrict attr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
PyAPI_FUNC(int) pthread_cond_destroy(pthread_cond_t *cond)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_cond_wait(pthread_cond_t *restrict cond,
|
|
pthread_mutex_t *restrict mutex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_cond_timedwait(pthread_cond_t *restrict cond,
|
|
pthread_mutex_t *restrict mutex,
|
|
const struct timespec *restrict abstime)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_cond_signal(pthread_cond_t *cond)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_condattr_init(pthread_condattr_t *attr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_condattr_setclock(pthread_condattr_t *attr, clockid_t clock_id)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// pthread
|
|
int
|
|
pthread_create(pthread_t *restrict thread,
|
|
const pthread_attr_t *restrict attr,
|
|
void *(*start_routine)(void *),
|
|
void *restrict arg)
|
|
{
|
|
return EAGAIN;
|
|
}
|
|
|
|
int
|
|
pthread_detach(pthread_t thread)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_join(pthread_t thread, void** value_ptr)
|
|
{
|
|
if (value_ptr) {
|
|
*value_ptr = NULL;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
PyAPI_FUNC(pthread_t) pthread_self(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_exit(void *retval)
|
|
{
|
|
exit(0);
|
|
}
|
|
|
|
int
|
|
pthread_attr_init(pthread_attr_t *attr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_attr_setstacksize(
|
|
pthread_attr_t *attr, size_t stacksize)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
pthread_attr_destroy(pthread_attr_t *attr)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
|
|
typedef struct py_stub_tls_entry py_tls_entry;
|
|
|
|
#define py_tls_entries (_PyRuntime.threads.stubs.tls_entries)
|
|
|
|
int
|
|
pthread_key_create(pthread_key_t *key, void (*destr_function)(void *))
|
|
{
|
|
if (!key) {
|
|
return EINVAL;
|
|
}
|
|
if (destr_function != NULL) {
|
|
Py_FatalError("pthread_key_create destructor is not supported");
|
|
}
|
|
for (pthread_key_t idx = 0; idx < PTHREAD_KEYS_MAX; idx++) {
|
|
if (!py_tls_entries[idx].in_use) {
|
|
py_tls_entries[idx].in_use = true;
|
|
*key = idx;
|
|
return 0;
|
|
}
|
|
}
|
|
return EAGAIN;
|
|
}
|
|
|
|
int
|
|
pthread_key_delete(pthread_key_t key)
|
|
{
|
|
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
|
|
return EINVAL;
|
|
}
|
|
py_tls_entries[key].in_use = false;
|
|
py_tls_entries[key].value = NULL;
|
|
return 0;
|
|
}
|
|
|
|
|
|
void *
|
|
pthread_getspecific(pthread_key_t key) {
|
|
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
|
|
return NULL;
|
|
}
|
|
return py_tls_entries[key].value;
|
|
}
|
|
|
|
int
|
|
pthread_setspecific(pthread_key_t key, const void *value)
|
|
{
|
|
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
|
|
return EINVAL;
|
|
}
|
|
py_tls_entries[key].value = (void *)value;
|
|
return 0;
|
|
}
|
|
|
|
// let thread_pthread define the Python API
|
|
#include "thread_pthread.h"
|