Blame sysdeps/htl/pt-rwlock-timedrdlock.c

Packit Service 82fcde
/* Acquire a rwlock for reading.  Generic version.
Packit Service 82fcde
   Copyright (C) 2002-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
/* Acquire the rwlock *RWLOCK for reading blocking until *ABSTIME if
Packit Service 82fcde
   it is already held.  As a GNU extension, if TIMESPEC is NULL then
Packit Service 82fcde
   wait forever.  */
Packit Service 82fcde
int
Packit Service 82fcde
__pthread_rwlock_timedrdlock_internal (struct __pthread_rwlock *rwlock,
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
Packit Service 82fcde
  __pthread_spin_lock (&rwlock->__lock);
Packit Service 82fcde
  if (__pthread_spin_trylock (&rwlock->__held) == 0)
Packit Service 82fcde
    /* Successfully acquired the lock.  */
Packit Service 82fcde
    {
Packit Service 82fcde
      assert (rwlock->__readerqueue == 0);
Packit Service 82fcde
      assert (rwlock->__writerqueue == 0);
Packit Service 82fcde
      assert (rwlock->__readers == 0);
Packit Service 82fcde
Packit Service 82fcde
      rwlock->__readers = 1;
Packit Service 82fcde
      __pthread_spin_unlock (&rwlock->__lock);
Packit Service 82fcde
      return 0;
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    /* Lock is held, but is held by a reader?  */
Packit Service 82fcde
  if (rwlock->__readers > 0)
Packit Service 82fcde
    /* Just add ourself to number of readers.  */
Packit Service 82fcde
    {
Packit Service 82fcde
      assert (rwlock->__readerqueue == 0);
Packit Service 82fcde
      rwlock->__readers++;
Packit Service 82fcde
      __pthread_spin_unlock (&rwlock->__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
  /* Better be blocked by a writer.  */
Packit Service 82fcde
  assert (rwlock->__readers == 0);
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
  self = _pthread_self ();
Packit Service 82fcde
Packit Service 82fcde
  /* Add ourself to the queue.  */
Packit Service 82fcde
  __pthread_enqueue (&rwlock->__readerqueue, self);
Packit Service 82fcde
  __pthread_spin_unlock (&rwlock->__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 (&rwlock->__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 rwlock 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 (&rwlock->__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
  /* The reader count has already been increment by whoever woke us
Packit Service 82fcde
     up.  */
Packit Service 82fcde
Packit Service 82fcde
  assert (rwlock->__readers > 0);
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
__pthread_rwlock_timedrdlock (struct __pthread_rwlock *rwlock,
Packit Service 82fcde
			      const struct timespec *abstime)
Packit Service 82fcde
{
Packit Service 82fcde
  return __pthread_rwlock_timedrdlock_internal (rwlock, abstime);
Packit Service 82fcde
}
Packit Service 82fcde
weak_alias (__pthread_rwlock_timedrdlock, pthread_rwlock_timedrdlock)