Blame nptl/lll_timedwait_tid.c

Packit 6c4009
/* Timed waiting for thread death.  Generic futex-using version.
Packit 6c4009
   Copyright (C) 2003-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Paul Mackerras <paulus@au.ibm.com>, 2003.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <atomic.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <lowlevellock.h>
Packit 6c4009
#include <sys/time.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* The kernel notifies a process which uses CLONE_CHILD_CLEARTID via futex
Packit 6c4009
   wake-up when the clone terminates.  The memory location contains the
Packit 6c4009
   thread ID while the clone is running and is reset to zero by the kernel
Packit 6c4009
   afterwards.  The kernel up to version 3.16.3 does not use the private futex
Packit 6c4009
   operations for futex wake-up when the clone terminates.  */
Packit 6c4009
int
Packit 6c4009
__lll_timedwait_tid (int *tidp, const struct timespec *abstime)
Packit 6c4009
{
Packit 6c4009
  int tid;
Packit 6c4009
Packit 6c4009
  if (abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000)
Packit 6c4009
    return EINVAL;
Packit 6c4009
Packit 6c4009
  /* Repeat until thread terminated.  */
Packit 6c4009
  while ((tid = *tidp) != 0)
Packit 6c4009
    {
Packit 6c4009
      struct timeval tv;
Packit 6c4009
      struct timespec rt;
Packit 6c4009
Packit 6c4009
      /* Get the current time.  */
Packit 6c4009
      (void) __gettimeofday (&tv, NULL);
Packit 6c4009
Packit 6c4009
      /* Compute relative timeout.  */
Packit 6c4009
      rt.tv_sec = abstime->tv_sec - tv.tv_sec;
Packit 6c4009
      rt.tv_nsec = abstime->tv_nsec - tv.tv_usec * 1000;
Packit 6c4009
      if (rt.tv_nsec < 0)
Packit 6c4009
        {
Packit 6c4009
          rt.tv_nsec += 1000000000;
Packit 6c4009
          --rt.tv_sec;
Packit 6c4009
        }
Packit 6c4009
Packit 6c4009
      /* Already timed out?  */
Packit 6c4009
      if (rt.tv_sec < 0)
Packit 6c4009
        return ETIMEDOUT;
Packit 6c4009
Packit 6c4009
      /* If *tidp == tid, wait until thread terminates or the wait times out.
Packit 6c4009
         The kernel up to version 3.16.3 does not use the private futex
Packit 6c4009
         operations for futex wake-up when the clone terminates.
Packit 6c4009
      */
Packit 6c4009
      if (lll_futex_timed_wait (tidp, tid, &rt, LLL_SHARED) == -ETIMEDOUT)
Packit 6c4009
        return ETIMEDOUT;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}