2 This file is part of PulseAudio.
4 Copyright 2004-2006 Lennart Poettering
5 Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB
7 PulseAudio is free software; you can redistribute it and/or modify
8 it under the terms of the GNU Lesser General Public License as
9 published by the Free Software Foundation; either version 2.1 of the
10 License, or (at your option) any later version.
12 PulseAudio is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public
18 License along with PulseAudio; if not, write to the Free Software
19 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
32 #ifdef HAVE_SYS_PRCTL_H
33 #include <sys/prctl.h>
37 #include <CoreServices/CoreServices.h>
38 #include <mach/mach.h>
39 #include <mach/mach_time.h>
42 #include <pulse/timeval.h>
43 #include <pulsecore/macro.h>
44 #include <pulsecore/core-error.h>
46 #include "core-rtclock.h"
48 pa_usec_t
pa_rtclock_age(const struct timeval
*tv
) {
52 return pa_timeval_diff(pa_rtclock_get(&now
), tv
);
55 struct timeval
*pa_rtclock_get(struct timeval
*tv
) {
57 #if defined(HAVE_CLOCK_GETTIME)
60 #ifdef CLOCK_MONOTONIC
61 /* No locking or atomic ops for no_monotonic here */
62 static pa_bool_t no_monotonic
= FALSE
;
65 if (clock_gettime(CLOCK_MONOTONIC
, &ts
) < 0)
69 #endif /* CLOCK_MONOTONIC */
70 pa_assert_se(clock_gettime(CLOCK_REALTIME
, &ts
) == 0);
74 tv
->tv_sec
= ts
.tv_sec
;
75 tv
->tv_usec
= ts
.tv_nsec
/ PA_NSEC_PER_USEC
;
79 #elif defined(OS_IS_DARWIN)
80 static mach_timebase_info_data_t tbi
;
84 /* Refer Apple ADC QA1398
85 Also: http://devworld.apple.com/documentation/Darwin/Conceptual/KernelProgramming/services/services.html
87 Note: argument is timespec NOT timeval (timespec uses nsec, timeval uses usec)
90 /* try and be a mite efficient - maybe I should keep the N/D as a float !? */
92 mach_timebase_info(&tbi
);
94 nticks
= mach_absolute_time();
95 time_nsec
= nticks
* tbi
.numer
/ tbi
.denom
; // see above
97 tv
->tv_sec
= time_nsec
/ PA_NSEC_PER_SEC
;
98 tv
->tv_usec
= time_nsec
/ PA_NSEC_PER_USEC
;
102 #else /* OS_IS_DARWIN */
104 return pa_gettimeofday(tv
);
109 pa_bool_t
pa_rtclock_hrtimer(void) {
111 #if defined(HAVE_CLOCK_GETTIME)
114 #ifdef CLOCK_MONOTONIC
116 if (clock_getres(CLOCK_MONOTONIC
, &ts
) >= 0)
117 return ts
.tv_sec
== 0 && ts
.tv_nsec
<= (long) (PA_HRTIMER_THRESHOLD_USEC
*PA_NSEC_PER_USEC
);
118 #endif /* CLOCK_MONOTONIC */
120 pa_assert_se(clock_getres(CLOCK_REALTIME
, &ts
) == 0);
121 return ts
.tv_sec
== 0 && ts
.tv_nsec
<= (long) (PA_HRTIMER_THRESHOLD_USEC
*PA_NSEC_PER_USEC
);
123 #elif defined (OS_IS_DARWIN)
124 mach_timebase_info_data_t tbi
;
127 mach_timebase_info(&tbi
);
129 /* nsec = nticks * (N/D) - we want 1 tick == resolution !? */
130 time_nsec
= tbi
.numer
/ tbi
.denom
;
131 return time_nsec
<= (long) (PA_HRTIMER_THRESHOLD_USEC
*PA_NSEC_PER_USEC
);
133 #else /* OS_IS_DARWIN */
139 #define TIMER_SLACK_NS (int) ((500 * PA_NSEC_PER_USEC))
141 void pa_rtclock_hrtimer_enable(void) {
143 #ifdef PR_SET_TIMERSLACK
146 if ((slack_ns
= prctl(PR_GET_TIMERSLACK
, 0, 0, 0, 0)) < 0) {
147 pa_log_info("PR_GET_TIMERSLACK/PR_SET_TIMERSLACK not supported.");
151 pa_log_debug("Timer slack is set to %i us.", (int) (slack_ns
/PA_NSEC_PER_USEC
));
153 if (slack_ns
> TIMER_SLACK_NS
) {
154 slack_ns
= TIMER_SLACK_NS
;
156 pa_log_debug("Setting timer slack to %i us.", (int) (slack_ns
/PA_NSEC_PER_USEC
));
158 if (prctl(PR_SET_TIMERSLACK
, slack_ns
, 0, 0, 0) < 0) {
159 pa_log_warn("PR_SET_TIMERSLACK failed: %s", pa_cstrerror(errno
));
167 struct timeval
* pa_rtclock_from_wallclock(struct timeval
*tv
) {
169 #ifdef HAVE_CLOCK_GETTIME
170 struct timeval wc_now
, rt_now
;
172 pa_gettimeofday(&wc_now
);
173 pa_rtclock_get(&rt_now
);
177 /* pa_timeval_sub() saturates on underflow! */
179 if (pa_timeval_cmp(&wc_now
, tv
) < 0)
180 pa_timeval_add(&rt_now
, pa_timeval_diff(tv
, &wc_now
));
182 pa_timeval_sub(&rt_now
, pa_timeval_diff(&wc_now
, tv
));
190 pa_usec_t
pa_timespec_load(const struct timespec
*ts
) {
192 if (PA_UNLIKELY(!ts
))
193 return PA_USEC_INVALID
;
196 (pa_usec_t
) ts
->tv_sec
* PA_USEC_PER_SEC
+
197 (pa_usec_t
) ts
->tv_nsec
/ PA_NSEC_PER_USEC
;
200 struct timespec
* pa_timespec_store(struct timespec
*ts
, pa_usec_t v
) {
203 if (PA_UNLIKELY(v
== PA_USEC_INVALID
)) {
204 ts
->tv_sec
= PA_INT_TYPE_MAX(time_t);
205 ts
->tv_nsec
= (long) (PA_NSEC_PER_SEC
-1);
209 ts
->tv_sec
= (time_t) (v
/ PA_USEC_PER_SEC
);
210 ts
->tv_nsec
= (long) ((v
% PA_USEC_PER_SEC
) * PA_NSEC_PER_USEC
);
215 static struct timeval
* wallclock_from_rtclock(struct timeval
*tv
) {
217 #ifdef HAVE_CLOCK_GETTIME
218 struct timeval wc_now
, rt_now
;
220 pa_gettimeofday(&wc_now
);
221 pa_rtclock_get(&rt_now
);
225 /* pa_timeval_sub() saturates on underflow! */
227 if (pa_timeval_cmp(&rt_now
, tv
) < 0)
228 pa_timeval_add(&wc_now
, pa_timeval_diff(tv
, &rt_now
));
230 pa_timeval_sub(&wc_now
, pa_timeval_diff(&rt_now
, tv
));
238 struct timeval
* pa_timeval_rtstore(struct timeval
*tv
, pa_usec_t v
, pa_bool_t rtclock
) {
241 if (v
== PA_USEC_INVALID
)
244 pa_timeval_store(tv
, v
);
247 tv
->tv_usec
|= PA_TIMEVAL_RTCLOCK
;
249 wallclock_from_rtclock(tv
);