Blame sysdeps/htl/pt-key-create.c

Packit 6c4009
/* pthread_key_create.  Hurd version.
Packit 6c4009
   Copyright (C) 2002-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library;  if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <pthread.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <assert.h>
Packit 6c4009
Packit 6c4009
#include <pt-internal.h>
Packit 6c4009
#include <pthreadP.h>
Packit 6c4009
Packit 6c4009
pthread_mutex_t __pthread_key_lock;
Packit 6c4009
Packit 6c4009
void (**__pthread_key_destructors) (void *arg);
Packit 6c4009
int __pthread_key_size;
Packit 6c4009
int __pthread_key_count;
Packit 6c4009
int __pthread_key_invalid_count;
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
__pthread_key_create (pthread_key_t *key, void (*destructor) (void *))
Packit 6c4009
{
Packit 6c4009
  /* Where to look for the next key slot.  */
Packit 6c4009
  static int index;
Packit 6c4009
Packit 6c4009
  __pthread_key_lock_ready ();
Packit 6c4009
Packit 6c4009
  __pthread_mutex_lock (&__pthread_key_lock);
Packit 6c4009
Packit 6c4009
do_search:
Packit 6c4009
  /* Use the search hint and try to find a free slot.  */
Packit 6c4009
  for (; index < __pthread_key_count
Packit 6c4009
       && __pthread_key_destructors[index] != PTHREAD_KEY_INVALID; index++)
Packit 6c4009
    ;
Packit 6c4009
Packit 6c4009
  /* See if we actually found a free element.  */
Packit 6c4009
  if (index < __pthread_key_count)
Packit 6c4009
    {
Packit 6c4009
      assert (__pthread_key_destructors[index] == PTHREAD_KEY_INVALID);
Packit 6c4009
      assert (__pthread_key_invalid_count > 0);
Packit 6c4009
Packit 6c4009
      __pthread_key_invalid_count--;
Packit 6c4009
      __pthread_key_destructors[index] = destructor;
Packit 6c4009
      *key = index++;
Packit 6c4009
Packit 6c4009
      __pthread_mutex_unlock (&__pthread_key_lock);
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  assert (index == __pthread_key_count);
Packit 6c4009
Packit 6c4009
  /* No space at the end.  */
Packit 6c4009
  if (__pthread_key_size == __pthread_key_count)
Packit 6c4009
    {
Packit 6c4009
      /* See if it is worth looking for a free element.  */
Packit 6c4009
      if (__pthread_key_invalid_count > 4
Packit 6c4009
	  && __pthread_key_invalid_count > __pthread_key_size / 8)
Packit 6c4009
	{
Packit 6c4009
	  index = 0;
Packit 6c4009
	  goto do_search;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
Packit 6c4009
      /* Resize the array.  */
Packit 6c4009
      {
Packit 6c4009
	void *t;
Packit 6c4009
	int newsize;
Packit 6c4009
Packit 6c4009
	if (__pthread_key_size == 0)
Packit 6c4009
	  newsize = 8;
Packit 6c4009
	else
Packit 6c4009
	  newsize = __pthread_key_size * 2;
Packit 6c4009
Packit 6c4009
	t = realloc (__pthread_key_destructors,
Packit 6c4009
		     newsize * sizeof (*__pthread_key_destructors));
Packit 6c4009
	if (t == NULL)
Packit 6c4009
	  {
Packit 6c4009
	    __pthread_mutex_unlock (&__pthread_key_lock);
Packit 6c4009
	    return ENOMEM;
Packit 6c4009
	  }
Packit 6c4009
Packit 6c4009
	__pthread_key_size = newsize;
Packit 6c4009
	__pthread_key_destructors = t;
Packit 6c4009
      }
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  __pthread_key_destructors[index] = destructor;
Packit 6c4009
  *key = index;
Packit 6c4009
Packit 6c4009
  index++;
Packit 6c4009
  __pthread_key_count++;
Packit 6c4009
Packit 6c4009
  __pthread_mutex_unlock (&__pthread_key_lock);
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
strong_alias (__pthread_key_create, pthread_key_create)
Packit 6c4009
hidden_def (__pthread_key_create)