freeswitch/libs/sofia-sip/libsofia-sip-ua/su/su_time.c

485 lines
11 KiB
C

/*
* This file is part of the Sofia-SIP package
*
* Copyright (C) 2005,2006 Nokia Corporation.
*
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
*
* This library 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.
*
* This library 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 this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA
*
*/
/**@ingroup su_time
* @CFILE su_time.c
*
* @brief Implementation of OS-independent time functions.
*
* @author Pekka Pessi <Pekka.Pessi@nokia.com>
* @author Jari Selin <Jari.Selin@nokia.com>
* @author Kai Vehmanen <first.surname@nokia.com>
*
* @date Created: Thu Mar 18 19:40:51 1999 pessi
*/
#include "config.h"
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "sofia-sip/su_types.h"
#include "su_module_debug.h"
/* Include bodies of inlined functions */
#undef su_inline
#define su_inline
#undef SU_HAVE_INLINE
#define SU_HAVE_INLINE 1
#include "sofia-sip/su_time.h"
#if HAVE_SYS_TIME_H
#include <sys/time.h> /* Get struct timeval */
#endif
/**@defgroup su_time Time Handling
*
* OS-independent timing functions and types for the @b su library.
*
* The @b su library provides three different time formats with different
* ranges and epochs in <sofia-sip/su_time.h>:
*
* - #su_time_t, second and microsecond as 32-bit values since 1900,
* - #su_duration_t, milliseconds between two times, and
* - #su_ntp_t, standard NTP timestamp (seconds since 1900 as
* a fixed-point 64-bit value with 32 bits representing subsecond
* value).
*/
/**
* Compare two timestamps.
*
* The function su_time_cmp() compares two su_time_t timestamps.
*
* @param t1 first NTP timestamp in su_time_t structure
* @param t2 second NTP timestamp in su_time_t structure
*
* @retval Negative, if @a t1 is before @a t2,
* @retval Zero, if @a t1 is same as @a t2, or
* @retval Positive, if @a t1 is after @a t2.
*
*/
long su_time_cmp(su_time_t const t1, su_time_t const t2)
{
long retval = 0;
if (t1.tv_sec > t2.tv_sec)
retval = 1;
else if (t1.tv_sec < t2.tv_sec)
retval = -1;
else {
if (t1.tv_usec > t2.tv_usec)
retval = 1;
else if (t1.tv_usec < t2.tv_usec)
retval = -1;
}
return retval;
}
/**@def SU_TIME_CMP(t1, t2)
*
* Compare two timestamps.
*
* The macro SU_TIME_CMP() compares two su_time_t timestamps.
*
* @param t1 first NTP timestamp in su_time_t structure
* @param t2 second NTP timestamp in su_time_t structure
*
* @retval negative, if t1 is before t2,
* @retval zero, if t1 is same as t2, or
* @retval positive, if t1 is after t2.
*
* @hideinitializer
*/
/** Difference between two timestamps.
*
* The function returns difference between two timestamps
* in seconds (t1 - t2).
*
* @param t1 first timeval
* @param t2 second timeval
*
* @return
* The difference between two timestamps in seconds as a double.
*/
double su_time_diff(su_time_t const t1, su_time_t const t2)
{
return
((double)t1.tv_sec - (double)t2.tv_sec)
+ (double)((long)t1.tv_usec - (long)t2.tv_usec) / 1000000.0;
}
/** Get current time.
*
* Return the current timestamp in su_time_t structure.
*
* @return
* The structure containing the current NTP timestamp.
*/
su_time_t su_now(void)
{
su_time_t retval;
su_time(&retval);
return retval;
}
/** Print su_time_t timestamp.
*
* This function prints a su_time_t timestamp as a decimal number to the
* given buffer.
*
* @param s pointer to buffer
* @param n buffer size
* @param tv pointer to the timeval object
*
* @return
* The number of characters printed, excluding the final @c NUL.
*/
int su_time_print(char *s, int n, su_time_t const *tv)
{
#ifdef _WIN32
return _snprintf(s, n, "%lu.%06lu", tv->tv_sec, tv->tv_usec);
#else
return snprintf(s, n, "%lu.%06lu", tv->tv_sec, tv->tv_usec);
#endif
}
/** Time difference in milliseconds.
*
* Calculates the duration from t2 to t1 in milliseconds.
*
* @param t1 after time
* @param t2 before time
*
* @return The duration in milliseconds between the two times.
* If the difference is bigger than #SU_DURATION_MAX, return #SU_DURATION_MAX
* instead.
* If the difference is smaller than -#SU_DURATION_MAX, return
* -#SU_DURATION_MAX.
*/
su_duration_t su_duration(su_time_t const t1, su_time_t const t2)
{
su_duration_t diff, udiff, tdiff;
diff = t1.tv_sec - t2.tv_sec;
udiff = t1.tv_usec - t2.tv_usec;
tdiff = diff * 1000 + udiff / 1000;
if (diff > (SU_DURATION_MAX / 1000) || (diff > 0 && diff > tdiff))
return SU_DURATION_MAX;
if (diff < (-SU_DURATION_MAX / 1000) || (diff < 0 && diff < tdiff))
return -SU_DURATION_MAX;
return tdiff;
}
typedef uint64_t su_t64_t; /* time with 64 bits */
static su_time_t su_t64_to_time(su_t64_t const us);
const uint32_t su_res32 = 1000000UL;
const su_t64_t su_res64 = (su_t64_t)1000000UL;
#define SU_TIME_TO_T64(tv) \
(su_res64 * (su_t64_t)(tv).tv_sec + (su_t64_t)(tv).tv_usec)
#define SU_DUR_TO_T64(d) (1000 * (int64_t)(d))
#define E9 (1000000000)
/** Get NTP timestamp.
*
* The function su_ntp_now() returns the current NTP timestamp. NTP
* timestamp is seconds elapsed since January 1st, 1900.
*
* @return
* The current time as NTP timestamp is returned.
*/
su_ntp_t su_ntp_now(void)
{
su_nanotime_t now;
su_nanotime(&now);
if (sizeof(long) == sizeof(now)) {
return ((now / E9) << 32) | ((((now % E9) << 32) + E9/2) / E9);
}
else {
su_nanotime_t usec;
int nsec;
unsigned rem;
usec = now / 1000;
nsec = now % 1000;
/*
* Multiply usec by 4294.967296 (ie. 2**32 / 1E6)
*
* Utilize fact that 4294.967296 == 4295 - 511 / 15625
*/
now = 4295 * usec - 511 * usec / 15625;
rem = (511U * usec) % 15625U;
/* Multiply nsec by 4.294967296 */
nsec = 4295 * 125 * nsec - (511 * nsec) / 125;
nsec -= 8 * rem; /* usec rounding */
nsec = (nsec + (nsec < 0 ? -62499 : +62499)) / 125000;
return now + nsec;
}
}
/** Get NTP seconds.
*
* The function su_ntp_sec() returns the seconds elapsed since January 1st,
* 1900.
*
* @return
* The current time as NTP seconds is returned.
*/
uint32_t su_ntp_sec(void)
{
su_time_t t;
su_time(&t);
return t.tv_sec;
}
/** High 32 bit of NTP timestamp.
*
* @param ntp 64bit NTP timestamp.
*
* @return The function su_ntp_hi() returns high 32 bits of NTP timestamp.
*/
uint32_t su_ntp_hi(su_ntp_t ntp)
{
return (uint32_t) (ntp >> 32) & 0xffffffffLU;
}
su_ntp_t su_ntp_hilo(uint32_t hi, uint32_t lo)
{
return ((((su_ntp_t)hi) << 32)) | lo;
}
/** Low 32 bit of NTP timestamp.
*
* @param ntp 64bit NTP timestamp.
*
* @return The function su_ntp_hi() returns low 32 bits of NTP timestamp.
*/
uint32_t su_ntp_lo(su_ntp_t ntp)
{
return (uint32_t) ntp & 0xffffffffLU;
}
/** Middle 32 bit of NTP timestamp.
*
* @param ntp 64bit NTP timestamp.
*
* @return The function su_ntp_mw() returns bits 48..16 (middle word) of NTP
* timestamp.
*/
uint32_t su_ntp_mw(su_ntp_t ntp)
{
return (uint32_t) (ntp >> 16) & 0xffffffffLU;
}
static
su_time_t su_t64_to_time(su_t64_t const us)
{
su_time_t tv;
tv.tv_sec = (unsigned long) (us / su_res64);
tv.tv_usec = (unsigned long) (us % su_res64);
return tv;
}
/**
* Add milliseconds to the time.
*
* @param t0 time in seconds and microseconds as @c su_time_t
* @param dur milliseconds to be added
*/
su_time_t su_time_add(su_time_t t0, su_duration_t dur)
{
return su_t64_to_time(SU_TIME_TO_T64(t0) + SU_DUR_TO_T64(dur));
}
/**
* Add seconds to the time.
*
* @param t0 time in seconds and microseconds as @c su_time_t
* @param sec seconds to be added
*
* @return
* New time as @c su_time_t.
*/
su_time_t su_time_dadd(su_time_t t0, double sec)
{
return su_t64_to_time(SU_TIME_TO_T64(t0) + (int64_t)(su_res32 * sec));
}
#ifdef WIN32
#include <windows.h>
uint64_t su_nanocounter(void)
{
static ULONGLONG counterfreq = 0; /*performance counter frequency*/
LARGE_INTEGER LargeIntCount = { 0, 0 };
ULONGLONG count;
if(!counterfreq) {
LARGE_INTEGER Freq = { 0, 0 };
if (!QueryPerformanceFrequency(&Freq)) {
SU_DEBUG_1(("su_counter: QueryPerformanceFrequency failed\n"));
return 0;
}
counterfreq = Freq.QuadPart;
}
if (!QueryPerformanceCounter(&LargeIntCount)) {
SU_DEBUG_1(("su_counter: QueryPeformanceCounter failed\n"));
return 0;
}
count = (ULONGLONG) LargeIntCount.QuadPart;
/* return value is in ns */
return ((count * 1000 * 1000 * 1000) / counterfreq) ;
}
uint64_t su_counter(void)
{
return su_nanocounter() / 1000U;
}
#elif HAVE_CLOCK_GETTIME
/** Return CPU counter value in nanoseconds.
*
* The function su_nanocounter() returns a CPU counter value in nanoseconds
* for timing purposes.
*
* Parameters:
*
* @return
* The current CPU counter value in nanoseconds
*/
uint64_t su_nanocounter(void)
{
struct timespec tp;
struct timeval tv;
static int init = 0;
static clockid_t cpu = CLOCK_REALTIME;
# define CLOCK_GETTIMEOFDAY ((clockid_t)-1)
if (init == 0) {
init = 1;
if (0)
;
#if HAVE_CLOCK_GETCPUCLOCKID
else if (clock_getcpuclockid(0, &cpu) != -1 &&
clock_gettime(cpu, &tp) != -1)
;
#endif
#ifdef _POSIX_CPUTIME
else if (clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &tp) >= 0)
cpu = CLOCK_PROCESS_CPUTIME_ID;
#endif
else if (clock_gettime(CLOCK_REALTIME, &tp) >= 0)
cpu = CLOCK_REALTIME;
else
cpu = CLOCK_GETTIMEOFDAY;
}
if (cpu == CLOCK_GETTIMEOFDAY) {
gettimeofday(&tv, NULL);
tp.tv_sec = tv.tv_sec, tp.tv_nsec = tv.tv_usec * 1000;
}
else if (clock_gettime(cpu, &tp) < 0)
perror("clock_gettime");
/* return value is in nanoseconds */
return
(uint64_t)((unsigned long)tp.tv_nsec) +
(uint64_t)((unsigned long)tp.tv_sec) * 1000000000ULL;
}
/** Return CPU counter value in microseconds.
*
* The function su_counter() returns the CPU counter value in microseconds
* for timing purposes.
*
* Parameters:
*
* @return
* The current CPU counter value in microseconds.
*/
uint64_t su_counter(void)
{
return su_nanocounter() / 1000U;
}
#elif HAVE_GETTIMEOFDAY
uint64_t su_counter(void)
{
struct timeval tv;
gettimeofday(&tv, NULL);
/* return value is in microseconds */
return
(uint64_t)((unsigned long)tv.tv_usec) +
(uint64_t)((unsigned long)tv.tv_sec) * 1000000ULL;
}
uint64_t su_nanocounter(void)
{
struct timeval tv;
gettimeofday(&tv, NULL);
/* return value is in nanoseconds */
return
(uint64_t)((unsigned long)tv.tv_usec) * 1000ULL +
(uint64_t)((unsigned long)tv.tv_sec) * 1000000000ULL;
}
#else
uint64_t su_counter(void)
{
return (uint64_t)clock();
}
uint64_t su_nanocounter(void)
{
return (uint64_t)clock() * 1000;
}
#endif /* WIN32 */