xref: /freebsd/contrib/tzcode/difftime.c (revision bc42155199b5b0b479311e05b07aee7f6f9c5172)
1 *bc421551SDag-Erling Smørgrav /* Return the difference between two timestamps.  */
2 *bc421551SDag-Erling Smørgrav 
3 *bc421551SDag-Erling Smørgrav /*
4 *bc421551SDag-Erling Smørgrav ** This file is in the public domain, so clarified as of
5 *bc421551SDag-Erling Smørgrav ** 1996-06-05 by Arthur David Olson.
6 *bc421551SDag-Erling Smørgrav */
7 *bc421551SDag-Erling Smørgrav 
8 *bc421551SDag-Erling Smørgrav /*LINTLIBRARY*/
9 *bc421551SDag-Erling Smørgrav 
10 *bc421551SDag-Erling Smørgrav #include "namespace.h"
11 *bc421551SDag-Erling Smørgrav #include "private.h"	/* for time_t and TYPE_SIGNED */
12 *bc421551SDag-Erling Smørgrav #include "un-namespace.h"
13 *bc421551SDag-Erling Smørgrav 
14 *bc421551SDag-Erling Smørgrav /* Return -X as a double.  Using this avoids casting to 'double'.  */
15 *bc421551SDag-Erling Smørgrav static double
dminus(double x)16 *bc421551SDag-Erling Smørgrav dminus(double x)
17 *bc421551SDag-Erling Smørgrav {
18 *bc421551SDag-Erling Smørgrav   return -x;
19 *bc421551SDag-Erling Smørgrav }
20 *bc421551SDag-Erling Smørgrav 
21 *bc421551SDag-Erling Smørgrav double
difftime(time_t time1,time_t time0)22 *bc421551SDag-Erling Smørgrav difftime(time_t time1, time_t time0)
23 *bc421551SDag-Erling Smørgrav {
24 *bc421551SDag-Erling Smørgrav 	/*
25 *bc421551SDag-Erling Smørgrav 	** If double is large enough, simply convert and subtract
26 *bc421551SDag-Erling Smørgrav 	** (assuming that the larger type has more precision).
27 *bc421551SDag-Erling Smørgrav 	*/
28 *bc421551SDag-Erling Smørgrav 	if (sizeof(time_t) < sizeof(double)) {
29 *bc421551SDag-Erling Smørgrav 	  double t1 = time1, t0 = time0;
30 *bc421551SDag-Erling Smørgrav 	  return t1 - t0;
31 *bc421551SDag-Erling Smørgrav 	}
32 *bc421551SDag-Erling Smørgrav 
33 *bc421551SDag-Erling Smørgrav 	/*
34 *bc421551SDag-Erling Smørgrav 	** The difference of two unsigned values can't overflow
35 *bc421551SDag-Erling Smørgrav 	** if the minuend is greater than or equal to the subtrahend.
36 *bc421551SDag-Erling Smørgrav 	*/
37 *bc421551SDag-Erling Smørgrav 	if (!TYPE_SIGNED(time_t))
38 *bc421551SDag-Erling Smørgrav 	  return time0 <= time1 ? time1 - time0 : dminus(time0 - time1);
39 *bc421551SDag-Erling Smørgrav 
40 *bc421551SDag-Erling Smørgrav 	/* Use uintmax_t if wide enough.  */
41 *bc421551SDag-Erling Smørgrav 	if (sizeof(time_t) <= sizeof(uintmax_t)) {
42 *bc421551SDag-Erling Smørgrav 	  uintmax_t t1 = time1, t0 = time0;
43 *bc421551SDag-Erling Smørgrav 	  return time0 <= time1 ? t1 - t0 : dminus(t0 - t1);
44 *bc421551SDag-Erling Smørgrav 	}
45 *bc421551SDag-Erling Smørgrav 
46 *bc421551SDag-Erling Smørgrav 	/*
47 *bc421551SDag-Erling Smørgrav 	** Handle cases where both time1 and time0 have the same sign
48 *bc421551SDag-Erling Smørgrav 	** (meaning that their difference cannot overflow).
49 *bc421551SDag-Erling Smørgrav 	*/
50 *bc421551SDag-Erling Smørgrav 	if ((time1 < 0) == (time0 < 0))
51 *bc421551SDag-Erling Smørgrav 	  return time1 - time0;
52 *bc421551SDag-Erling Smørgrav 
53 *bc421551SDag-Erling Smørgrav 	/*
54 *bc421551SDag-Erling Smørgrav 	** The values have opposite signs and uintmax_t is too narrow.
55 *bc421551SDag-Erling Smørgrav 	** This suffers from double rounding; attempt to lessen that
56 *bc421551SDag-Erling Smørgrav 	** by using long double temporaries.
57 *bc421551SDag-Erling Smørgrav 	*/
58 *bc421551SDag-Erling Smørgrav 	{
59 *bc421551SDag-Erling Smørgrav 	  long double t1 = time1, t0 = time0;
60 *bc421551SDag-Erling Smørgrav 	  return t1 - t0;
61 *bc421551SDag-Erling Smørgrav 	}
62 *bc421551SDag-Erling Smørgrav }
63