OSDN Git Service

libpthread/nptl: core of the "Native Posix Threading Library" for uClibc
[uclinux-h8/uClibc.git] / libpthread / nptl / sysdeps / unix / sysv / linux / sem_timedwait.c
1 /* sem_timedwait -- wait on a semaphore.  Generic futex-using version.
2    Copyright (C) 2003 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4    Contributed by Paul Mackerras <paulus@au.ibm.com>, 2003.
5
6    The GNU C Library is free software; you can redistribute it and/or
7    modify it under the terms of the GNU Lesser General Public
8    License as published by the Free Software Foundation; either
9    version 2.1 of the License, or (at your option) any later version.
10
11    The GNU C Library is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    Lesser General Public License for more details.
15
16    You should have received a copy of the GNU Lesser General Public
17    License along with the GNU C Library; if not, write to the Free
18    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
19    02111-1307 USA.  */
20
21 #include <errno.h>
22 #include <sysdep.h>
23 #include <lowlevellock.h>
24 #include <internaltypes.h>
25 #include <semaphore.h>
26 #include <pthreadP.h>
27
28
29 int
30 sem_timedwait (sem_t *sem, const struct timespec *abstime)
31 {
32   /* First check for cancellation.  */
33   CANCELLATION_P (THREAD_SELF);
34
35   int *futex = (int *) sem;
36   int val;
37   int err;
38
39   if (*futex > 0)
40     {
41       val = atomic_decrement_if_positive (futex);
42       if (val > 0)
43         return 0;
44     }
45
46   err = -EINVAL;
47   if (abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000)
48     goto error_return;
49
50   do
51     {
52       struct timeval tv;
53       struct timespec rt;
54       int sec, nsec;
55
56       /* Get the current time.  */
57       gettimeofday (&tv, NULL);
58
59       /* Compute relative timeout.  */
60       sec = abstime->tv_sec - tv.tv_sec;
61       nsec = abstime->tv_nsec - tv.tv_usec * 1000;
62       if (nsec < 0)
63         {
64           nsec += 1000000000;
65           --sec;
66         }
67
68       /* Already timed out?  */
69       err = -ETIMEDOUT;
70       if (sec < 0)
71         goto error_return;
72
73       /* Do wait.  */
74       rt.tv_sec = sec;
75       rt.tv_nsec = nsec;
76
77       /* Enable asynchronous cancellation.  Required by the standard.  */
78       int oldtype = __pthread_enable_asynccancel ();
79
80       err = lll_futex_timed_wait (futex, 0, &rt);
81
82       /* Disable asynchronous cancellation.  */
83       __pthread_disable_asynccancel (oldtype);
84
85       if (err != 0 && err != -EWOULDBLOCK)
86         goto error_return;
87
88       val = atomic_decrement_if_positive (futex);
89     }
90   while (val <= 0);
91
92   return 0;
93
94  error_return:
95   __set_errno (-err);
96   return -1;
97 }