ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/cebix/BasiliskII/src/Unix/timer_unix.cpp
Revision: 1.14
Committed: 2004-01-12T15:29:25Z (20 years, 6 months ago) by cebix
Branch: MAIN
CVS Tags: nigel-build-16, nigel-build-15
Changes since 1.13: +1 -1 lines
Log Message:
Happy New Year! :)

File Contents

# Content
1 /*
2 * timer_unix.cpp - Time Manager emulation, Unix specific stuff
3 *
4 * Basilisk II (C) 1997-2004 Christian Bauer
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program 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
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 */
20
21 #include "sysdeps.h"
22 #include "macos_util.h"
23 #include "timer.h"
24
25 #include <errno.h>
26
27 #define DEBUG 0
28 #include "debug.h"
29
30 // For NetBSD with broken pthreads headers
31 #ifndef CLOCK_REALTIME
32 #define CLOCK_REALTIME 0
33 #endif
34
35
36 /*
37 * Return microseconds since boot (64 bit)
38 */
39
40 void Microseconds(uint32 &hi, uint32 &lo)
41 {
42 D(bug("Microseconds\n"));
43 #ifdef HAVE_CLOCK_GETTIME
44 struct timespec t;
45 clock_gettime(CLOCK_REALTIME, &t);
46 uint64 tl = (uint64)t.tv_sec * 1000000 + t.tv_nsec / 1000;
47 #else
48 struct timeval t;
49 gettimeofday(&t, NULL);
50 uint64 tl = (uint64)t.tv_sec * 1000000 + t.tv_usec;
51 #endif
52 hi = tl >> 32;
53 lo = tl;
54 }
55
56
57 /*
58 * Return local date/time in Mac format (seconds since 1.1.1904)
59 */
60
61 uint32 TimerDateTime(void)
62 {
63 return TimeToMacTime(time(NULL));
64 }
65
66
67 /*
68 * Get current time
69 */
70
71 void timer_current_time(tm_time_t &t)
72 {
73 #ifdef HAVE_CLOCK_GETTIME
74 clock_gettime(CLOCK_REALTIME, &t);
75 #else
76 gettimeofday(&t, NULL);
77 #endif
78 }
79
80
81 /*
82 * Add times
83 */
84
85 void timer_add_time(tm_time_t &res, tm_time_t a, tm_time_t b)
86 {
87 #ifdef HAVE_CLOCK_GETTIME
88 res.tv_sec = a.tv_sec + b.tv_sec;
89 res.tv_nsec = a.tv_nsec + b.tv_nsec;
90 if (res.tv_nsec >= 1000000000) {
91 res.tv_sec++;
92 res.tv_nsec -= 1000000000;
93 }
94 #else
95 res.tv_sec = a.tv_sec + b.tv_sec;
96 res.tv_usec = a.tv_usec + b.tv_usec;
97 if (res.tv_usec >= 1000000) {
98 res.tv_sec++;
99 res.tv_usec -= 1000000;
100 }
101 #endif
102 }
103
104
105 /*
106 * Subtract times
107 */
108
109 void timer_sub_time(tm_time_t &res, tm_time_t a, tm_time_t b)
110 {
111 #ifdef HAVE_CLOCK_GETTIME
112 res.tv_sec = a.tv_sec - b.tv_sec;
113 res.tv_nsec = a.tv_nsec - b.tv_nsec;
114 if (res.tv_nsec < 0) {
115 res.tv_sec--;
116 res.tv_nsec += 1000000000;
117 }
118 #else
119 res.tv_sec = a.tv_sec - b.tv_sec;
120 res.tv_usec = a.tv_usec - b.tv_usec;
121 if (res.tv_usec < 0) {
122 res.tv_sec--;
123 res.tv_usec += 1000000;
124 }
125 #endif
126 }
127
128
129 /*
130 * Compare times (<0: a < b, =0: a = b, >0: a > b)
131 */
132
133 int timer_cmp_time(tm_time_t a, tm_time_t b)
134 {
135 #ifdef HAVE_CLOCK_GETTIME
136 if (a.tv_sec == b.tv_sec)
137 return a.tv_nsec - b.tv_nsec;
138 else
139 return a.tv_sec - b.tv_sec;
140 #else
141 if (a.tv_sec == b.tv_sec)
142 return a.tv_usec - b.tv_usec;
143 else
144 return a.tv_sec - b.tv_sec;
145 #endif
146 }
147
148
149 /*
150 * Convert Mac time value (>0: microseconds, <0: microseconds) to tm_time_t
151 */
152
153 void timer_mac2host_time(tm_time_t &res, int32 mactime)
154 {
155 #ifdef HAVE_CLOCK_GETTIME
156 if (mactime > 0) {
157 // Time in milliseconds
158 res.tv_sec = mactime / 1000;
159 res.tv_nsec = (mactime % 1000) * 1000000;
160 } else {
161 // Time in negative microseconds
162 res.tv_sec = -mactime / 1000000;
163 res.tv_nsec = (-mactime % 1000000) * 1000;
164 }
165 #else
166 if (mactime > 0) {
167 // Time in milliseconds
168 res.tv_sec = mactime / 1000;
169 res.tv_usec = (mactime % 1000) * 1000;
170 } else {
171 // Time in negative microseconds
172 res.tv_sec = -mactime / 1000000;
173 res.tv_usec = -mactime % 1000000;
174 }
175 #endif
176 }
177
178
179 /*
180 * Convert positive tm_time_t to Mac time value (>0: microseconds, <0: microseconds)
181 * A negative input value for hosttime results in a zero return value
182 * As long as the microseconds value fits in 32 bit, it must not be converted to milliseconds!
183 */
184
185 int32 timer_host2mac_time(tm_time_t hosttime)
186 {
187 if (hosttime.tv_sec < 0)
188 return 0;
189 else {
190 #ifdef HAVE_CLOCK_GETTIME
191 uint64 t = (uint64)hosttime.tv_sec * 1000000 + hosttime.tv_nsec / 1000;
192 #else
193 uint64 t = (uint64)hosttime.tv_sec * 1000000 + hosttime.tv_usec;
194 #endif
195 if (t > 0x7fffffff)
196 return t / 1000; // Time in milliseconds
197 else
198 return -t; // Time in negative microseconds
199 }
200 }
201
202
203 /*
204 * Get current value of microsecond timer
205 */
206
207 uint64 GetTicks_usec(void)
208 {
209 #ifdef HAVE_CLOCK_GETTIME
210 struct timespec t;
211 clock_gettime(CLOCK_REALTIME, &t);
212 return (uint64)t.tv_sec * 1000000 + t.tv_nsec / 1000;
213 #else
214 struct timeval t;
215 gettimeofday(&t, NULL);
216 return (uint64)t.tv_sec * 1000000 + t.tv_usec;
217 #endif
218 }
219
220
221 /*
222 * Delay by specified number of microseconds (<1 second)
223 * (adapted from SDL_Delay() source; this function is designed to provide
224 * the highest accuracy possible)
225 */
226
227 #if defined(linux)
228 // Linux select() changes its timeout parameter upon return to contain
229 // the remaining time. Most other unixen leave it unchanged or undefined.
230 #define SELECT_SETS_REMAINING
231 #elif defined(__FreeBSD__) || defined(__sun__)
232 #define USE_NANOSLEEP
233 #elif defined(HAVE_PTHREADS) && defined(sgi)
234 // SGI pthreads has a bug when using pthreads+signals+nanosleep,
235 // so instead of using nanosleep, wait on a CV which is never signalled.
236 #include <pthread.h>
237 #define USE_COND_TIMEDWAIT
238 #endif
239
240 void Delay_usec(uint32 usec)
241 {
242 int was_error;
243
244 #if defined(USE_NANOSLEEP)
245 struct timespec elapsed, tv;
246 #elif defined(USE_COND_TIMEDWAIT)
247 // Use a local mutex and cv, so threads remain independent
248 pthread_cond_t delay_cond = PTHREAD_COND_INITIALIZER;
249 pthread_mutex_t delay_mutex = PTHREAD_MUTEX_INITIALIZER;
250 struct timespec elapsed;
251 uint64 future;
252 #else
253 struct timeval tv;
254 #ifndef SELECT_SETS_REMAINING
255 uint64 then, now, elapsed;
256 #endif
257 #endif
258
259 // Set the timeout interval - Linux only needs to do this once
260 #if defined(SELECT_SETS_REMAINING)
261 tv.tv_sec = 0;
262 tv.tv_usec = usec;
263 #elif defined(USE_NANOSLEEP)
264 elapsed.tv_sec = 0;
265 elapsed.tv_nsec = usec * 1000;
266 #elif defined(USE_COND_TIMEDWAIT)
267 future = GetTicks_usec() + usec;
268 elapsed.tv_sec = future / 1000000;
269 elapsed.tv_nsec = (future % 1000000) * 1000;
270 #else
271 then = GetTicks_usec();
272 #endif
273
274 do {
275 errno = 0;
276 #if defined(USE_NANOSLEEP)
277 tv.tv_sec = elapsed.tv_sec;
278 tv.tv_nsec = elapsed.tv_nsec;
279 was_error = nanosleep(&tv, &elapsed);
280 #elif defined(USE_COND_TIMEDWAIT)
281 was_error = pthread_mutex_lock(&delay_mutex);
282 was_error = pthread_cond_timedwait(&delay_cond, &delay_mutex, &elapsed);
283 was_error = pthread_mutex_unlock(&delay_mutex);
284 #else
285 #ifndef SELECT_SETS_REMAINING
286 // Calculate the time interval left (in case of interrupt)
287 now = GetTicks_usec();
288 elapsed = now - then;
289 then = now;
290 if (elapsed >= usec)
291 break;
292 usec -= elapsed;
293 tv.tv_sec = 0;
294 tv.tv_usec = usec;
295 #endif
296 was_error = select(0, NULL, NULL, NULL, &tv);
297 #endif
298 } while (was_error && (errno == EINTR));
299 }