Blame nptl/lll_timedwait_tid.c

Packit Service 82fcde
/* Timed waiting for thread death.  Generic futex-using version.
Packit Service 82fcde
   Copyright (C) 2003-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Paul Mackerras <paulus@au.ibm.com>, 2003.
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 <atomic.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <lowlevellock.h>
Packit Service 82fcde
#include <sys/time.h>
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* The kernel notifies a process which uses CLONE_CHILD_CLEARTID via futex
Packit Service 82fcde
   wake-up when the clone terminates.  The memory location contains the
Packit Service 82fcde
   thread ID while the clone is running and is reset to zero by the kernel
Packit Service 82fcde
   afterwards.  The kernel up to version 3.16.3 does not use the private futex
Packit Service 82fcde
   operations for futex wake-up when the clone terminates.  */
Packit Service 82fcde
int
Packit Service 82fcde
__lll_timedwait_tid (int *tidp, const struct timespec *abstime)
Packit Service 82fcde
{
Packit Service 82fcde
  int tid;
Packit Service 82fcde
Packit Service 82fcde
  if (abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000)
Packit Service 82fcde
    return EINVAL;
Packit Service 82fcde
Packit Service 82fcde
  /* Repeat until thread terminated.  */
Packit Service 82fcde
  while ((tid = *tidp) != 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      struct timeval tv;
Packit Service 82fcde
      struct timespec rt;
Packit Service 82fcde
Packit Service 82fcde
      /* Get the current time.  */
Packit Service 82fcde
      (void) __gettimeofday (&tv, NULL);
Packit Service 82fcde
Packit Service 82fcde
      /* Compute relative timeout.  */
Packit Service 82fcde
      rt.tv_sec = abstime->tv_sec - tv.tv_sec;
Packit Service 82fcde
      rt.tv_nsec = abstime->tv_nsec - tv.tv_usec * 1000;
Packit Service 82fcde
      if (rt.tv_nsec < 0)
Packit Service 82fcde
        {
Packit Service 82fcde
          rt.tv_nsec += 1000000000;
Packit Service 82fcde
          --rt.tv_sec;
Packit Service 82fcde
        }
Packit Service 82fcde
Packit Service 82fcde
      /* Already timed out?  */
Packit Service 82fcde
      if (rt.tv_sec < 0)
Packit Service 82fcde
        return ETIMEDOUT;
Packit Service 82fcde
Packit Service 82fcde
      /* If *tidp == tid, wait until thread terminates or the wait times out.
Packit Service 82fcde
         The kernel up to version 3.16.3 does not use the private futex
Packit Service 82fcde
         operations for futex wake-up when the clone terminates.
Packit Service 82fcde
      */
Packit Service 82fcde
      if (lll_futex_timed_wait (tidp, tid, &rt, LLL_SHARED) == -ETIMEDOUT)
Packit Service 82fcde
        return ETIMEDOUT;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  return 0;
Packit Service 82fcde
}