Blame htl/pt-exit.c

Packit Service 82fcde
/* Thread termination.
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
Packit Service 82fcde
#include <pt-internal.h>
Packit Service 82fcde
#include <pthreadP.h>
Packit Service 82fcde
Packit Service 82fcde
#include <atomic.h>
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* Terminate the current thread and make STATUS available to any
Packit Service 82fcde
   thread that might join it.  */
Packit Service 82fcde
void
Packit Service 82fcde
__pthread_exit (void *status)
Packit Service 82fcde
{
Packit Service 82fcde
  struct __pthread *self = _pthread_self ();
Packit Service 82fcde
  struct __pthread_cancelation_handler **handlers;
Packit Service 82fcde
  int oldstate;
Packit Service 82fcde
Packit Service 82fcde
  /* Run any cancelation handlers.  According to POSIX, the
Packit Service 82fcde
     cancellation cleanup handlers should be called with cancellation
Packit Service 82fcde
     disabled.  */
Packit Service 82fcde
  __pthread_setcancelstate (PTHREAD_CANCEL_DISABLE, &oldstate);
Packit Service 82fcde
Packit Service 82fcde
  for (handlers = ___pthread_get_cleanup_stack ();
Packit Service 82fcde
       *handlers != NULL;
Packit Service 82fcde
       *handlers = (*handlers)->__next)
Packit Service 82fcde
    (*handlers)->__handler ((*handlers)->__arg);
Packit Service 82fcde
Packit Service 82fcde
  __pthread_setcancelstate (oldstate, &oldstate);
Packit Service 82fcde
Packit Service 82fcde
  /* Decrease the number of threads.  We use an atomic operation to
Packit Service 82fcde
     make sure that only the last thread calls `exit'.  */
Packit Service 82fcde
  if (atomic_decrement_and_test (&__pthread_total))
Packit Service 82fcde
    /* We are the last thread.  */
Packit Service 82fcde
    exit (0);
Packit Service 82fcde
Packit Service 82fcde
  /* Note that after this point the process can be terminated at any
Packit Service 82fcde
     point if another thread calls `pthread_exit' and happens to be
Packit Service 82fcde
     the last thread.  */
Packit Service 82fcde
Packit Service 82fcde
  __pthread_mutex_lock (&self->state_lock);
Packit Service 82fcde
Packit Service 82fcde
  if (self->cancel_state == PTHREAD_CANCEL_ENABLE && self->cancel_pending)
Packit Service 82fcde
    status = PTHREAD_CANCELED;
Packit Service 82fcde
Packit Service 82fcde
  switch (self->state)
Packit Service 82fcde
    {
Packit Service 82fcde
    default:
Packit Service 82fcde
      assert (!"Consistency error: unexpected self->state");
Packit Service 82fcde
      abort ();
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case PTHREAD_DETACHED:
Packit Service 82fcde
      __pthread_mutex_unlock (&self->state_lock);
Packit Service 82fcde
Packit Service 82fcde
      break;
Packit Service 82fcde
Packit Service 82fcde
    case PTHREAD_JOINABLE:
Packit Service 82fcde
      /* We need to stay around for a while since another thread
Packit Service 82fcde
         might want to join us.  */
Packit Service 82fcde
      self->state = PTHREAD_EXITED;
Packit Service 82fcde
Packit Service 82fcde
      /* We need to remember the exit status.  A thread joining us
Packit Service 82fcde
         might ask for it.  */
Packit Service 82fcde
      self->status = status;
Packit Service 82fcde
Packit Service 82fcde
      /* Broadcast the condition.  This will wake up threads that are
Packit Service 82fcde
         waiting to join us.  */
Packit Service 82fcde
      __pthread_cond_broadcast (&self->state_cond);
Packit Service 82fcde
      __pthread_mutex_unlock (&self->state_lock);
Packit Service 82fcde
Packit Service 82fcde
      break;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Destroy any thread specific data.  */
Packit Service 82fcde
  __pthread_destroy_specific (self);
Packit Service 82fcde
Packit Service 82fcde
  /* Destroy any signal state.  */
Packit Service 82fcde
  __pthread_sigstate_destroy (self);
Packit Service 82fcde
Packit Service 82fcde
  /* Self terminating requires TLS, so defer the release of the TCB until
Packit Service 82fcde
     the thread structure is reused.  */
Packit Service 82fcde
Packit Service 82fcde
  /* Release kernel resources, including the kernel thread and the stack,
Packit Service 82fcde
     and drop the self reference.  */
Packit Service 82fcde
  __pthread_thread_terminate (self);
Packit Service 82fcde
Packit Service 82fcde
  /* NOTREACHED */
Packit Service 82fcde
  abort ();
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
strong_alias (__pthread_exit, pthread_exit);