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
34 #include <pulsecore/macro.h>
35 #include <pulsecore/core-util.h>
39 struct timeval
*pa_gettimeofday(struct timeval
*tv
) {
42 #if defined(OS_IS_WIN32)
44 * Copied from implementation by Steven Edwards (LGPL).
45 * Found on wine mailing list.
47 #if defined(_MSC_VER) || defined(__BORLANDC__)
48 #define EPOCHFILETIME (116444736000000000i64)
50 #define EPOCHFILETIME (116444736000000000LL)
56 GetSystemTimeAsFileTime(&ft
);
57 li
.LowPart
= ft
.dwLowDateTime
;
58 li
.HighPart
= ft
.dwHighDateTime
;
59 t
= li
.QuadPart
; /* In 100-nanosecond intervals */
60 t
-= EPOCHFILETIME
; /* Offset to the Epoch time */
61 t
/= 10; /* In microseconds */
62 tv
->tv_sec
= (time_t) (t
/ PA_USEC_PER_SEC
);
63 tv
->tv_usec
= (suseconds_t
) (t
% PA_USEC_PER_SEC
);
64 #elif defined(HAVE_GETTIMEOFDAY)
65 pa_assert_se(gettimeofday(tv
, NULL
) == 0);
67 #error "Platform lacks gettimeofday() or equivalent function."
73 pa_usec_t
pa_timeval_diff(const struct timeval
*a
, const struct timeval
*b
) {
79 /* Check which whan is the earlier time and swap the two arguments if required. */
80 if (PA_UNLIKELY(pa_timeval_cmp(a
, b
) < 0)) {
81 const struct timeval
*c
;
87 /* Calculate the second difference*/
88 r
= ((pa_usec_t
) a
->tv_sec
- (pa_usec_t
) b
->tv_sec
) * PA_USEC_PER_SEC
;
90 /* Calculate the microsecond difference */
91 if (a
->tv_usec
> b
->tv_usec
)
92 r
+= (pa_usec_t
) a
->tv_usec
- (pa_usec_t
) b
->tv_usec
;
93 else if (a
->tv_usec
< b
->tv_usec
)
94 r
-= (pa_usec_t
) b
->tv_usec
- (pa_usec_t
) a
->tv_usec
;
99 int pa_timeval_cmp(const struct timeval
*a
, const struct timeval
*b
) {
103 if (a
->tv_sec
< b
->tv_sec
)
106 if (a
->tv_sec
> b
->tv_sec
)
109 if (a
->tv_usec
< b
->tv_usec
)
112 if (a
->tv_usec
> b
->tv_usec
)
118 pa_usec_t
pa_timeval_age(const struct timeval
*tv
) {
122 return pa_timeval_diff(pa_gettimeofday(&now
), tv
);
125 struct timeval
* pa_timeval_add(struct timeval
*tv
, pa_usec_t v
) {
129 secs
= (time_t) (v
/PA_USEC_PER_SEC
);
131 if (PA_UNLIKELY(tv
->tv_sec
> PA_INT_TYPE_MAX(time_t) - secs
))
135 v
-= (pa_usec_t
) secs
* PA_USEC_PER_SEC
;
136 tv
->tv_usec
+= (suseconds_t
) v
;
139 while ((pa_usec_t
) tv
->tv_usec
>= PA_USEC_PER_SEC
) {
141 if (PA_UNLIKELY(tv
->tv_sec
>= PA_INT_TYPE_MAX(time_t)))
145 tv
->tv_usec
-= (suseconds_t
) PA_USEC_PER_SEC
;
151 tv
->tv_sec
= PA_INT_TYPE_MAX(time_t);
152 tv
->tv_usec
= (suseconds_t
) (PA_USEC_PER_SEC
-1);
156 struct timeval
* pa_timeval_sub(struct timeval
*tv
, pa_usec_t v
) {
160 secs
= (time_t) (v
/PA_USEC_PER_SEC
);
162 if (PA_UNLIKELY(tv
->tv_sec
< secs
))
166 v
-= (pa_usec_t
) secs
* PA_USEC_PER_SEC
;
168 if (tv
->tv_usec
>= (suseconds_t
) v
)
169 tv
->tv_usec
-= (suseconds_t
) v
;
172 if (PA_UNLIKELY(tv
->tv_sec
<= 0))
176 tv
->tv_usec
+= (suseconds_t
) (PA_USEC_PER_SEC
- v
);
187 struct timeval
* pa_timeval_store(struct timeval
*tv
, pa_usec_t v
) {
190 if (PA_UNLIKELY(v
== PA_USEC_INVALID
)) {
191 tv
->tv_sec
= PA_INT_TYPE_MAX(time_t);
192 tv
->tv_usec
= (suseconds_t
) (PA_USEC_PER_SEC
-1);
197 tv
->tv_sec
= (time_t) (v
/ PA_USEC_PER_SEC
);
198 tv
->tv_usec
= (suseconds_t
) (v
% PA_USEC_PER_SEC
);
203 pa_usec_t
pa_timeval_load(const struct timeval
*tv
) {
205 if (PA_UNLIKELY(!tv
))
206 return PA_USEC_INVALID
;
209 (pa_usec_t
) tv
->tv_sec
* PA_USEC_PER_SEC
+
210 (pa_usec_t
) tv
->tv_usec
;