2018-04-01 23:43:22 +00:00
|
|
|
|
/* Thread creation.
|
2019-01-01 00:11:28 +00:00
|
|
|
|
Copyright (C) 2000-2019 Free Software Foundation, Inc.
|
2018-04-01 23:43:22 +00:00
|
|
|
|
This file is part of the GNU C Library.
|
|
|
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
2018-04-02 14:37:36 +00:00
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
2018-04-01 23:43:22 +00:00
|
|
|
|
|
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
2018-04-02 14:37:36 +00:00
|
|
|
|
Lesser General Public License for more details.
|
2018-04-01 23:43:22 +00:00
|
|
|
|
|
2018-04-02 14:37:36 +00:00
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
License along with the GNU C Library; if not, see
|
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
2018-04-01 23:43:22 +00:00
|
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <pthread.h>
|
|
|
|
|
#include <signal.h>
|
|
|
|
|
#include <resolv.h>
|
|
|
|
|
|
|
|
|
|
#include <atomic.h>
|
|
|
|
|
#include <hurd/resource.h>
|
|
|
|
|
|
|
|
|
|
#include <pt-internal.h>
|
hurd: Avoid local PLTs in libpthread.
* htl/cthreads-compat.c (__cthread_detach): Call __pthread_detach
instead of pthread_detach.
(__cthread_fork): Call __pthread_create instead of pthread_create.
(__cthread_keycreate): Call __pthread_key_create instead of
pthread_key_create.
(__cthread_getspecific): Call __pthread_getspecific instead of
pthread_getspecific.
(__cthread_setspecific): Call __pthread_setspecific instead of
pthread_setspecific.
* htl/pt-alloc.c (__pthread_alloc): Call __pthread_mutex_lock and
__pthread_mutex_unlock instead of pthread_mutex_lock and
pthread_mutex_unlock.
* htl/pt-cleanup.c (__pthread_get_cleanup_stack): Rename to
___pthread_get_cleanup_stack.
(__pthread_get_cleanup_stack): New strong alias.
* htl/pt-create.c: Include <pthreadP.h>.
(entry_point): Call __pthread_exit instead of pthread_exit.
(pthread_create): Rename to __pthread_create.
(pthread_create): New strong alias.
* htl/pt-detach.c (pthread_detach): Rename to __pthread_detach.
(pthread_detach): New strong alias.
(__pthread_detach): Call __pthread_cond_broadcast instead of
pthread_cond_broadcast.
* htl/pt-exit.c (__pthread_exit): Call __pthread_setcancelstate
instead of pthread_setcancelstate.
* htl/pt-testcancel.c: Include <pthreadP.h>.
(pthread_testcancel): Call __pthread_exit instead of pthread_exit.
* sysdeps/htl/pt-attr-getstack.c: Include <pthreadP.h>
(__pthread_attr_getstack): Call __pthread_attr_getstackaddr and
__pthread_attr_getstacksize instead of pthread_attr_getstackaddr and
pthread_attr_getstacksize.
* sysdeps/htl/pt-attr-getstackaddr.c (pthread_attr_getstackaddr):
Rename to __pthread_attr_getstackaddr.
(pthread_attr_getstackaddr): New strong alias.
* sysdeps/htl/pt-attr-getstacksize.c (pthread_attr_getstacksize):
Rename to __pthread_attr_getstacksize.
(pthread_attr_getstacksize): New strong alias.
* sysdeps/htl/pt-attr-setstack.c: Include <pthreadP.h>.
(pthread_attr_setstack): Rename to __pthread_attr_setstack.
(pthread_attr_setstack): New strong alias.
(__pthread_attr_setstack): Call __pthread_attr_getstacksize,
__pthread_attr_setstacksize and __pthread_attr_setstackaddr instead of
pthread_attr_getstacksize, pthread_attr_setstacksize and
pthread_attr_setstackaddr.
* sysdeps/htl/pt-attr-setstackaddr.c (pthread_attr_setstackaddr):
Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/htl/pt-attr-setstacksize.c (pthread_attr_setstacksize):
Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/htl/pt-cond-timedwait.c: Include <pthreadP.h>.
(__pthread_cond_timedwait_internal): Use __pthread_exit instead of
pthread_exit.
* sysdeps/htl/pt-key-create.c: Include <pthreadP.h>.
(__pthread_key_create): New hidden def.
* sysdeps/htl/pt-key.h: Include <pthreadP.h>.
* sysdeps/htl/pthreadP.h (_pthread_mutex_init,
__pthread_cond_broadcast, __pthread_create, __pthread_detach,
__pthread_exit, __pthread_key_create, __pthread_getspecific,
__pthread_setspecific, __pthread_setcancelstate,
__pthread_attr_getstackaddr, __pthread_attr_setstackaddr,
__pthread_attr_getstacksize, __pthread_attr_setstacksize,
__pthread_attr_setstack, ___pthread_get_cleanup_stack): New
declarations.
(__pthread_key_create, _pthread_mutex_init): New hidden declarations.
* sysdeps/mach/hurd/htl/pt-attr-setstackaddr.c
(pthread_attr_setstackaddr): Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/mach/hurd/htl/pt-attr-setstacksize.c
(pthread_attr_setstacksize): Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/mach/hurd/htl/pt-docancel.c: Include <pthreadP.h>.
(call_exit): Call __pthread_exit instead of pthread_exit.
* sysdeps/mach/hurd/htl/pt-mutex-init.c: Include <pthreadP.h>.
(_pthread_mutex_init): New hidden definition.
* sysdeps/mach/hurd/htl/pt-sysdep.c: Include <pthreadP.h>.
(_init_routine): Call __pthread_attr_init and __pthread_attr_setstack
instead of pthread_attr_init and pthread_attr_setstack.
2018-04-02 18:08:37 +00:00
|
|
|
|
#include <pthreadP.h>
|
2018-04-01 23:43:22 +00:00
|
|
|
|
|
|
|
|
|
#if IS_IN (libpthread)
|
|
|
|
|
# include <ctype.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_USELOCALE
|
|
|
|
|
# include <locale.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* The total number of pthreads currently active. This is defined
|
|
|
|
|
here since it would be really stupid to have a threads-using
|
|
|
|
|
program that doesn't call `pthread_create'. */
|
|
|
|
|
unsigned int __pthread_total;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The entry-point for new threads. */
|
|
|
|
|
static void
|
|
|
|
|
entry_point (struct __pthread *self, void *(*start_routine) (void *), void *arg)
|
|
|
|
|
{
|
|
|
|
|
___pthread_self = self;
|
|
|
|
|
__resp = &self->res_state;
|
|
|
|
|
|
|
|
|
|
#if IS_IN (libpthread)
|
|
|
|
|
/* Initialize pointers to locale data. */
|
|
|
|
|
__ctype_init ();
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_USELOCALE
|
|
|
|
|
/* A fresh thread needs to be bound to the global locale. */
|
|
|
|
|
uselocale (LC_GLOBAL_LOCALE);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
__pthread_startup ();
|
|
|
|
|
|
hurd: Avoid local PLTs in libpthread.
* htl/cthreads-compat.c (__cthread_detach): Call __pthread_detach
instead of pthread_detach.
(__cthread_fork): Call __pthread_create instead of pthread_create.
(__cthread_keycreate): Call __pthread_key_create instead of
pthread_key_create.
(__cthread_getspecific): Call __pthread_getspecific instead of
pthread_getspecific.
(__cthread_setspecific): Call __pthread_setspecific instead of
pthread_setspecific.
* htl/pt-alloc.c (__pthread_alloc): Call __pthread_mutex_lock and
__pthread_mutex_unlock instead of pthread_mutex_lock and
pthread_mutex_unlock.
* htl/pt-cleanup.c (__pthread_get_cleanup_stack): Rename to
___pthread_get_cleanup_stack.
(__pthread_get_cleanup_stack): New strong alias.
* htl/pt-create.c: Include <pthreadP.h>.
(entry_point): Call __pthread_exit instead of pthread_exit.
(pthread_create): Rename to __pthread_create.
(pthread_create): New strong alias.
* htl/pt-detach.c (pthread_detach): Rename to __pthread_detach.
(pthread_detach): New strong alias.
(__pthread_detach): Call __pthread_cond_broadcast instead of
pthread_cond_broadcast.
* htl/pt-exit.c (__pthread_exit): Call __pthread_setcancelstate
instead of pthread_setcancelstate.
* htl/pt-testcancel.c: Include <pthreadP.h>.
(pthread_testcancel): Call __pthread_exit instead of pthread_exit.
* sysdeps/htl/pt-attr-getstack.c: Include <pthreadP.h>
(__pthread_attr_getstack): Call __pthread_attr_getstackaddr and
__pthread_attr_getstacksize instead of pthread_attr_getstackaddr and
pthread_attr_getstacksize.
* sysdeps/htl/pt-attr-getstackaddr.c (pthread_attr_getstackaddr):
Rename to __pthread_attr_getstackaddr.
(pthread_attr_getstackaddr): New strong alias.
* sysdeps/htl/pt-attr-getstacksize.c (pthread_attr_getstacksize):
Rename to __pthread_attr_getstacksize.
(pthread_attr_getstacksize): New strong alias.
* sysdeps/htl/pt-attr-setstack.c: Include <pthreadP.h>.
(pthread_attr_setstack): Rename to __pthread_attr_setstack.
(pthread_attr_setstack): New strong alias.
(__pthread_attr_setstack): Call __pthread_attr_getstacksize,
__pthread_attr_setstacksize and __pthread_attr_setstackaddr instead of
pthread_attr_getstacksize, pthread_attr_setstacksize and
pthread_attr_setstackaddr.
* sysdeps/htl/pt-attr-setstackaddr.c (pthread_attr_setstackaddr):
Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/htl/pt-attr-setstacksize.c (pthread_attr_setstacksize):
Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/htl/pt-cond-timedwait.c: Include <pthreadP.h>.
(__pthread_cond_timedwait_internal): Use __pthread_exit instead of
pthread_exit.
* sysdeps/htl/pt-key-create.c: Include <pthreadP.h>.
(__pthread_key_create): New hidden def.
* sysdeps/htl/pt-key.h: Include <pthreadP.h>.
* sysdeps/htl/pthreadP.h (_pthread_mutex_init,
__pthread_cond_broadcast, __pthread_create, __pthread_detach,
__pthread_exit, __pthread_key_create, __pthread_getspecific,
__pthread_setspecific, __pthread_setcancelstate,
__pthread_attr_getstackaddr, __pthread_attr_setstackaddr,
__pthread_attr_getstacksize, __pthread_attr_setstacksize,
__pthread_attr_setstack, ___pthread_get_cleanup_stack): New
declarations.
(__pthread_key_create, _pthread_mutex_init): New hidden declarations.
* sysdeps/mach/hurd/htl/pt-attr-setstackaddr.c
(pthread_attr_setstackaddr): Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/mach/hurd/htl/pt-attr-setstacksize.c
(pthread_attr_setstacksize): Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/mach/hurd/htl/pt-docancel.c: Include <pthreadP.h>.
(call_exit): Call __pthread_exit instead of pthread_exit.
* sysdeps/mach/hurd/htl/pt-mutex-init.c: Include <pthreadP.h>.
(_pthread_mutex_init): New hidden definition.
* sysdeps/mach/hurd/htl/pt-sysdep.c: Include <pthreadP.h>.
(_init_routine): Call __pthread_attr_init and __pthread_attr_setstack
instead of pthread_attr_init and pthread_attr_setstack.
2018-04-02 18:08:37 +00:00
|
|
|
|
__pthread_exit (start_routine (arg));
|
2018-04-01 23:43:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Create a thread with attributes given by ATTR, executing
|
|
|
|
|
START_ROUTINE with argument ARG. */
|
|
|
|
|
int
|
hurd: Avoid local PLTs in libpthread.
* htl/cthreads-compat.c (__cthread_detach): Call __pthread_detach
instead of pthread_detach.
(__cthread_fork): Call __pthread_create instead of pthread_create.
(__cthread_keycreate): Call __pthread_key_create instead of
pthread_key_create.
(__cthread_getspecific): Call __pthread_getspecific instead of
pthread_getspecific.
(__cthread_setspecific): Call __pthread_setspecific instead of
pthread_setspecific.
* htl/pt-alloc.c (__pthread_alloc): Call __pthread_mutex_lock and
__pthread_mutex_unlock instead of pthread_mutex_lock and
pthread_mutex_unlock.
* htl/pt-cleanup.c (__pthread_get_cleanup_stack): Rename to
___pthread_get_cleanup_stack.
(__pthread_get_cleanup_stack): New strong alias.
* htl/pt-create.c: Include <pthreadP.h>.
(entry_point): Call __pthread_exit instead of pthread_exit.
(pthread_create): Rename to __pthread_create.
(pthread_create): New strong alias.
* htl/pt-detach.c (pthread_detach): Rename to __pthread_detach.
(pthread_detach): New strong alias.
(__pthread_detach): Call __pthread_cond_broadcast instead of
pthread_cond_broadcast.
* htl/pt-exit.c (__pthread_exit): Call __pthread_setcancelstate
instead of pthread_setcancelstate.
* htl/pt-testcancel.c: Include <pthreadP.h>.
(pthread_testcancel): Call __pthread_exit instead of pthread_exit.
* sysdeps/htl/pt-attr-getstack.c: Include <pthreadP.h>
(__pthread_attr_getstack): Call __pthread_attr_getstackaddr and
__pthread_attr_getstacksize instead of pthread_attr_getstackaddr and
pthread_attr_getstacksize.
* sysdeps/htl/pt-attr-getstackaddr.c (pthread_attr_getstackaddr):
Rename to __pthread_attr_getstackaddr.
(pthread_attr_getstackaddr): New strong alias.
* sysdeps/htl/pt-attr-getstacksize.c (pthread_attr_getstacksize):
Rename to __pthread_attr_getstacksize.
(pthread_attr_getstacksize): New strong alias.
* sysdeps/htl/pt-attr-setstack.c: Include <pthreadP.h>.
(pthread_attr_setstack): Rename to __pthread_attr_setstack.
(pthread_attr_setstack): New strong alias.
(__pthread_attr_setstack): Call __pthread_attr_getstacksize,
__pthread_attr_setstacksize and __pthread_attr_setstackaddr instead of
pthread_attr_getstacksize, pthread_attr_setstacksize and
pthread_attr_setstackaddr.
* sysdeps/htl/pt-attr-setstackaddr.c (pthread_attr_setstackaddr):
Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/htl/pt-attr-setstacksize.c (pthread_attr_setstacksize):
Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/htl/pt-cond-timedwait.c: Include <pthreadP.h>.
(__pthread_cond_timedwait_internal): Use __pthread_exit instead of
pthread_exit.
* sysdeps/htl/pt-key-create.c: Include <pthreadP.h>.
(__pthread_key_create): New hidden def.
* sysdeps/htl/pt-key.h: Include <pthreadP.h>.
* sysdeps/htl/pthreadP.h (_pthread_mutex_init,
__pthread_cond_broadcast, __pthread_create, __pthread_detach,
__pthread_exit, __pthread_key_create, __pthread_getspecific,
__pthread_setspecific, __pthread_setcancelstate,
__pthread_attr_getstackaddr, __pthread_attr_setstackaddr,
__pthread_attr_getstacksize, __pthread_attr_setstacksize,
__pthread_attr_setstack, ___pthread_get_cleanup_stack): New
declarations.
(__pthread_key_create, _pthread_mutex_init): New hidden declarations.
* sysdeps/mach/hurd/htl/pt-attr-setstackaddr.c
(pthread_attr_setstackaddr): Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/mach/hurd/htl/pt-attr-setstacksize.c
(pthread_attr_setstacksize): Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/mach/hurd/htl/pt-docancel.c: Include <pthreadP.h>.
(call_exit): Call __pthread_exit instead of pthread_exit.
* sysdeps/mach/hurd/htl/pt-mutex-init.c: Include <pthreadP.h>.
(_pthread_mutex_init): New hidden definition.
* sysdeps/mach/hurd/htl/pt-sysdep.c: Include <pthreadP.h>.
(_init_routine): Call __pthread_attr_init and __pthread_attr_setstack
instead of pthread_attr_init and pthread_attr_setstack.
2018-04-02 18:08:37 +00:00
|
|
|
|
__pthread_create (pthread_t * thread, const pthread_attr_t * attr,
|
|
|
|
|
void *(*start_routine) (void *), void *arg)
|
2018-04-01 23:43:22 +00:00
|
|
|
|
{
|
|
|
|
|
int err;
|
|
|
|
|
struct __pthread *pthread;
|
|
|
|
|
|
|
|
|
|
err = __pthread_create_internal (&pthread, attr, start_routine, arg);
|
|
|
|
|
if (!err)
|
|
|
|
|
*thread = pthread->thread;
|
|
|
|
|
else if (err == ENOMEM)
|
|
|
|
|
err = EAGAIN;
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
hurd: Avoid local PLTs in libpthread.
* htl/cthreads-compat.c (__cthread_detach): Call __pthread_detach
instead of pthread_detach.
(__cthread_fork): Call __pthread_create instead of pthread_create.
(__cthread_keycreate): Call __pthread_key_create instead of
pthread_key_create.
(__cthread_getspecific): Call __pthread_getspecific instead of
pthread_getspecific.
(__cthread_setspecific): Call __pthread_setspecific instead of
pthread_setspecific.
* htl/pt-alloc.c (__pthread_alloc): Call __pthread_mutex_lock and
__pthread_mutex_unlock instead of pthread_mutex_lock and
pthread_mutex_unlock.
* htl/pt-cleanup.c (__pthread_get_cleanup_stack): Rename to
___pthread_get_cleanup_stack.
(__pthread_get_cleanup_stack): New strong alias.
* htl/pt-create.c: Include <pthreadP.h>.
(entry_point): Call __pthread_exit instead of pthread_exit.
(pthread_create): Rename to __pthread_create.
(pthread_create): New strong alias.
* htl/pt-detach.c (pthread_detach): Rename to __pthread_detach.
(pthread_detach): New strong alias.
(__pthread_detach): Call __pthread_cond_broadcast instead of
pthread_cond_broadcast.
* htl/pt-exit.c (__pthread_exit): Call __pthread_setcancelstate
instead of pthread_setcancelstate.
* htl/pt-testcancel.c: Include <pthreadP.h>.
(pthread_testcancel): Call __pthread_exit instead of pthread_exit.
* sysdeps/htl/pt-attr-getstack.c: Include <pthreadP.h>
(__pthread_attr_getstack): Call __pthread_attr_getstackaddr and
__pthread_attr_getstacksize instead of pthread_attr_getstackaddr and
pthread_attr_getstacksize.
* sysdeps/htl/pt-attr-getstackaddr.c (pthread_attr_getstackaddr):
Rename to __pthread_attr_getstackaddr.
(pthread_attr_getstackaddr): New strong alias.
* sysdeps/htl/pt-attr-getstacksize.c (pthread_attr_getstacksize):
Rename to __pthread_attr_getstacksize.
(pthread_attr_getstacksize): New strong alias.
* sysdeps/htl/pt-attr-setstack.c: Include <pthreadP.h>.
(pthread_attr_setstack): Rename to __pthread_attr_setstack.
(pthread_attr_setstack): New strong alias.
(__pthread_attr_setstack): Call __pthread_attr_getstacksize,
__pthread_attr_setstacksize and __pthread_attr_setstackaddr instead of
pthread_attr_getstacksize, pthread_attr_setstacksize and
pthread_attr_setstackaddr.
* sysdeps/htl/pt-attr-setstackaddr.c (pthread_attr_setstackaddr):
Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/htl/pt-attr-setstacksize.c (pthread_attr_setstacksize):
Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/htl/pt-cond-timedwait.c: Include <pthreadP.h>.
(__pthread_cond_timedwait_internal): Use __pthread_exit instead of
pthread_exit.
* sysdeps/htl/pt-key-create.c: Include <pthreadP.h>.
(__pthread_key_create): New hidden def.
* sysdeps/htl/pt-key.h: Include <pthreadP.h>.
* sysdeps/htl/pthreadP.h (_pthread_mutex_init,
__pthread_cond_broadcast, __pthread_create, __pthread_detach,
__pthread_exit, __pthread_key_create, __pthread_getspecific,
__pthread_setspecific, __pthread_setcancelstate,
__pthread_attr_getstackaddr, __pthread_attr_setstackaddr,
__pthread_attr_getstacksize, __pthread_attr_setstacksize,
__pthread_attr_setstack, ___pthread_get_cleanup_stack): New
declarations.
(__pthread_key_create, _pthread_mutex_init): New hidden declarations.
* sysdeps/mach/hurd/htl/pt-attr-setstackaddr.c
(pthread_attr_setstackaddr): Rename to __pthread_attr_setstackaddr.
(pthread_attr_setstackaddr): New strong alias.
* sysdeps/mach/hurd/htl/pt-attr-setstacksize.c
(pthread_attr_setstacksize): Rename to __pthread_attr_setstacksize.
(pthread_attr_setstacksize): New strong alias.
* sysdeps/mach/hurd/htl/pt-docancel.c: Include <pthreadP.h>.
(call_exit): Call __pthread_exit instead of pthread_exit.
* sysdeps/mach/hurd/htl/pt-mutex-init.c: Include <pthreadP.h>.
(_pthread_mutex_init): New hidden definition.
* sysdeps/mach/hurd/htl/pt-sysdep.c: Include <pthreadP.h>.
(_init_routine): Call __pthread_attr_init and __pthread_attr_setstack
instead of pthread_attr_init and pthread_attr_setstack.
2018-04-02 18:08:37 +00:00
|
|
|
|
strong_alias (__pthread_create, pthread_create)
|
2018-04-01 23:43:22 +00:00
|
|
|
|
|
|
|
|
|
/* Internal version of pthread_create. See comment in
|
|
|
|
|
pt-internal.h. */
|
|
|
|
|
int
|
|
|
|
|
__pthread_create_internal (struct __pthread **thread,
|
|
|
|
|
const pthread_attr_t * attr,
|
|
|
|
|
void *(*start_routine) (void *), void *arg)
|
|
|
|
|
{
|
|
|
|
|
int err;
|
|
|
|
|
struct __pthread *pthread;
|
|
|
|
|
const struct __pthread_attr *setup;
|
|
|
|
|
sigset_t sigset;
|
|
|
|
|
size_t stacksize;
|
|
|
|
|
|
|
|
|
|
/* Allocate a new thread structure. */
|
|
|
|
|
err = __pthread_alloc (&pthread);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed;
|
|
|
|
|
|
|
|
|
|
/* Use the default attributes if ATTR is NULL. */
|
|
|
|
|
setup = attr ? attr : &__pthread_default_attr;
|
|
|
|
|
|
|
|
|
|
stacksize = setup->__stacksize;
|
|
|
|
|
if (stacksize == 0)
|
|
|
|
|
{
|
|
|
|
|
struct rlimit rlim;
|
|
|
|
|
__getrlimit (RLIMIT_STACK, &rlim);
|
|
|
|
|
if (rlim.rlim_cur != RLIM_INFINITY)
|
|
|
|
|
stacksize = rlim.rlim_cur;
|
|
|
|
|
if (stacksize == 0)
|
|
|
|
|
stacksize = PTHREAD_STACK_DEFAULT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Initialize the thread state. */
|
|
|
|
|
pthread->state = (setup->__detachstate == PTHREAD_CREATE_DETACHED
|
|
|
|
|
? PTHREAD_DETACHED : PTHREAD_JOINABLE);
|
|
|
|
|
|
|
|
|
|
if (setup->__stackaddr)
|
|
|
|
|
{
|
|
|
|
|
pthread->stackaddr = setup->__stackaddr;
|
|
|
|
|
|
|
|
|
|
/* If the user supplied a stack, it is not our responsibility to
|
|
|
|
|
setup a stack guard. */
|
|
|
|
|
pthread->guardsize = 0;
|
|
|
|
|
pthread->stack = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Allocate a stack. */
|
|
|
|
|
err = __pthread_stack_alloc (&pthread->stackaddr,
|
|
|
|
|
((setup->__guardsize + __vm_page_size - 1)
|
|
|
|
|
/ __vm_page_size) * __vm_page_size
|
|
|
|
|
+ stacksize);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed_stack_alloc;
|
|
|
|
|
|
|
|
|
|
pthread->guardsize = setup->__guardsize;
|
|
|
|
|
pthread->stack = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pthread->stacksize = stacksize;
|
|
|
|
|
|
|
|
|
|
/* Allocate the kernel thread and other required resources. */
|
|
|
|
|
err = __pthread_thread_alloc (pthread);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed_thread_alloc;
|
|
|
|
|
|
|
|
|
|
pthread->tcb = _dl_allocate_tls (NULL);
|
|
|
|
|
if (pthread->tcb == NULL)
|
|
|
|
|
{
|
|
|
|
|
err = ENOMEM;
|
|
|
|
|
goto failed_thread_tls_alloc;
|
|
|
|
|
}
|
|
|
|
|
pthread->tcb->tcb = pthread->tcb;
|
|
|
|
|
|
|
|
|
|
/* And initialize the rest of the machine context. This may include
|
|
|
|
|
additional machine- and system-specific initializations that
|
|
|
|
|
prove convenient. */
|
|
|
|
|
err = __pthread_setup (pthread, entry_point, start_routine, arg);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed_setup;
|
|
|
|
|
|
|
|
|
|
/* Initialize the system-specific signal state for the new
|
|
|
|
|
thread. */
|
|
|
|
|
err = __pthread_sigstate_init (pthread);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed_sigstate;
|
|
|
|
|
|
|
|
|
|
/* If the new thread is joinable, add a reference for the caller. */
|
|
|
|
|
if (pthread->state == PTHREAD_JOINABLE)
|
|
|
|
|
pthread->nr_refs++;
|
|
|
|
|
|
|
|
|
|
/* Set the new thread's signal mask and set the pending signals to
|
|
|
|
|
empty. POSIX says: "The signal mask shall be inherited from the
|
|
|
|
|
creating thread. The set of signals pending for the new thread
|
|
|
|
|
shall be empty." If the currnet thread is not a pthread then we
|
|
|
|
|
just inherit the process' sigmask. */
|
|
|
|
|
if (__pthread_num_threads == 1)
|
|
|
|
|
err = sigprocmask (0, 0, &sigset);
|
|
|
|
|
else
|
|
|
|
|
err = __pthread_sigstate (_pthread_self (), 0, 0, &sigset, 0);
|
|
|
|
|
assert_perror (err);
|
|
|
|
|
|
|
|
|
|
err = __pthread_sigstate (pthread, SIG_SETMASK, &sigset, 0, 1);
|
|
|
|
|
assert_perror (err);
|
|
|
|
|
|
|
|
|
|
/* Increase the total number of threads. We do this before actually
|
|
|
|
|
starting the new thread, since the new thread might immediately
|
|
|
|
|
call `pthread_exit' which decreases the number of threads and
|
|
|
|
|
calls `exit' if the number of threads reaches zero. Increasing
|
|
|
|
|
the number of threads from within the new thread isn't an option
|
|
|
|
|
since this thread might return and call `pthread_exit' before the
|
|
|
|
|
new thread runs. */
|
|
|
|
|
atomic_increment (&__pthread_total);
|
|
|
|
|
|
|
|
|
|
/* Store a pointer to this thread in the thread ID lookup table. We
|
|
|
|
|
could use __thread_setid, however, we only lock for reading as no
|
|
|
|
|
other thread should be using this entry (we also assume that the
|
|
|
|
|
store is atomic). */
|
|
|
|
|
__pthread_rwlock_rdlock (&__pthread_threads_lock);
|
|
|
|
|
__pthread_threads[pthread->thread - 1] = pthread;
|
|
|
|
|
__pthread_rwlock_unlock (&__pthread_threads_lock);
|
|
|
|
|
|
|
|
|
|
/* At this point it is possible to guess our pthread ID. We have to
|
|
|
|
|
make sure that all functions taking a pthread_t argument can
|
|
|
|
|
handle the fact that this thread isn't really running yet. Since
|
|
|
|
|
the new thread might be passed its ID through pthread_create (to
|
|
|
|
|
avoid calling pthread_self), read it before starting the thread. */
|
|
|
|
|
*thread = pthread;
|
|
|
|
|
|
|
|
|
|
/* Schedule the new thread. */
|
|
|
|
|
err = __pthread_thread_start (pthread);
|
|
|
|
|
if (err)
|
|
|
|
|
goto failed_starting;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
failed_starting:
|
|
|
|
|
/* If joinable, a reference was added for the caller. */
|
|
|
|
|
if (pthread->state == PTHREAD_JOINABLE)
|
|
|
|
|
__pthread_dealloc (pthread);
|
|
|
|
|
|
|
|
|
|
__pthread_setid (pthread->thread, NULL);
|
|
|
|
|
atomic_decrement (&__pthread_total);
|
|
|
|
|
failed_sigstate:
|
|
|
|
|
__pthread_sigstate_destroy (pthread);
|
|
|
|
|
failed_setup:
|
|
|
|
|
_dl_deallocate_tls (pthread->tcb, 1);
|
|
|
|
|
pthread->tcb = NULL;
|
|
|
|
|
failed_thread_tls_alloc:
|
|
|
|
|
__pthread_thread_terminate (pthread);
|
|
|
|
|
|
|
|
|
|
/* __pthread_thread_terminate has taken care of deallocating the stack and
|
|
|
|
|
the thread structure. */
|
|
|
|
|
goto failed;
|
|
|
|
|
failed_thread_alloc:
|
|
|
|
|
if (pthread->stack)
|
|
|
|
|
__pthread_stack_dealloc (pthread->stackaddr,
|
|
|
|
|
((setup->__guardsize + __vm_page_size - 1)
|
|
|
|
|
/ __vm_page_size) * __vm_page_size + stacksize);
|
|
|
|
|
failed_stack_alloc:
|
|
|
|
|
__pthread_dealloc (pthread);
|
|
|
|
|
failed:
|
|
|
|
|
return err;
|
|
|
|
|
}
|