Blame sysdeps/htl/pt-mutex-timedlock.c

Packit Service 82fcde
/* Lock a mutex with a timeout.  Generic version.
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 <pthread.h>
Packit Service 82fcde
#include <assert.h>
Packit Service 82fcde
Packit Service 82fcde
#include <pt-internal.h>
Packit Service 82fcde
Packit Service 82fcde
#define LOSE do { * (int *) 0 = 0; } while (1)
Packit Service 82fcde
Packit Service 82fcde
/* Try to lock MUTEX, block until *ABSTIME if it is already held.  As
Packit Service 82fcde
   a GNU extension, if TIMESPEC is NULL then wait forever.  */
Packit Service 82fcde
int
Packit Service 82fcde
__pthread_mutex_timedlock_internal (struct __pthread_mutex *mutex,
Packit Service 82fcde
				    const struct timespec *abstime)
Packit Service 82fcde
{
Packit Service 82fcde
  error_t err;
Packit Service 82fcde
  int drain;
Packit Service 82fcde
  struct __pthread *self;
Packit Service 82fcde
  const struct __pthread_mutexattr *attr = mutex->__attr;
Packit Service 82fcde
Packit Service 82fcde
  if (attr == __PTHREAD_ERRORCHECK_MUTEXATTR)
Packit Service 82fcde
    attr = &__pthread_errorcheck_mutexattr;
Packit Service 82fcde
  if (attr == __PTHREAD_RECURSIVE_MUTEXATTR)
Packit Service 82fcde
    attr = &__pthread_recursive_mutexattr;
Packit Service 82fcde
Packit Service 82fcde
  __pthread_spin_lock (&mutex->__lock);
Packit Service 82fcde
  if (__pthread_spin_trylock (&mutex->__held) == 0)
Packit Service 82fcde
    /* Successfully acquired the lock.  */
Packit Service 82fcde
    {
Packit Service 82fcde
#ifdef ALWAYS_TRACK_MUTEX_OWNER
Packit Service 82fcde
# ifndef NDEBUG
Packit Service 82fcde
      self = _pthread_self ();
Packit Service 82fcde
      if (self != NULL)
Packit Service 82fcde
	/* The main thread may take a lock before the library is fully
Packit Service 82fcde
	   initialized, in particular, before the main thread has a
Packit Service 82fcde
	   TCB.  */
Packit Service 82fcde
	{
Packit Service 82fcde
	  assert (mutex->__owner == NULL);
Packit Service 82fcde
	  mutex->__owner = _pthread_self ();
Packit Service 82fcde
	}
Packit Service 82fcde
# endif
Packit Service 82fcde
#endif
Packit Service 82fcde
Packit Service 82fcde
      if (attr != NULL)
Packit Service 82fcde
	switch (attr->__mutex_type)
Packit Service 82fcde
	  {
Packit Service 82fcde
	  case PTHREAD_MUTEX_NORMAL:
Packit Service 82fcde
	    break;
Packit Service 82fcde
Packit Service 82fcde
	  case PTHREAD_MUTEX_RECURSIVE:
Packit Service 82fcde
	    mutex->__locks = 1;
Packit Service 82fcde
	  case PTHREAD_MUTEX_ERRORCHECK:
Packit Service 82fcde
	    mutex->__owner = _pthread_self ();
Packit Service 82fcde
	    break;
Packit Service 82fcde
Packit Service 82fcde
	  default:
Packit Service 82fcde
	    LOSE;
Packit Service 82fcde
	  }
Packit Service 82fcde
Packit Service 82fcde
      __pthread_spin_unlock (&mutex->__lock);
Packit Service 82fcde
      return 0;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* The lock is busy.  */
Packit Service 82fcde
Packit Service 82fcde
  self = _pthread_self ();
Packit Service 82fcde
  assert (self);
Packit Service 82fcde
Packit Service 82fcde
  if (attr == NULL || attr->__mutex_type == PTHREAD_MUTEX_NORMAL)
Packit Service 82fcde
    {
Packit Service 82fcde
#if defined(ALWAYS_TRACK_MUTEX_OWNER)
Packit Service 82fcde
      assert (mutex->__owner != self);
Packit Service 82fcde
#endif
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      switch (attr->__mutex_type)
Packit Service 82fcde
	{
Packit Service 82fcde
	case PTHREAD_MUTEX_ERRORCHECK:
Packit Service 82fcde
	  if (mutex->__owner == self)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      __pthread_spin_unlock (&mutex->__lock);
Packit Service 82fcde
	      return EDEADLK;
Packit Service 82fcde
	    }
Packit Service 82fcde
	  break;
Packit Service 82fcde
Packit Service 82fcde
	case PTHREAD_MUTEX_RECURSIVE:
Packit Service 82fcde
	  if (mutex->__owner == self)
Packit Service 82fcde
	    {
Packit Service 82fcde
	      mutex->__locks++;
Packit Service 82fcde
	      __pthread_spin_unlock (&mutex->__lock);
Packit Service 82fcde
	      return 0;
Packit Service 82fcde
	    }
Packit Service 82fcde
	  break;
Packit Service 82fcde
Packit Service 82fcde
	default:
Packit Service 82fcde
	  LOSE;
Packit Service 82fcde
	}
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
#if !defined(ALWAYS_TRACK_MUTEX_OWNER)
Packit Service 82fcde
  if (attr != NULL && attr->__mutex_type != PTHREAD_MUTEX_NORMAL)
Packit Service 82fcde
#endif
Packit Service 82fcde
    assert (mutex->__owner);
Packit Service 82fcde
Packit Service 82fcde
  if (abstime != NULL && (abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000))
Packit Service 82fcde
    return EINVAL;
Packit Service 82fcde
Packit Service 82fcde
  /* Add ourselves to the queue.  */
Packit Service 82fcde
  __pthread_enqueue (&mutex->__queue, self);
Packit Service 82fcde
  __pthread_spin_unlock (&mutex->__lock);
Packit Service 82fcde
Packit Service 82fcde
  /* Block the thread.  */
Packit Service 82fcde
  if (abstime != NULL)
Packit Service 82fcde
    err = __pthread_timedblock (self, abstime, CLOCK_REALTIME);
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      err = 0;
Packit Service 82fcde
      __pthread_block (self);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  __pthread_spin_lock (&mutex->__lock);
Packit Service 82fcde
  if (self->prevp == NULL)
Packit Service 82fcde
    /* Another thread removed us from the queue, which means a wakeup message
Packit Service 82fcde
       has been sent.  It was either consumed while we were blocking, or
Packit Service 82fcde
       queued after we timed out and before we acquired the mutex lock, in
Packit Service 82fcde
       which case the message queue must be drained.  */
Packit Service 82fcde
    drain = err ? 1 : 0;
Packit Service 82fcde
  else
Packit Service 82fcde
    {
Packit Service 82fcde
      /* We're still in the queue.  Noone attempted to wake us up, i.e. we
Packit Service 82fcde
         timed out.  */
Packit Service 82fcde
      __pthread_dequeue (self);
Packit Service 82fcde
      drain = 0;
Packit Service 82fcde
    }
Packit Service 82fcde
  __pthread_spin_unlock (&mutex->__lock);
Packit Service 82fcde
Packit Service 82fcde
  if (drain)
Packit Service 82fcde
    __pthread_block (self);
Packit Service 82fcde
Packit Service 82fcde
  if (err)
Packit Service 82fcde
    {
Packit Service 82fcde
      assert (err == ETIMEDOUT);
Packit Service 82fcde
      return err;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
#if !defined(ALWAYS_TRACK_MUTEX_OWNER)
Packit Service 82fcde
  if (attr != NULL && attr->__mutex_type != PTHREAD_MUTEX_NORMAL)
Packit Service 82fcde
#endif
Packit Service 82fcde
    {
Packit Service 82fcde
      assert (mutex->__owner == self);
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (attr != NULL)
Packit Service 82fcde
    switch (attr->__mutex_type)
Packit Service 82fcde
      {
Packit Service 82fcde
      case PTHREAD_MUTEX_NORMAL:
Packit Service 82fcde
	break;
Packit Service 82fcde
Packit Service 82fcde
      case PTHREAD_MUTEX_RECURSIVE:
Packit Service 82fcde
	assert (mutex->__locks == 0);
Packit Service 82fcde
	mutex->__locks = 1;
Packit Service 82fcde
      case PTHREAD_MUTEX_ERRORCHECK:
Packit Service 82fcde
	mutex->__owner = self;
Packit Service 82fcde
	break;
Packit Service 82fcde
Packit Service 82fcde
      default:
Packit Service 82fcde
	LOSE;
Packit Service 82fcde
      }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
pthread_mutex_timedlock (struct __pthread_mutex *mutex,
Packit Service 82fcde
			 const struct timespec *abstime)
Packit Service 82fcde
{
Packit Service 82fcde
  return __pthread_mutex_timedlock_internal (mutex, abstime);
Packit Service 82fcde
}