Blame sysdeps/htl/sem-timedwait.c

Packit Service 82fcde
/* Wait on a semaphore with a timeout.  Generic version.
Packit Service 82fcde
   Copyright (C) 2005-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 <semaphore.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <assert.h>
Packit Service 82fcde
Packit Service 82fcde
#include <pt-internal.h>
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
__sem_timedwait_internal (sem_t *restrict sem,
Packit Service 82fcde
			  const struct timespec *restrict timeout)
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 (&sem->__lock);
Packit Service 82fcde
  if (sem->__value > 0)
Packit Service 82fcde
    /* Successful down.  */
Packit Service 82fcde
    {
Packit Service 82fcde
      sem->__value--;
Packit Service 82fcde
      __pthread_spin_unlock (&sem->__lock);
Packit Service 82fcde
      return 0;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (timeout != NULL && (timeout->tv_nsec < 0 || timeout->tv_nsec >= 1000000000))
Packit Service 82fcde
    {
Packit Service 82fcde
      errno = EINVAL;
Packit Service 82fcde
      return -1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  /* Add ourselves to the queue.  */
Packit Service 82fcde
  self = _pthread_self ();
Packit Service 82fcde
Packit Service 82fcde
  __pthread_enqueue (&sem->__queue, self);
Packit Service 82fcde
  __pthread_spin_unlock (&sem->__lock);
Packit Service 82fcde
Packit Service 82fcde
  /* Block the thread.  */
Packit Service 82fcde
  if (timeout != NULL)
Packit Service 82fcde
    err = __pthread_timedblock (self, timeout, 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 (&sem->__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 semaphore 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 (&sem->__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
      errno = err;
Packit Service 82fcde
      return -1;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
int
Packit Service 82fcde
__sem_timedwait (sem_t *restrict sem, const struct timespec *restrict timeout)
Packit Service 82fcde
{
Packit Service 82fcde
  return __sem_timedwait_internal (sem, timeout);
Packit Service 82fcde
}
Packit Service 82fcde
Packit Service 82fcde
weak_alias (__sem_timedwait, sem_timedwait);