semtimedop implementation for Linux/m68k.
[glibc.git] / linuxthreads / sysdeps / pthread / posix-timer.h
blob6710291b9bd27214bc6c1ea63923159fd6aafc5c
1 /* Definitions for POSIX timer implementation on top of LinuxThreads.
2 Copyright (C) 2000, 2002 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4 Contributed by Kaz Kylheku <kaz@ashi.footprints.net>.
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 License as
8 published by the Free Software Foundation; either version 2.1 of the
9 License, or (at your option) any later version.
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.
16 You should have received a copy of the GNU Lesser General Public
17 License along with the GNU C Library; see the file COPYING.LIB. If not,
18 write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 Boston, MA 02111-1307, USA. */
21 #include <limits.h>
22 #include <signal.h>
24 /* Double linked list. */
25 struct list_links
27 struct list_links *next;
28 struct list_links *prev;
32 /* Forward declaration. */
33 struct timer_node;
36 /* Definitions for an internal thread of the POSIX timer implementation. */
37 struct thread_node
39 struct list_links links;
40 pthread_attr_t attr;
41 pthread_t id;
42 unsigned int exists;
43 struct list_links timer_queue;
44 pthread_cond_t cond;
45 struct timer_node *current_timer;
46 pthread_t captured;
47 clockid_t clock_id;
51 /* Internal representation of a timer. */
52 struct timer_node
54 struct list_links links;
55 struct sigevent event;
56 clockid_t clock;
57 struct itimerspec value;
58 struct timespec expirytime;
59 pthread_attr_t attr;
60 unsigned int abstime;
61 unsigned int armed;
62 enum {
63 TIMER_FREE, TIMER_INUSE, TIMER_DELETED
64 } inuse;
65 struct thread_node *thread;
66 pid_t creator_pid;
67 int refcount;
68 int overrun_count;
72 /* Static array with the structures for all the timers. */
73 extern struct timer_node __timer_array[TIMER_MAX];
75 /* Global lock to protect operation on the lists. */
76 extern pthread_mutex_t __timer_mutex;
78 /* Variable to protext initialization. */
79 extern pthread_once_t __timer_init_once_control;
81 /* Nonzero if initialization of timer implementation failed. */
82 extern int __timer_init_failed;
84 /* Nodes for the threads used to deliver signals. */
85 /* A distinct thread is used for each clock type. */
87 extern struct thread_node __timer_signal_thread_rclk;
88 #ifdef _POSIX_CPUTIME
89 extern struct thread_node __timer_signal_thread_pclk;
90 #endif
91 #ifdef _POSIX_THREAD_CPUTIME
92 extern struct thread_node __timer_signal_thread_tclk;
93 #endif
96 /* Return pointer to timer structure corresponding to ID. */
97 static inline struct timer_node *
98 timer_id2ptr (timer_t timerid)
100 if (timerid >= 0 && timerid < TIMER_MAX)
101 return &__timer_array[timerid];
103 return NULL;
106 /* Return ID of TIMER. */
107 static inline int
108 timer_ptr2id (struct timer_node *timer)
110 return timer - __timer_array;
113 /* Check whether timer is valid; global mutex must be held. */
114 static inline int
115 timer_valid (struct timer_node *timer)
117 return timer && timer->inuse == TIMER_INUSE;
120 /* Timer refcount functions; need global mutex. */
121 extern void __timer_dealloc (struct timer_node *timer);
123 static inline void
124 timer_addref (struct timer_node *timer)
126 timer->refcount++;
129 static inline void
130 timer_delref (struct timer_node *timer)
132 if (--timer->refcount == 0)
133 __timer_dealloc (timer);
136 /* Timespec helper routines. */
137 static inline int
138 timespec_compare (const struct timespec *left, const struct timespec *right)
140 if (left->tv_sec < right->tv_sec)
141 return -1;
142 if (left->tv_sec > right->tv_sec)
143 return 1;
145 if (left->tv_nsec < right->tv_nsec)
146 return -1;
147 if (left->tv_nsec > right->tv_nsec)
148 return 1;
150 return 0;
153 static inline void
154 timespec_add (struct timespec *sum, const struct timespec *left,
155 const struct timespec *right)
157 sum->tv_sec = left->tv_sec + right->tv_sec;
158 sum->tv_nsec = left->tv_nsec + right->tv_nsec;
160 if (sum->tv_nsec >= 1000000000)
162 ++sum->tv_sec;
163 sum->tv_nsec -= 1000000000;
167 static inline void
168 timespec_sub (struct timespec *diff, const struct timespec *left,
169 const struct timespec *right)
171 diff->tv_sec = left->tv_sec - right->tv_sec;
172 diff->tv_nsec = left->tv_nsec - right->tv_nsec;
174 if (diff->tv_nsec < 0)
176 --diff->tv_sec;
177 diff->tv_nsec += 1000000000;
182 /* We need one of the list functions in the other modules. */
183 static inline void
184 list_unlink_ip (struct list_links *list)
186 struct list_links *lnext = list->next, *lprev = list->prev;
188 lnext->prev = lprev;
189 lprev->next = lnext;
191 /* The suffix ip means idempotent; list_unlink_ip can be called
192 * two or more times on the same node.
195 list->next = list;
196 list->prev = list;
200 /* Functions in the helper file. */
201 extern void __timer_mutex_cancel_handler (void *arg);
202 extern void __timer_init_once (void);
203 extern struct timer_node *__timer_alloc (void);
204 extern int __timer_thread_start (struct thread_node *thread);
205 extern struct thread_node *__timer_thread_find_matching (const pthread_attr_t *desired_attr, clockid_t);
206 extern struct thread_node *__timer_thread_alloc (const pthread_attr_t *desired_attr, clockid_t);
207 extern void __timer_thread_dealloc (struct thread_node *thread);
208 extern int __timer_thread_queue_timer (struct thread_node *thread,
209 struct timer_node *insert);
210 extern void __timer_thread_wakeup (struct thread_node *thread);