mirror of
https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
synced 2025-11-06 13:29:56 -05:00
add monotonic clock abstraction pa_rtclock
git-svn-id: file:///home/lennart/svn/public/pulseaudio/branches/lennart@1687 fefdeb5f-60dc-0310-8127-8f9354f1896f
This commit is contained in:
parent
6bfeef1818
commit
ef2bc41b87
2 changed files with 201 additions and 0 deletions
153
src/pulsecore/rtclock.c
Normal file
153
src/pulsecore/rtclock.c
Normal file
|
|
@ -0,0 +1,153 @@
|
|||
/* $Id$ */
|
||||
|
||||
/***
|
||||
This file is part of PulseAudio.
|
||||
|
||||
Copyright 2004-2006 Lennart Poettering
|
||||
Copyright 2006 Pierre Ossman <ossman@cendio.se> for Cendio AB
|
||||
|
||||
PulseAudio is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU Lesser General Public License as
|
||||
published by the Free Software Foundation; either version 2.1 of the
|
||||
License, or (at your option) any later version.
|
||||
|
||||
PulseAudio is distributed in the hope that it will be useful, but
|
||||
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with PulseAudio; if not, write to the Free Software
|
||||
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
||||
USA.
|
||||
***/
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
#include <config.h>
|
||||
#endif
|
||||
|
||||
#include <stddef.h>
|
||||
#include <sys/time.h>
|
||||
|
||||
#include <pulse/timeval.h>
|
||||
#include <pulsecore/macro.h>
|
||||
|
||||
#include "rtclock.h"
|
||||
|
||||
struct timespec *pa_timespec_store(struct timespec *a, pa_usec_t u) {
|
||||
pa_assert(a);
|
||||
|
||||
a->tv_sec = u / PA_USEC_PER_SEC;
|
||||
|
||||
u -= (pa_usec_t) a->tv_sec * PA_USEC_PER_SEC;
|
||||
|
||||
a->tv_nsec = u * 1000;
|
||||
|
||||
return a;
|
||||
}
|
||||
|
||||
pa_usec_t pa_timespec_load(struct timespec *ts) {
|
||||
pa_assert(ts);
|
||||
|
||||
return (pa_usec_t) ts->tv_sec * PA_USEC_PER_SEC + (pa_usec_t) (ts->tv_nsec / 1000);
|
||||
}
|
||||
|
||||
pa_usec_t pa_timespec_diff(const struct timespec *a, const struct timespec *b) {
|
||||
pa_usec_t r;
|
||||
|
||||
pa_assert(a);
|
||||
pa_assert(b);
|
||||
|
||||
/* Check which whan is the earlier time and swap the two arguments if required. */
|
||||
if (pa_timespec_cmp(a, b) < 0) {
|
||||
const struct timespec *c;
|
||||
c = a;
|
||||
a = b;
|
||||
b = c;
|
||||
}
|
||||
|
||||
/* Calculate the second difference*/
|
||||
r = ((pa_usec_t) a->tv_sec - b->tv_sec) * PA_USEC_PER_SEC;
|
||||
|
||||
/* Calculate the microsecond difference */
|
||||
if (a->tv_nsec > b->tv_nsec)
|
||||
r += (pa_usec_t) ((a->tv_nsec - b->tv_nsec) / 1000);
|
||||
else if (a->tv_nsec < b->tv_nsec)
|
||||
r -= (pa_usec_t) ((b->tv_nsec - a->tv_nsec) / 1000);
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
int pa_timespec_cmp(const struct timespec *a, const struct timespec *b) {
|
||||
pa_assert(a);
|
||||
pa_assert(b);
|
||||
|
||||
if (a->tv_sec < b->tv_sec)
|
||||
return -1;
|
||||
|
||||
if (a->tv_sec > b->tv_sec)
|
||||
return 1;
|
||||
|
||||
if (a->tv_nsec < b->tv_nsec)
|
||||
return -1;
|
||||
|
||||
if (a->tv_nsec > b->tv_nsec)
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct timespec* pa_timespec_add(struct timespec *ts, pa_usec_t v) {
|
||||
unsigned long secs;
|
||||
pa_assert(ts);
|
||||
|
||||
secs = (unsigned long) (v/PA_USEC_PER_SEC);
|
||||
ts->tv_sec += secs;
|
||||
v -= ((pa_usec_t) secs) * PA_USEC_PER_SEC;
|
||||
|
||||
ts->tv_nsec += (long) (v*1000);
|
||||
|
||||
/* Normalize */
|
||||
while (ts->tv_nsec >= PA_NSEC_PER_SEC) {
|
||||
ts->tv_sec++;
|
||||
ts->tv_nsec -= PA_NSEC_PER_SEC;
|
||||
}
|
||||
|
||||
return ts;
|
||||
}
|
||||
|
||||
pa_usec_t pa_rtclock_age(const struct timespec *ts) {
|
||||
struct timespec now;
|
||||
pa_assert(ts);
|
||||
|
||||
return pa_timespec_diff(pa_rtclock_get(&now), ts);
|
||||
}
|
||||
|
||||
struct timespec *pa_rtclock_get(struct timespec *ts) {
|
||||
static int no_monotonic = 0;
|
||||
|
||||
/* No locking or atomic ops for no_monotonic here */
|
||||
|
||||
pa_assert(ts);
|
||||
|
||||
if (!no_monotonic) {
|
||||
if (clock_gettime(CLOCK_MONOTONIC, ts) >= 0)
|
||||
return ts;
|
||||
|
||||
no_monotonic = 1;
|
||||
}
|
||||
|
||||
pa_assert_se(clock_gettime(CLOCK_REALTIME, ts) == 0);
|
||||
return ts;
|
||||
}
|
||||
|
||||
int pa_rtclock_hrtimer(void) {
|
||||
struct timespec ts;
|
||||
|
||||
if (clock_getres(CLOCK_MONOTONIC, &ts) >= 0)
|
||||
return ts.tv_sec == 0 && ts.tv_nsec <= PA_HRTIMER_THRESHOLD_USEC*1000;
|
||||
|
||||
pa_assert_se(clock_getres(CLOCK_REALTIME, &ts) == 0);
|
||||
return ts.tv_sec == 0 && ts.tv_nsec <= PA_HRTIMER_THRESHOLD_USEC*1000;
|
||||
}
|
||||
|
||||
48
src/pulsecore/rtclock.h
Normal file
48
src/pulsecore/rtclock.h
Normal file
|
|
@ -0,0 +1,48 @@
|
|||
#ifndef foopulsertclockhfoo
|
||||
#define foopulsertclockhfoo
|
||||
|
||||
/* $Id$ */
|
||||
|
||||
/***
|
||||
This file is part of PulseAudio.
|
||||
|
||||
Copyright 2004-2006 Lennart Poettering
|
||||
|
||||
PulseAudio is free software; you can redistribute it and/or modify
|
||||
it under the terms of the GNU Lesser General Public License as
|
||||
published by the Free Software Foundation; either version 2.1 of the
|
||||
License, or (at your option) any later version.
|
||||
|
||||
PulseAudio is distributed in the hope that it will be useful, but
|
||||
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with PulseAudio; if not, write to the Free Software
|
||||
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
||||
USA.
|
||||
***/
|
||||
|
||||
#include <sys/types.h>
|
||||
#include <time.h>
|
||||
|
||||
#include <pulse/sample.h>
|
||||
|
||||
/* Something like pulse/timeval.h but based on CLOCK_MONOTONIC and
|
||||
* timespec instead of timeval */
|
||||
|
||||
struct timespec *pa_rtclock_get(struct timespec *ts);
|
||||
pa_usec_t pa_rtclock_age(const struct timespec *tv);
|
||||
int pa_rtclock_hrtimer(void);
|
||||
|
||||
struct timespec *pa_timespec_store(struct timespec *a, pa_usec_t u);
|
||||
pa_usec_t pa_timespec_load(struct timespec *tv);
|
||||
struct timespec *pa_timespec_add(struct timespec *tv, pa_usec_t t);
|
||||
pa_usec_t pa_timespec_diff(const struct timespec *a, const struct timespec *b);
|
||||
int pa_timespec_cmp(const struct timespec *a, const struct timespec *b);
|
||||
|
||||
/* timer with a resolution better than this are considered high-resolution */
|
||||
#define PA_HRTIMER_THRESHOLD_USEC 10
|
||||
|
||||
#endif
|
||||
Loading…
Add table
Add a link
Reference in a new issue