Blame htl/pt-alloc.c

Packit Service 82fcde
/* Allocate a new thread structure.
Packit Service 82fcde
   Copyright (C) 2000-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library;  if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <assert.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <pthread.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
Packit Service 82fcde
#include <pt-internal.h>
Packit Service 82fcde
Packit Service 82fcde
/* This braindamage is necessary because the standard says that some
Packit Service 82fcde
   of the threads functions "shall fail" if "No thread could be found
Packit Service 82fcde
   corresponding to that specified by the given thread ID."  */
Packit Service 82fcde
Packit Service 82fcde
/* Thread ID lookup table.  */
Packit Service 82fcde
struct __pthread **__pthread_threads;
Packit Service 82fcde
Packit Service 82fcde
/* The size of the thread ID lookup table.  */
Packit Service 82fcde
int __pthread_max_threads;
Packit Service 82fcde
Packit Service 82fcde
/* The total number of thread IDs currently in use, or on the list of
Packit Service 82fcde
   available thread IDs.  */
Packit Service 82fcde
int __pthread_num_threads;
Packit Service 82fcde
Packit Service 82fcde
/* A lock for the table, and the other variables above.  */
Packit Service 82fcde
pthread_rwlock_t __pthread_threads_lock;
Packit Service 82fcde
Packit Service 82fcde
/* List of thread structures corresponding to free thread IDs.  */
Packit Service 82fcde
struct __pthread *__pthread_free_threads;
Packit Service 82fcde
pthread_mutex_t __pthread_free_threads_lock;
Packit Service 82fcde
Packit Service 82fcde
static inline error_t
Packit Service 82fcde
initialize_pthread (struct __pthread *new)
Packit Service 82fcde
{
Packit Service 82fcde
  error_t err;
Packit Service 82fcde
Packit Service 82fcde
  err = __pthread_init_specific (new);
Packit Service 82fcde
  if (err)
Packit Service 82fcde
    return err;
Packit Service 82fcde
Packit Service 82fcde
  new->nr_refs = 1;
Packit Service 82fcde
  new->cancel_lock = (pthread_mutex_t) PTHREAD_MUTEX_INITIALIZER;
Packit Service 82fcde
  new->cancel_hook = NULL;
Packit Service 82fcde
  new->cancel_hook_arg = NULL;
Packit Service 82fcde
  new->cancel_state = PTHREAD_CANCEL_ENABLE;
Packit Service 82fcde
  new->cancel_type = PTHREAD_CANCEL_DEFERRED;
Packit Service 82fcde
  new->cancel_pending = 0;
Packit Service 82fcde
Packit Service 82fcde
  new->state_lock = (pthread_mutex_t) PTHREAD_MUTEX_INITIALIZER;
Packit Service 82fcde
  new->state_cond = (pthread_cond_t) PTHREAD_COND_INITIALIZER;
Packit Service 82fcde
Packit Service 82fcde
  new->cancelation_handlers = 0;
Packit Service 82fcde
Packit Service 82fcde
  memset (&new->res_state, '\0', sizeof (new->res_state));
Packit Service 82fcde
Packit Service 82fcde
  new->tcb = NULL;
Packit Service 82fcde
Packit Service 82fcde
  new->next = 0;
Packit Service 82fcde
  new->prevp = 0;
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* Allocate a new thread structure and its pthread thread ID (but not
Packit Service 82fcde
   a kernel thread).  */
Packit Service 82fcde
int
Packit Service 82fcde
__pthread_alloc (struct __pthread **pthread)
Packit Service 82fcde
{
Packit Service 82fcde
  error_t err;
Packit Service 82fcde
Packit Service 82fcde
  struct __pthread *new;
Packit Service 82fcde
  struct __pthread **threads;
Packit Service 82fcde
  struct __pthread **old_threads;
Packit Service 82fcde
  int max_threads;
Packit Service 82fcde
  int new_max_threads;
Packit Service 82fcde
Packit Service 82fcde
  __pthread_mutex_lock (&__pthread_free_threads_lock);
Packit Service 82fcde
  for (new = __pthread_free_threads; new; new = new->next)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* There is no need to take NEW->STATE_LOCK: if NEW is on this
Packit Service 82fcde
         list, then it is protected by __PTHREAD_FREE_THREADS_LOCK
Packit Service 82fcde
         except in __pthread_dealloc where after it is added to the
Packit Service 82fcde
         list (with the lock held), it drops the lock and then sets
Packit Service 82fcde
         NEW->STATE and immediately stops using NEW.  */
Packit Service 82fcde
      if (new->state == PTHREAD_TERMINATED)
Packit Service 82fcde
	{
Packit Service 82fcde
	  __pthread_dequeue (new);
Packit Service 82fcde
	  break;
Packit Service 82fcde
	}
Packit Service 82fcde
    }
Packit Service 82fcde
  __pthread_mutex_unlock (&__pthread_free_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
  if (new)
Packit Service 82fcde
    {
Packit Service 82fcde
      if (new->tcb)
Packit Service 82fcde
	{
Packit Service 82fcde
	  /* Drop old values */
Packit Service 82fcde
	  _dl_deallocate_tls (new->tcb, 1);
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      err = initialize_pthread (new);
Packit Service 82fcde
      if (!err)
Packit Service 82fcde
	*pthread = new;
Packit Service 82fcde
      return err;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Allocate a new thread structure.  */
Packit Service 82fcde
  new = malloc (sizeof (struct __pthread));
Packit Service 82fcde
  if (new == NULL)
Packit Service 82fcde
    return ENOMEM;
Packit Service 82fcde
Packit Service 82fcde
  err = initialize_pthread (new);
Packit Service 82fcde
  if (err)
Packit Service 82fcde
    {
Packit Service 82fcde
      free (new);
Packit Service 82fcde
      return err;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
retry:
Packit Service 82fcde
  __pthread_rwlock_wrlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
  if (__pthread_num_threads < __pthread_max_threads)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* We have a free slot.  Use the slot number plus one as the
Packit Service 82fcde
         thread ID for the new thread.  */
Packit Service 82fcde
      new->thread = 1 + __pthread_num_threads++;
Packit Service 82fcde
      __pthread_threads[new->thread - 1] = NULL;
Packit Service 82fcde
Packit Service 82fcde
      __pthread_rwlock_unlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
      *pthread = new;
Packit Service 82fcde
      return 0;
Packit Service 82fcde
    }
Packit Service 82fcde
#ifdef PTHREAD_THREADS_MAX
Packit Service 82fcde
  else if (__pthread_num_threads >= PTHREAD_THREADS_MAX)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* We have reached the limit on the number of threads per process.  */
Packit Service 82fcde
      __pthread_rwlock_unlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
      free (new);
Packit Service 82fcde
      return EAGAIN;
Packit Service 82fcde
    }
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
  /* We are going to enlarge the threads table.  Save its current
Packit Service 82fcde
     size.  We're going to release the lock before doing the necessary
Packit Service 82fcde
     memory allocation, since that's a potentially blocking operation.  */
Packit Service 82fcde
  max_threads = __pthread_max_threads;
Packit Service 82fcde
Packit Service 82fcde
  __pthread_rwlock_unlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
  /* Allocate a new lookup table that's twice as large.  */
Packit Service 82fcde
  new_max_threads
Packit Service 82fcde
      = max_threads > 0 ? max_threads * 2 : _POSIX_THREAD_THREADS_MAX;
Packit Service 82fcde
  threads = malloc (new_max_threads * sizeof (struct __pthread *));
Packit Service 82fcde
  if (threads == NULL)
Packit Service 82fcde
    {
Packit Service 82fcde
      free (new);
Packit Service 82fcde
      return ENOMEM;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  __pthread_rwlock_wrlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
  /* Check if nobody else has already enlarged the table.  */
Packit Service 82fcde
  if (max_threads != __pthread_max_threads)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* Yep, they did.  */
Packit Service 82fcde
      __pthread_rwlock_unlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
      /* Free the newly allocated table and try again to allocate a slot.  */
Packit Service 82fcde
      free (threads);
Packit Service 82fcde
      goto retry;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Copy over the contents of the old table.  */
Packit Service 82fcde
  memcpy (threads, __pthread_threads,
Packit Service 82fcde
	  __pthread_max_threads * sizeof (struct __pthread *));
Packit Service 82fcde
Packit Service 82fcde
  /* Save the location of the old table.  We want to deallocate its
Packit Service 82fcde
     storage after we released the lock.  */
Packit Service 82fcde
  old_threads = __pthread_threads;
Packit Service 82fcde
Packit Service 82fcde
  /* Replace the table with the new one.  */
Packit Service 82fcde
  __pthread_max_threads = new_max_threads;
Packit Service 82fcde
  __pthread_threads = threads;
Packit Service 82fcde
Packit Service 82fcde
  /* And allocate ourselves one of the newly created slots.  */
Packit Service 82fcde
  new->thread = 1 + __pthread_num_threads++;
Packit Service 82fcde
  __pthread_threads[new->thread - 1] = NULL;
Packit Service 82fcde
Packit Service 82fcde
  __pthread_rwlock_unlock (&__pthread_threads_lock);
Packit Service 82fcde
Packit Service 82fcde
  free (old_threads);
Packit Service 82fcde
Packit Service 82fcde
  *pthread = new;
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}