xref: /illumos-gate/usr/src/uts/common/sys/time.h (revision 617b28ccfac40348ebae033171d0643fcca18c22)
1 /*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
2 /*	  All Rights Reserved	*/
3 
4 
5 /*
6  * Copyright (c) 1982, 1986, 1993 Regents of the University of California.
7  * All rights reserved.  The Berkeley software License Agreement
8  * specifies the terms and conditions for redistribution.
9  */
10 
11 /*
12  * Copyright 2014 Garrett D'Amore <garrett@damore.org>
13  *
14  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
15  * Use is subject to license terms.
16  *
17  * Copyright 2013 Nexenta Systems, Inc.  All rights reserved.
18  *
19  * Copyright 2020 OmniOS Community Edition (OmniOSce) Association.
20  */
21 
22 /*
23  * Copyright (c) 2013, 2016 by Delphix. All rights reserved.
24  */
25 
26 #ifndef _SYS_TIME_H
27 #define	_SYS_TIME_H
28 
29 #include <sys/feature_tests.h>
30 
31 /*
32  * Structure returned by gettimeofday(2) system call,
33  * and used in other calls.
34  */
35 
36 #ifdef	__cplusplus
37 extern "C" {
38 #endif
39 
40 #if !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) || \
41 	defined(__EXTENSIONS__)
42 #ifndef	_ASM
43 
44 #if !defined(_TIME_T) || __cplusplus >= 199711L
45 #define	_TIME_T
46 typedef	long	time_t;		/* time of day in seconds */
47 #endif	/* _TIME_T */
48 
49 #ifndef	_SUSECONDS_T
50 #define	_SUSECONDS_T
51 typedef	long	suseconds_t;	/* signed # of microseconds */
52 #endif	/* _SUSECONDS_T */
53 
54 struct timeval {
55 	time_t		tv_sec;		/* seconds */
56 	suseconds_t	tv_usec;	/* and microseconds */
57 };
58 
59 #if defined(_SYSCALL32)
60 
61 #include <sys/types32.h>
62 
63 #define	TIMEVAL32_TO_TIMEVAL(tv, tv32)	{	\
64 	(tv)->tv_sec = (time_t)(tv32)->tv_sec;	\
65 	(tv)->tv_usec = (tv32)->tv_usec;	\
66 }
67 
68 #define	TIMEVAL_TO_TIMEVAL32(tv32, tv)	{		\
69 	(tv32)->tv_sec = (time32_t)(tv)->tv_sec;	\
70 	(tv32)->tv_usec = (int32_t)(tv)->tv_usec;	\
71 }
72 
73 #define	TIME32_MAX	INT32_MAX
74 #define	TIME32_MIN	INT32_MIN
75 
76 #define	TIMEVAL_OVERFLOW(tv)	\
77 	((tv)->tv_sec < TIME32_MIN || (tv)->tv_sec > TIME32_MAX)
78 
79 #endif	/* _SYSCALL32 */
80 
81 #endif	/* _ASM */
82 #endif	/* !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) ... */
83 
84 #if !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
85 #ifndef	_ASM
86 struct timezone {
87 	int	tz_minuteswest;	/* minutes west of Greenwich */
88 	int	tz_dsttime;	/* type of dst correction */
89 };
90 
91 #endif	/* _ASM */
92 #endif /* !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__) */
93 
94 #ifdef	__cplusplus
95 }
96 #endif
97 
98 /*
99  * Needed for longlong_t type.  Placement of this due to <sys/types.h>
100  * including <sys/select.h> which relies on the presense of the itimerval
101  * structure.
102  */
103 #ifndef	_ASM
104 #include <sys/types.h>
105 #endif	/* _ASM */
106 
107 #ifdef	__cplusplus
108 extern "C" {
109 #endif
110 
111 #if !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
112 
113 #define	DST_NONE	0	/* not on dst */
114 #define	DST_USA		1	/* USA style dst */
115 #define	DST_AUST	2	/* Australian style dst */
116 #define	DST_WET		3	/* Western European dst */
117 #define	DST_MET		4	/* Middle European dst */
118 #define	DST_EET		5	/* Eastern European dst */
119 #define	DST_CAN		6	/* Canada */
120 #define	DST_GB		7	/* Great Britain and Eire */
121 #define	DST_RUM		8	/* Rumania */
122 #define	DST_TUR		9	/* Turkey */
123 #define	DST_AUSTALT	10	/* Australian style with shift in 1986 */
124 
125 /*
126  * Operations on timevals.
127  */
128 #define	timerisset(tvp)		((tvp)->tv_sec || (tvp)->tv_usec)
129 #define	timercmp(tvp, uvp, cmp) \
130 	(((tvp)->tv_sec == (uvp)->tv_sec) ? \
131 	    /* CSTYLED */ \
132 	    ((tvp)->tv_usec cmp (uvp)->tv_usec) : \
133 	    /* CSTYLED */ \
134 	    ((tvp)->tv_sec cmp (uvp)->tv_sec))
135 
136 #define	timerclear(tvp)		(tvp)->tv_sec = (tvp)->tv_usec = 0
137 
138 #ifdef __lint
139 /*
140  * Make innocuous, lint-happy versions until do {} while (0) is acknowleged as
141  * lint-safe.  If the compiler could know that we always make tv_usec < 1000000
142  * we wouldn't need a special linted version.
143  */
144 #define	timeradd(tvp, uvp, vvp)					\
145 	do								\
146 	{								\
147 		(vvp)->tv_sec = (tvp)->tv_sec + (uvp)->tv_sec;		\
148 		(vvp)->tv_usec = (tvp)->tv_usec + (uvp)->tv_usec;	\
149 		if ((vvp)->tv_usec >= 1000000)				\
150 		{							\
151 			(vvp)->tv_sec++;				\
152 			(vvp)->tv_usec -= 1000000;			\
153 		}							\
154 	} while ((vvp)->tv_usec >= 1000000)
155 #define	timersub(tvp, uvp, vvp)					\
156 	do								\
157 	{								\
158 		(vvp)->tv_sec = (tvp)->tv_sec - (uvp)->tv_sec;		\
159 		(vvp)->tv_usec = (tvp)->tv_usec - (uvp)->tv_usec;	\
160 		if ((vvp)->tv_usec < 0)					\
161 		{							\
162 			(vvp)->tv_sec--;				\
163 			(vvp)->tv_usec += 1000000;			\
164 		}							\
165 	} while ((vvp)->tv_usec >= 1000000)
166 #else
167 #define	timeradd(tvp, uvp, vvp)					\
168 	do								\
169 	{								\
170 		(vvp)->tv_sec = (tvp)->tv_sec + (uvp)->tv_sec;		\
171 		(vvp)->tv_usec = (tvp)->tv_usec + (uvp)->tv_usec;	\
172 		if ((vvp)->tv_usec >= 1000000)				\
173 		{							\
174 			(vvp)->tv_sec++;				\
175 			(vvp)->tv_usec -= 1000000;			\
176 		}							\
177 	} while (0)
178 
179 #define	timersub(tvp, uvp, vvp)					\
180 	do								\
181 	{								\
182 		(vvp)->tv_sec = (tvp)->tv_sec - (uvp)->tv_sec;		\
183 		(vvp)->tv_usec = (tvp)->tv_usec - (uvp)->tv_usec;	\
184 		if ((vvp)->tv_usec < 0)					\
185 		{							\
186 			(vvp)->tv_sec--;				\
187 			(vvp)->tv_usec += 1000000;			\
188 		}							\
189 	} while (0)
190 #endif /* __lint */
191 
192 #endif /* !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__) */
193 
194 #if !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) || defined(__EXTENSIONS__)
195 /*
196  * Names of the interval timers, and structure
197  * defining a timer setting.
198  */
199 #define	ITIMER_REAL	0	/* Decrements in real time */
200 #define	ITIMER_VIRTUAL	1	/* Decrements in process virtual time */
201 #define	ITIMER_PROF	2	/* Decrements both in process virtual */
202 				/* time and when system is running on */
203 				/* behalf of the process. */
204 #define	ITIMER_REALPROF	3	/* Decrements in real time for real- */
205 				/* time profiling of multithreaded */
206 				/* programs. */
207 
208 #ifndef	_ASM
209 struct	itimerval {
210 	struct	timeval it_interval;	/* timer interval */
211 	struct	timeval it_value;	/* current value */
212 };
213 
214 #if defined(_SYSCALL32)
215 
216 struct itimerval32 {
217 	struct	timeval32 it_interval;
218 	struct	timeval32 it_value;
219 };
220 
221 #define	ITIMERVAL32_TO_ITIMERVAL(itv, itv32)	{	\
222 	TIMEVAL32_TO_TIMEVAL(&(itv)->it_interval, &(itv32)->it_interval); \
223 	TIMEVAL32_TO_TIMEVAL(&(itv)->it_value, &(itv32)->it_value);	\
224 }
225 
226 #define	ITIMERVAL_TO_ITIMERVAL32(itv32, itv)	{	\
227 	TIMEVAL_TO_TIMEVAL32(&(itv32)->it_interval, &(itv)->it_interval); \
228 	TIMEVAL_TO_TIMEVAL32(&(itv32)->it_value, &(itv)->it_value);	\
229 }
230 
231 #define	ITIMERVAL_OVERFLOW(itv)				\
232 	(TIMEVAL_OVERFLOW(&(itv)->it_interval) ||	\
233 	TIMEVAL_OVERFLOW(&(itv)->it_value))
234 
235 #endif	/* _SYSCALL32 */
236 #endif	/* _ASM */
237 #endif /* !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) ... */
238 
239 
240 #if !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
241 /*
242  *	Definitions for commonly used resolutions.
243  */
244 #define	SEC		1
245 #define	MILLISEC	1000
246 #define	MICROSEC	1000000
247 #define	NANOSEC		1000000000LL
248 
249 #define	MSEC2NSEC(m)	((hrtime_t)(m) * (NANOSEC / MILLISEC))
250 #define	NSEC2MSEC(n)	((n) / (NANOSEC / MILLISEC))
251 
252 #define	USEC2NSEC(m)	((hrtime_t)(m) * (NANOSEC / MICROSEC))
253 #define	NSEC2USEC(n)	((n) / (NANOSEC / MICROSEC))
254 
255 #define	NSEC2SEC(n)	((n) / (NANOSEC / SEC))
256 #define	SEC2NSEC(m)	((hrtime_t)(m) * (NANOSEC / SEC))
257 
258 #endif /* !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__) */
259 
260 #ifndef	_ASM
261 
262 /*
263  * Time expressed as a 64-bit nanosecond counter.
264  */
265 typedef	longlong_t	hrtime_t;
266 
267 #if defined(_KERNEL) || defined(_FAKE_KERNEL)
268 
269 #include <sys/time_impl.h>
270 #include <sys/mutex.h>
271 
272 extern int tick_per_msec;	/* clock ticks per millisecond (may be zero) */
273 extern int msec_per_tick;	/* milliseconds per clock tick (may be zero) */
274 extern int usec_per_tick;	/* microseconds per clock tick */
275 extern int nsec_per_tick;	/* nanoseconds per clock tick */
276 
277 /*
278  * Macros to convert from common units of time (sec, msec, usec, nsec,
279  * timeval, timestruc) to clock ticks and vice versa.
280  */
281 #define	TICK_TO_SEC(tick)	((tick) / hz)
282 #define	SEC_TO_TICK(sec)	((sec) * hz)
283 
284 #define	TICK_TO_MSEC(tick)	\
285 	(msec_per_tick ? (tick) * msec_per_tick : (tick) / tick_per_msec)
286 #define	MSEC_TO_TICK(msec)	\
287 	(msec_per_tick ? (msec) / msec_per_tick : (msec) * tick_per_msec)
288 #define	MSEC_TO_TICK_ROUNDUP(msec)	\
289 	(msec_per_tick ? \
290 	((msec) == 0 ? 0 : ((msec) - 1) / msec_per_tick + 1) : \
291 	(msec) * tick_per_msec)
292 
293 #define	TICK_TO_USEC(tick)		((tick) * usec_per_tick)
294 #define	USEC_TO_TICK(usec)		((usec) / usec_per_tick)
295 #define	USEC_TO_TICK_ROUNDUP(usec)	\
296 	((usec) == 0 ? 0 : USEC_TO_TICK((usec) - 1) + 1)
297 
298 #define	TICK_TO_NSEC(tick)		((hrtime_t)(tick) * nsec_per_tick)
299 #define	NSEC_TO_TICK(nsec)		((nsec) / nsec_per_tick)
300 #define	NSEC_TO_TICK_ROUNDUP(nsec)	\
301 	((nsec) == 0 ? 0 : NSEC_TO_TICK((nsec) - 1) + 1)
302 
303 #define	TICK_TO_TIMEVAL(tick, tvp) {	\
304 	clock_t __tmptck = (tick);	\
305 	(tvp)->tv_sec = TICK_TO_SEC(__tmptck);	\
306 	(tvp)->tv_usec = TICK_TO_USEC(__tmptck - SEC_TO_TICK((tvp)->tv_sec)); \
307 }
308 
309 #define	TICK_TO_TIMEVAL32(tick, tvp) {	\
310 	clock_t __tmptck = (tick);	\
311 	time_t __tmptm = TICK_TO_SEC(__tmptck);	\
312 	(tvp)->tv_sec = (time32_t)__tmptm;	\
313 	(tvp)->tv_usec = TICK_TO_USEC(__tmptck - SEC_TO_TICK(__tmptm)); \
314 }
315 
316 #define	TICK_TO_TIMESTRUC(tick, tsp) {	\
317 	clock_t __tmptck = (tick);	\
318 	(tsp)->tv_sec = TICK_TO_SEC(__tmptck);	\
319 	(tsp)->tv_nsec = TICK_TO_NSEC(__tmptck - SEC_TO_TICK((tsp)->tv_sec)); \
320 }
321 
322 #define	TICK_TO_TIMESTRUC32(tick, tsp) {	\
323 	clock_t __tmptck = (tick);			\
324 	time_t __tmptm = TICK_TO_SEC(__tmptck);		\
325 	(tsp)->tv_sec = (time32_t)__tmptm;		\
326 	(tsp)->tv_nsec = TICK_TO_NSEC(__tmptck - SEC_TO_TICK(__tmptm));	\
327 }
328 
329 #define	TIMEVAL_TO_TICK(tvp)	\
330 	(SEC_TO_TICK((tvp)->tv_sec) + USEC_TO_TICK((tvp)->tv_usec))
331 
332 #define	TIMESTRUC_TO_TICK(tsp)	\
333 	(SEC_TO_TICK((tsp)->tv_sec) + NSEC_TO_TICK((tsp)->tv_nsec))
334 
335 typedef struct todinfo {
336 	int	tod_sec;	/* seconds 0-59 */
337 	int	tod_min;	/* minutes 0-59 */
338 	int	tod_hour;	/* hours 0-23 */
339 	int	tod_dow;	/* day of week 1-7 */
340 	int	tod_day;	/* day of month 1-31 */
341 	int	tod_month;	/* month 1-12 */
342 	int	tod_year;	/* year 70+ */
343 } todinfo_t;
344 
345 extern	int64_t		timedelta;
346 extern	int		timechanged;
347 extern	int		tod_needsync;
348 extern	kmutex_t	tod_lock;
349 extern	volatile timestruc_t	hrestime;
350 extern	hrtime_t	hres_last_tick;
351 extern	int64_t		hrestime_adj;
352 extern	uint_t		adj_shift;
353 
354 extern	timestruc_t	tod_get(void);
355 extern	void		tod_set(timestruc_t);
356 extern	void		set_hrestime(timestruc_t *);
357 extern	todinfo_t	utc_to_tod(time_t);
358 extern	time_t		tod_to_utc(todinfo_t);
359 extern	int		hr_clock_lock(void);
360 extern	void		hr_clock_unlock(int);
361 extern	hrtime_t	gethrtime(void);
362 extern	hrtime_t	gethrtime_unscaled(void);
363 extern	hrtime_t	gethrtime_max(void);
364 extern	hrtime_t	gethrtime_waitfree(void);
365 extern	void		scalehrtime(hrtime_t *);
366 extern	uint64_t	unscalehrtime(hrtime_t);
367 extern	void		gethrestime(timespec_t *);
368 extern	time_t		gethrestime_sec(void);
369 extern	void		gethrestime_lasttick(timespec_t *);
370 extern	void		hrt2ts(hrtime_t, timestruc_t *);
371 extern	hrtime_t	ts2hrt(const timestruc_t *);
372 extern	void		hrt2tv(hrtime_t, struct timeval *);
373 extern	hrtime_t	tv2hrt(struct timeval *);
374 extern	int		itimerfix(struct timeval *, int);
375 extern	int		itimerdecr(struct itimerval *, int);
376 extern	void		timevaladd(struct timeval *, struct timeval *);
377 extern	void		timevalsub(struct timeval *, struct timeval *);
378 extern	void		timevalfix(struct timeval *);
379 extern	void		dtrace_hres_tick(void);
380 
381 extern clock_t		ddi_get_lbolt(void);
382 extern int64_t		ddi_get_lbolt64(void);
383 
384 #if defined(_SYSCALL32)
385 extern	void		hrt2ts32(hrtime_t, timestruc32_t *);
386 #endif
387 
388 #endif /* _KERNEL */
389 
390 #if !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
391 int adjtime(struct timeval *, struct timeval *);
392 #endif /* !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) ... */
393 
394 #if !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) || \
395 	defined(_ATFILE_SOURCE) || defined(__EXTENSIONS__)
396 int futimesat(int, const char *, const struct timeval *);
397 #endif /* defined(__ATFILE_SOURCE) */
398 
399 #if !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) || \
400 	defined(__EXTENSIONS__)
401 
402 int getitimer(int, struct itimerval *);
403 int utimes(const char *, const struct timeval *);
404 
405 #if defined(_XPG4_2)
406 int setitimer(int, const struct itimerval *_RESTRICT_KYWD,
407 	struct itimerval *_RESTRICT_KYWD);
408 #else
409 int setitimer(int, struct itimerval *_RESTRICT_KYWD,
410 	struct itimerval *_RESTRICT_KYWD);
411 #endif /* defined(_XPG2_2) */
412 
413 #endif /* !defined(_KERNEL) ... defined(_XPG4_2) */
414 
415 #if !defined(_KERNEL) && !defined(_STRICT_SYMBOLS)
416 int futimes(int, const struct timeval *);
417 int lutimes(const char *, const struct timeval *);
418 
419 #define	TIMESPEC_TO_TIMEVAL(tv, ts) { \
420 	(tv)->tv_sec = (ts)->tv_sec; \
421 	(tv)->tv_usec = (ts)->tv_nsec / 1000; \
422 }
423 
424 #define	TIMEVAL_TO_TIMESPEC(tv, ts) { \
425 	(ts)->tv_sec = (tv)->tv_sec; \
426 	(ts)->tv_nsec = (tv)->tv_usec * 1000; \
427 }
428 
429 #endif /* !defined(_KERNEL) && !defined(_STRICT_SYMBOLS) */
430 
431 /*
432  * gettimeofday() and settimeofday() were included in SVr4 due to their
433  * common use in BSD based applications.  They were to be included exactly
434  * as in BSD, with two parameters.  However, AT&T/USL noted that the second
435  * parameter was unused and deleted it, thereby making a routine included
436  * for compatibility, incompatible.
437  *
438  * XSH4.2 (spec 1170) defines gettimeofday and settimeofday to have two
439  * parameters.
440  *
441  * This has caused general disagreement in the application community as to
442  * the syntax of these routines.  Solaris defaults to the XSH4.2 definition.
443  * The flag _SVID_GETTOD may be used to force the SVID version.
444  */
445 #if !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
446 
447 #if defined(_SVID_GETTOD)
448 int settimeofday(struct timeval *);
449 #else
450 int settimeofday(struct timeval *, void *);
451 #endif
452 hrtime_t	gethrtime(void);
453 hrtime_t	gethrvtime(void);
454 
455 #endif /* !(defined _KERNEL) && !defined(__XOPEN_OR_POSIX) ... */
456 
457 #if !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) || defined(_XPG4_2) || \
458 	defined(__EXTENSIONS__)
459 
460 #if defined(_SVID_GETTOD)
461 int gettimeofday(struct timeval *);
462 #else
463 int gettimeofday(struct timeval *_RESTRICT_KYWD, void *_RESTRICT_KYWD);
464 #endif
465 
466 #endif /* !defined(_KERNEL) && !defined(__XOPEN_OR_POSIX) ... */
467 
468 /*
469  * The inclusion of <time.h> is historical and was added for
470  * backward compatibility in delta 1.2 when a number of definitions
471  * were moved out of <sys/time.h>.  More recently, the timespec and
472  * itimerspec structure definitions, along with the _CLOCK_*, CLOCK_*,
473  * _TIMER_*, and TIMER_* symbols were moved to <sys/time_impl.h>,
474  * which is now included by <time.h>.  This change was due to POSIX
475  * 1003.1b-1993 and X/Open UNIX 98 requirements.  For non-POSIX and
476  * non-X/Open applications, including this header will still make
477  * visible these definitions.
478  */
479 #if !defined(_BOOT) && !defined(_KERNEL) && !defined(_FAKE_KERNEL) && \
480 	!defined(__XOPEN_OR_POSIX) || defined(__EXTENSIONS__)
481 #include <time.h>
482 #endif
483 
484 /*
485  * The inclusion of <sys/select.h> is needed for the FD_CLR,
486  * FD_ISSET, FD_SET, and FD_SETSIZE macros as well as the
487  * select() prototype defined in the XOpen specifications
488  * beginning with XSH4v2.  Placement required after definition
489  * for itimerval.
490  */
491 #if !defined(_KERNEL) && !defined(_FAKE_KERNEL) && \
492 	!defined(__XOPEN_OR_POSIX) || \
493 	defined(_XPG4_2) || defined(__EXTENSIONS__)
494 #include <sys/select.h>
495 #endif
496 
497 #endif	/* _ASM */
498 
499 #ifdef	__cplusplus
500 }
501 #endif
502 
503 #endif	/* _SYS_TIME_H */
504