1998-12-12 20:48:14 +00:00
|
|
|
/*
|
2001-01-09 22:01:04 +00:00
|
|
|
* Copyright (C) 1998-2001 Internet Software Consortium.
|
2000-08-01 01:33:37 +00:00
|
|
|
*
|
1998-12-12 20:48:14 +00:00
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
2000-08-01 01:33:37 +00:00
|
|
|
*
|
2000-07-27 09:55:03 +00:00
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM
|
|
|
|
* DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL
|
|
|
|
* INTERNET SOFTWARE CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
|
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
|
|
|
|
* FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
|
|
|
|
* NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
|
|
|
|
* WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
1998-12-12 20:48:14 +00:00
|
|
|
*/
|
1998-10-23 05:45:44 +00:00
|
|
|
|
2001-07-08 05:09:35 +00:00
|
|
|
/* $Id: time.c,v 1.22 2001/07/08 05:09:18 mayer Exp $ */
|
2001-07-06 05:08:39 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Windows has a different epoch than Unix. Therefore this code sets the epoch
|
|
|
|
* value to the Unix epoch. Care should be used when using these routines to
|
|
|
|
* ensure that this difference is taken into account. System and File times
|
|
|
|
* may require adjusting for this when modifying any time value that needs
|
|
|
|
* to be an absolute Windows time.
|
|
|
|
*
|
|
|
|
* Currently only epoch-specific code and the isc_time_seconds
|
|
|
|
* and isc_time_secondsastimet use the epoch-adjusted code.
|
|
|
|
*/
|
2000-06-22 22:00:42 +00:00
|
|
|
|
1998-12-12 19:25:20 +00:00
|
|
|
#include <config.h>
|
|
|
|
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
1998-10-23 05:45:44 +00:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
1998-10-23 06:02:07 +00:00
|
|
|
|
|
|
|
#include <windows.h>
|
1998-10-23 05:45:44 +00:00
|
|
|
|
|
|
|
#include <isc/assertions.h>
|
|
|
|
#include <isc/time.h>
|
2001-07-06 05:08:39 +00:00
|
|
|
#include <isc/util.h>
|
1998-10-23 05:45:44 +00:00
|
|
|
|
2000-03-10 17:50:36 +00:00
|
|
|
/*
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
* struct FILETIME uses "100-nanoseconds intervals".
|
2000-03-10 17:50:36 +00:00
|
|
|
* NS / S = 1000000000 (10^9).
|
|
|
|
* While it is reasonably obvious that this makes the needed
|
|
|
|
* conversion factor 10^7, it is coded this way for additional clarity.
|
|
|
|
*/
|
|
|
|
#define NS_PER_S 1000000000
|
|
|
|
#define NS_INTERVAL 100
|
|
|
|
#define INTERVALS_PER_S (NS_PER_S / NS_INTERVAL)
|
2001-07-06 05:08:39 +00:00
|
|
|
#define UINT64_MAX _UI64_MAX
|
|
|
|
|
|
|
|
/***
|
|
|
|
*** Absolute Times
|
|
|
|
***/
|
|
|
|
|
|
|
|
static isc_time_t epoch = { 0, 0 };
|
|
|
|
isc_time_t *isc_time_epoch = &epoch;
|
|
|
|
|
|
|
|
void
|
|
|
|
TimetToFileTime(time_t t, LPFILETIME pft) {
|
|
|
|
LONGLONG i;
|
2000-03-10 17:50:36 +00:00
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
i = Int32x32To64(t, 10000000) + 116444736000000000;
|
|
|
|
pft->dwLowDateTime = (DWORD) i;
|
|
|
|
pft->dwHighDateTime = (DWORD) (i >>32);
|
|
|
|
}
|
1998-10-26 23:07:57 +00:00
|
|
|
/***
|
|
|
|
*** Intervals
|
|
|
|
***/
|
|
|
|
|
1999-10-17 22:26:09 +00:00
|
|
|
static isc_interval_t zero_interval = { 0 };
|
|
|
|
isc_interval_t *isc_interval_zero = &zero_interval;
|
|
|
|
|
1998-10-26 23:07:57 +00:00
|
|
|
void
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_interval_set(isc_interval_t *i,
|
1998-10-26 23:07:57 +00:00
|
|
|
unsigned int seconds, unsigned int nanoseconds) {
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set 'i' to a value representing an interval of 'seconds' seconds
|
|
|
|
* and 'nanoseconds' nanoseconds, suitable for use in isc_time_add()
|
|
|
|
* and isc_time_subtract().
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(i != NULL);
|
2001-07-06 05:08:39 +00:00
|
|
|
REQUIRE(nanoseconds < NS_PER_S);
|
1998-10-26 23:07:57 +00:00
|
|
|
|
2000-03-10 17:50:36 +00:00
|
|
|
i->interval = (LONGLONG)seconds * INTERVALS_PER_S
|
|
|
|
+ nanoseconds / NS_INTERVAL;
|
1998-10-26 23:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
isc_boolean_t
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_interval_iszero(isc_interval_t *i) {
|
1998-10-26 23:07:57 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns ISC_TRUE iff. 'i' is the zero interval.
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(i != NULL);
|
|
|
|
if (i->interval == 0)
|
|
|
|
return (ISC_TRUE);
|
|
|
|
|
|
|
|
return (ISC_FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-04-24 20:58:03 +00:00
|
|
|
void
|
|
|
|
isc_time_set(isc_time_t *t, unsigned int seconds, unsigned int nanoseconds) {
|
|
|
|
ULARGE_INTEGER i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set 't' to a particular number of seconds + nanoseconds since the
|
|
|
|
* epoch.
|
|
|
|
*/
|
|
|
|
REQUIRE(t != NULL);
|
2001-07-06 05:08:39 +00:00
|
|
|
REQUIRE(nanoseconds < NS_PER_S);
|
2000-04-24 20:58:03 +00:00
|
|
|
|
|
|
|
i.QuadPart = (LONGLONG)seconds * INTERVALS_PER_S
|
|
|
|
+ nanoseconds / NS_INTERVAL;
|
|
|
|
|
|
|
|
t->absolute.dwLowDateTime = i.LowPart;
|
|
|
|
t->absolute.dwHighDateTime = i.HighPart;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
void
|
|
|
|
isc_time_initepoch() {
|
|
|
|
TimetToFileTime(0, &epoch.absolute);
|
|
|
|
}
|
|
|
|
|
1998-10-26 23:07:57 +00:00
|
|
|
void
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_time_settoepoch(isc_time_t *t) {
|
1998-10-26 23:07:57 +00:00
|
|
|
/*
|
|
|
|
* Set 't' to the time of the epoch.
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(t != NULL);
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
t->absolute.dwLowDateTime = epoch.absolute.dwLowDateTime;
|
|
|
|
t->absolute.dwHighDateTime = epoch.absolute.dwHighDateTime;
|
1998-10-26 23:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
isc_boolean_t
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_time_isepoch(isc_time_t *t) {
|
2001-07-06 05:08:39 +00:00
|
|
|
|
1998-10-26 23:07:57 +00:00
|
|
|
/*
|
|
|
|
* Returns ISC_TRUE iff. 't' is the epoch ("time zero").
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(t != NULL);
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
if (t->absolute.dwLowDateTime == epoch.absolute.dwLowDateTime &&
|
|
|
|
t->absolute.dwHighDateTime == epoch.absolute.dwHighDateTime)
|
1998-10-26 23:07:57 +00:00
|
|
|
return (ISC_TRUE);
|
|
|
|
|
|
|
|
return (ISC_FALSE);
|
|
|
|
}
|
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_result_t
|
|
|
|
isc_time_now(isc_time_t *t) {
|
2001-07-06 05:08:39 +00:00
|
|
|
|
|
|
|
char dtime[10];
|
1998-10-23 05:45:44 +00:00
|
|
|
/*
|
1998-10-26 23:07:57 +00:00
|
|
|
* Set *t to the current absolute time.
|
1998-10-23 05:45:44 +00:00
|
|
|
*/
|
2000-08-01 01:33:37 +00:00
|
|
|
|
1998-10-23 05:45:44 +00:00
|
|
|
REQUIRE(t != NULL);
|
1998-10-23 06:02:07 +00:00
|
|
|
|
1998-10-27 03:12:07 +00:00
|
|
|
GetSystemTimeAsFileTime(&t->absolute);
|
1998-10-23 06:02:07 +00:00
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
_strtime(dtime);
|
1998-10-23 05:45:44 +00:00
|
|
|
return (ISC_R_SUCCESS);
|
|
|
|
}
|
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_result_t
|
|
|
|
isc_time_nowplusinterval(isc_time_t *t, isc_interval_t *i) {
|
|
|
|
ULARGE_INTEGER i1;
|
1998-10-26 23:07:57 +00:00
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
/*
|
|
|
|
* Set *t to the current absolute time + i.
|
|
|
|
*/
|
2000-08-01 01:33:37 +00:00
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
REQUIRE(t != NULL);
|
|
|
|
REQUIRE(i != NULL);
|
2000-08-01 01:33:37 +00:00
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
GetSystemTimeAsFileTime(&t->absolute);
|
|
|
|
|
|
|
|
i1.LowPart = t->absolute.dwLowDateTime;
|
|
|
|
i1.HighPart = t->absolute.dwHighDateTime;
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
if (UINT64_MAX - i1.QuadPart < (unsigned __int64)i->interval)
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
return (ISC_R_RANGE);
|
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
i1.QuadPart += i->interval;
|
|
|
|
|
|
|
|
t->absolute.dwLowDateTime = i1.LowPart;
|
|
|
|
t->absolute.dwHighDateTime = i1.HighPart;
|
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
isc_time_compare(isc_time_t *t1, isc_time_t *t2) {
|
1998-10-23 05:45:44 +00:00
|
|
|
/*
|
|
|
|
* Compare the times referenced by 't1' and 't2'
|
|
|
|
*/
|
|
|
|
|
|
|
|
REQUIRE(t1 != NULL && t2 != NULL);
|
|
|
|
|
1998-10-27 03:12:07 +00:00
|
|
|
return ((int)CompareFileTime(&t1->absolute, &t2->absolute));
|
1998-10-23 05:45:44 +00:00
|
|
|
}
|
|
|
|
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
isc_result_t
|
2000-04-24 20:58:03 +00:00
|
|
|
isc_time_add(isc_time_t *t, isc_interval_t *i, isc_time_t *result) {
|
|
|
|
ULARGE_INTEGER i1;
|
1998-10-27 03:12:07 +00:00
|
|
|
|
1998-10-23 05:45:44 +00:00
|
|
|
/*
|
1998-10-26 23:07:57 +00:00
|
|
|
* Add 't' to 'i', storing the result in 'result'.
|
1998-10-23 05:45:44 +00:00
|
|
|
*/
|
|
|
|
|
1998-10-26 23:07:57 +00:00
|
|
|
REQUIRE(t != NULL && i != NULL && result != NULL);
|
1998-10-23 05:45:44 +00:00
|
|
|
|
1998-10-27 03:12:07 +00:00
|
|
|
i1.LowPart = t->absolute.dwLowDateTime;
|
|
|
|
i1.HighPart = t->absolute.dwHighDateTime;
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
if (UINT64_MAX - i1.QuadPart < (unsigned __int64)i->interval)
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
return (ISC_R_RANGE);
|
|
|
|
|
2000-04-24 20:58:03 +00:00
|
|
|
i1.QuadPart += i->interval;
|
|
|
|
|
|
|
|
result->absolute.dwLowDateTime = i1.LowPart;
|
|
|
|
result->absolute.dwHighDateTime = i1.HighPart;
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
1998-10-23 05:45:44 +00:00
|
|
|
}
|
|
|
|
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
isc_result_t
|
1999-09-23 18:03:39 +00:00
|
|
|
isc_time_subtract(isc_time_t *t, isc_interval_t *i, isc_time_t *result) {
|
2000-04-24 20:58:03 +00:00
|
|
|
ULARGE_INTEGER i1;
|
1998-10-27 03:12:07 +00:00
|
|
|
|
1998-10-23 05:45:44 +00:00
|
|
|
/*
|
1998-10-26 23:07:57 +00:00
|
|
|
* Subtract 'i' from 't', storing the result in 'result'.
|
1998-10-23 05:45:44 +00:00
|
|
|
*/
|
|
|
|
|
1998-10-26 23:07:57 +00:00
|
|
|
REQUIRE(t != NULL && i != NULL && result != NULL);
|
1998-10-27 03:12:07 +00:00
|
|
|
|
|
|
|
i1.LowPart = t->absolute.dwLowDateTime;
|
|
|
|
i1.HighPart = t->absolute.dwHighDateTime;
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
if (i1.QuadPart < (unsigned __int64) i->interval)
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
return (ISC_R_RANGE);
|
2000-04-24 20:58:03 +00:00
|
|
|
|
|
|
|
i1.QuadPart -= i->interval;
|
|
|
|
|
|
|
|
result->absolute.dwLowDateTime = i1.LowPart;
|
|
|
|
result->absolute.dwHighDateTime = i1.HighPart;
|
2001-07-06 05:08:39 +00:00
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
1998-10-23 05:45:44 +00:00
|
|
|
}
|
1998-10-26 23:07:57 +00:00
|
|
|
|
1999-10-09 02:39:53 +00:00
|
|
|
isc_uint64_t
|
|
|
|
isc_time_microdiff(isc_time_t *t1, isc_time_t *t2) {
|
1999-09-23 18:03:39 +00:00
|
|
|
ULARGE_INTEGER i1, i2;
|
1998-10-27 03:12:07 +00:00
|
|
|
LONGLONG i3;
|
|
|
|
|
1999-09-23 18:03:39 +00:00
|
|
|
REQUIRE(t1 != NULL && t2 != NULL);
|
|
|
|
|
|
|
|
i1.LowPart = t1->absolute.dwLowDateTime;
|
1998-10-27 03:12:07 +00:00
|
|
|
i1.HighPart = t1->absolute.dwHighDateTime;
|
1999-09-23 18:03:39 +00:00
|
|
|
i2.LowPart = t2->absolute.dwLowDateTime;
|
1998-10-27 03:12:07 +00:00
|
|
|
i2.HighPart = t2->absolute.dwHighDateTime;
|
|
|
|
|
|
|
|
if (i1.QuadPart <= i2.QuadPart)
|
|
|
|
return (0);
|
1999-09-23 18:03:39 +00:00
|
|
|
|
|
|
|
/*
|
1999-10-09 02:39:53 +00:00
|
|
|
* Convert to microseconds.
|
1999-09-23 18:03:39 +00:00
|
|
|
*/
|
1999-10-09 02:39:53 +00:00
|
|
|
i3 = (i1.QuadPart - i2.QuadPart) / 10;
|
1999-09-23 18:03:39 +00:00
|
|
|
|
1999-10-09 02:39:53 +00:00
|
|
|
return (i3);
|
1998-10-26 23:07:57 +00:00
|
|
|
}
|
2000-03-10 17:50:36 +00:00
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
/*
|
|
|
|
* Note that the value returned is the seconds relative to the Unix epoch rather than
|
|
|
|
* the seconds since Windows epoch.
|
|
|
|
* This is for compatibility with the Unix side.
|
|
|
|
*/
|
2000-03-10 17:50:36 +00:00
|
|
|
isc_uint32_t
|
|
|
|
isc_time_seconds(isc_time_t *t) {
|
|
|
|
ULARGE_INTEGER i;
|
|
|
|
|
2000-04-24 20:58:03 +00:00
|
|
|
REQUIRE(t != NULL);
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
i.LowPart = t->absolute.dwLowDateTime - epoch.absolute.dwLowDateTime;
|
|
|
|
i.HighPart = t->absolute.dwHighDateTime - epoch.absolute.dwHighDateTime;
|
2000-03-10 17:50:36 +00:00
|
|
|
|
|
|
|
return ((isc_uint32_t)(i.QuadPart / INTERVALS_PER_S));
|
|
|
|
}
|
|
|
|
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
isc_result_t
|
|
|
|
isc_time_secondsastimet(isc_time_t *t, time_t *secondsp) {
|
|
|
|
ULARGE_INTEGER i1, i2;
|
|
|
|
time_t seconds;
|
|
|
|
|
|
|
|
REQUIRE(t != NULL);
|
|
|
|
|
|
|
|
i1.LowPart = t->absolute.dwLowDateTime;
|
|
|
|
i1.HighPart = t->absolute.dwHighDateTime;
|
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
/*
|
|
|
|
* Get the time_t zero equivalent in FILETIME
|
|
|
|
* The zero point for FILETIME is 1 January, 1601
|
|
|
|
* while for timet it is 1 January, 1970
|
|
|
|
*/
|
|
|
|
i1.LowPart -= epoch.absolute.dwLowDateTime;
|
|
|
|
i1.HighPart -= epoch.absolute.dwHighDateTime;
|
|
|
|
|
189. [func] isc_time_secondsastimet(), a new function, will ensure
that the number of seconds in an isc_time_t does not
exceed the range of a time_t, or return ISC_R_RANGE.
Similarly, isc_time_now(), isc_time_nowplusinterval(),
isc_time_add() and isc_time_subtract() now check the
range for overflow/underflow. In the case of
isc_time_subtract, this changed a calling requirement
(ie, something that could generate an assertion)
into merely a condition that returns an error result.
isc_time_add() and isc_time_subtract() were void-
valued before but now return isc_result_t.
The seconds member isc_time_t on Unix platforms was changed from time_t
to unsigned int.
unix/time.c now uses macros for nanoseconds per second, nanoseconds per
microsecond and microseconds per second to make sure that the right
number of zeros appears each place the constant is used.
unix/time.c functions which take initialized isc_(interval|time)_t arguments
INSIST() that the nanoseconds value is less than one full second.
unix/time.c's isc_time_microdiff was broken because it did multiplication and
addition with unsigned integers and attempted to set them a 64 bit int to
avoid overflow, but C's ints don't promote to 64 bits on machines that only
have 32 bit longs. Fixed.
Added all the pertinent documentation to time.h.
2000-05-18 17:08:32 +00:00
|
|
|
i1.QuadPart /= INTERVALS_PER_S;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Ensure that the number of seconds can be represented by a time_t.
|
|
|
|
* Since the number seconds is an unsigned int and since time_t is
|
|
|
|
* mostly opaque, this is trickier than it seems. (This standardized
|
|
|
|
* opaqueness of time_t is *very* * frustrating; time_t is not even
|
|
|
|
* limited to being an integral type.) Thought it is known at the
|
|
|
|
* time of this writing that time_t is a signed long on the Win32
|
|
|
|
* platform, the full treatment is given to figuring out if things
|
|
|
|
* fit to allow for future Windows platforms where time_t is *not*
|
|
|
|
* a signed long, or where perhaps a signed long is longer than
|
|
|
|
* it currently is.
|
|
|
|
*/
|
|
|
|
seconds = (time_t)i1.QuadPart;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* First, only do the range tests if the type of size_t is integral.
|
|
|
|
* Float/double easily include the maximum possible values.
|
|
|
|
*/
|
|
|
|
if ((time_t)0.5 != 0.5) {
|
|
|
|
/*
|
|
|
|
* Did all the bits make it in?
|
|
|
|
*/
|
|
|
|
if ((seconds & i1.QuadPart) != i1.QuadPart)
|
|
|
|
return (ISC_R_RANGE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Is time_t signed with the high bit set?
|
|
|
|
*
|
|
|
|
* The first test (the sizeof comparison) determines
|
|
|
|
* whether we can even deduce the signedness of time_t
|
|
|
|
* by using ANSI's rule about integer conversion to
|
|
|
|
* wider integers.
|
|
|
|
*
|
|
|
|
* The second test uses that ANSI rule to see whether
|
|
|
|
* the value of time_t was sign extended into QuadPart.
|
|
|
|
* If the test is true, then time_t is signed.
|
|
|
|
*
|
|
|
|
* The final test ensures the high bit is not set, or
|
|
|
|
* the value is negative and hence there is a range error.
|
|
|
|
*/
|
|
|
|
if (sizeof(time_t) < sizeof(i2.QuadPart) &&
|
|
|
|
((i2.QuadPart = (time_t)-1) ^ (time_t)-1) != 0 &&
|
|
|
|
(seconds & (1 << (sizeof(time_t) * 8 - 1))) != 0)
|
|
|
|
return (ISC_R_RANGE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Last test ... the size of time_t is >= that of i2.QuadPart,
|
|
|
|
* so we can't determine its signedness. Unconditionally
|
|
|
|
* declare anything with the high bit set as out of range.
|
|
|
|
* Since even the maxed signed value is ludicrously far from
|
|
|
|
* when this is being written, this rule shall not impact
|
|
|
|
* anything for all intents and purposes.
|
|
|
|
*
|
|
|
|
* How far? Well ... if FILETIME is in 100 ns intervals since
|
|
|
|
* 1600, and a QuadPart can store 9223372036854775808 such
|
|
|
|
* intervals when interpreted as signed (ie, if sizeof(time_t)
|
|
|
|
* == sizeof(QuadPart) but time_t is signed), that means
|
|
|
|
* 9223372036854775808 / INTERVALS_PER_S = 922,337,203,685
|
|
|
|
* seconds. That number divided by 60 * 60 * 24 * 365 seconds
|
|
|
|
* per year means a signed time_t can store at least 29,247
|
|
|
|
* years, with only 400 of those years used up since 1600 as I
|
|
|
|
* write this in May, 2000.
|
|
|
|
*
|
|
|
|
* (Real date calculations are of course incredibly more
|
|
|
|
* complex; I'm only describing the approximate scale of
|
|
|
|
* the numbers involved here.)
|
|
|
|
*
|
|
|
|
* If the Galactic Federation is still running libisc's time
|
|
|
|
* libray on a Windows platform in the year 27647 A.D., then
|
|
|
|
* feel free to hunt down my greatgreatgreatgreatgreat(etc)
|
|
|
|
* grandchildren and whine at them about what I did.
|
|
|
|
*/
|
|
|
|
if ((seconds & (1 << (sizeof(time_t) * 8 - 1))) != 0)
|
|
|
|
return (ISC_R_RANGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
*secondsp = seconds;
|
|
|
|
|
|
|
|
return (ISC_R_SUCCESS);
|
|
|
|
}
|
|
|
|
|
2000-03-10 17:50:36 +00:00
|
|
|
isc_uint32_t
|
|
|
|
isc_time_nanoseconds(isc_time_t *t) {
|
2001-07-06 05:08:39 +00:00
|
|
|
SYSTEMTIME st;
|
2000-04-24 20:58:03 +00:00
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
/*
|
|
|
|
* Convert the time to a SYSTEMTIME structure and the grab the
|
|
|
|
* milliseconds
|
|
|
|
*/
|
|
|
|
FileTimeToSystemTime(&t->absolute, &st);
|
2000-03-10 17:50:36 +00:00
|
|
|
|
2001-07-06 05:08:39 +00:00
|
|
|
return ((isc_uint32_t)(st.wMilliseconds * 1000000));
|
2000-03-10 17:50:36 +00:00
|
|
|
}
|