]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/basic/time-util.h
coredump: fix format string on 32 bits
[thirdparty/systemd.git] / src / basic / time-util.h
CommitLineData
9a98c7a1
LP
1#pragma once
2
3/***
4 This file is part of systemd.
5
6 Copyright 2010 Lennart Poettering
7
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
12
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
17
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
20***/
21
cb0dac05 22#include <inttypes.h>
11c3a366
TA
23#include <stdbool.h>
24#include <stddef.h>
25#include <stdint.h>
7c67c79c
HV
26#include <stdio.h>
27#include <time.h>
9a98c7a1
LP
28
29typedef uint64_t usec_t;
30typedef uint64_t nsec_t;
31
ccd06097
ZJS
32#define NSEC_FMT "%" PRIu64
33#define USEC_FMT "%" PRIu64
34
9a98c7a1
LP
35#include "macro.h"
36
37typedef struct dual_timestamp {
38 usec_t realtime;
39 usec_t monotonic;
40} dual_timestamp;
41
fe624c4c
LP
42typedef struct triple_timestamp {
43 usec_t realtime;
44 usec_t monotonic;
45 usec_t boottime;
46} triple_timestamp;
47
3a43da28
KS
48#define USEC_INFINITY ((usec_t) -1)
49#define NSEC_INFINITY ((nsec_t) -1)
50
51#define MSEC_PER_SEC 1000ULL
609e002e
LP
52#define USEC_PER_SEC ((usec_t) 1000000ULL)
53#define USEC_PER_MSEC ((usec_t) 1000ULL)
3a43da28
KS
54#define NSEC_PER_SEC ((nsec_t) 1000000000ULL)
55#define NSEC_PER_MSEC ((nsec_t) 1000000ULL)
56#define NSEC_PER_USEC ((nsec_t) 1000ULL)
609e002e
LP
57
58#define USEC_PER_MINUTE ((usec_t) (60ULL*USEC_PER_SEC))
3a43da28 59#define NSEC_PER_MINUTE ((nsec_t) (60ULL*NSEC_PER_SEC))
609e002e 60#define USEC_PER_HOUR ((usec_t) (60ULL*USEC_PER_MINUTE))
3a43da28 61#define NSEC_PER_HOUR ((nsec_t) (60ULL*NSEC_PER_MINUTE))
609e002e 62#define USEC_PER_DAY ((usec_t) (24ULL*USEC_PER_HOUR))
3a43da28 63#define NSEC_PER_DAY ((nsec_t) (24ULL*NSEC_PER_HOUR))
609e002e 64#define USEC_PER_WEEK ((usec_t) (7ULL*USEC_PER_DAY))
3a43da28 65#define NSEC_PER_WEEK ((nsec_t) (7ULL*NSEC_PER_DAY))
609e002e 66#define USEC_PER_MONTH ((usec_t) (2629800ULL*USEC_PER_SEC))
3a43da28 67#define NSEC_PER_MONTH ((nsec_t) (2629800ULL*NSEC_PER_SEC))
609e002e 68#define USEC_PER_YEAR ((usec_t) (31557600ULL*USEC_PER_SEC))
3a43da28 69#define NSEC_PER_YEAR ((nsec_t) (31557600ULL*NSEC_PER_SEC))
9a98c7a1 70
21b3a0fc
LP
71/* We assume a maximum timezone length of 6. TZNAME_MAX is not defined on Linux, but glibc internally initializes this
72 * to 6. Let's rely on that. */
73#define FORMAT_TIMESTAMP_MAX (3+1+10+1+8+1+6+1+6+1)
c3f84106 74#define FORMAT_TIMESTAMP_WIDTH 28 /* when outputting, assume this width */
9185c8e6 75#define FORMAT_TIMESTAMP_RELATIVE_MAX 256
9a98c7a1
LP
76#define FORMAT_TIMESPAN_MAX 64
77
2d60169d 78#define TIME_T_MAX (time_t)((UINTMAX_C(1) << ((sizeof(time_t) << 3) - 1)) - 1)
e4746b57 79
fe624c4c
LP
80#define DUAL_TIMESTAMP_NULL ((struct dual_timestamp) {})
81#define TRIPLE_TIMESTAMP_NULL ((struct triple_timestamp) {})
842129f5 82
9a98c7a1 83usec_t now(clockid_t clock);
45d7a8bb 84nsec_t now_nsec(clockid_t clock);
9a98c7a1
LP
85
86dual_timestamp* dual_timestamp_get(dual_timestamp *ts);
87dual_timestamp* dual_timestamp_from_realtime(dual_timestamp *ts, usec_t u);
cae0c5e0 88dual_timestamp* dual_timestamp_from_monotonic(dual_timestamp *ts, usec_t u);
fbe55073 89dual_timestamp* dual_timestamp_from_boottime_or_monotonic(dual_timestamp *ts, usec_t u);
9a98c7a1 90
fe624c4c
LP
91triple_timestamp* triple_timestamp_get(triple_timestamp *ts);
92triple_timestamp* triple_timestamp_from_realtime(triple_timestamp *ts, usec_t u);
93
94#define DUAL_TIMESTAMP_HAS_CLOCK(clock) \
95 IN_SET(clock, CLOCK_REALTIME, CLOCK_REALTIME_ALARM, CLOCK_MONOTONIC)
96
97#define TRIPLE_TIMESTAMP_HAS_CLOCK(clock) \
98 IN_SET(clock, CLOCK_REALTIME, CLOCK_REALTIME_ALARM, CLOCK_MONOTONIC, CLOCK_BOOTTIME, CLOCK_BOOTTIME_ALARM)
99
966204e0 100static inline bool dual_timestamp_is_set(dual_timestamp *ts) {
3a43da28
KS
101 return ((ts->realtime > 0 && ts->realtime != USEC_INFINITY) ||
102 (ts->monotonic > 0 && ts->monotonic != USEC_INFINITY));
966204e0 103}
9a98c7a1 104
fe624c4c
LP
105static inline bool triple_timestamp_is_set(triple_timestamp *ts) {
106 return ((ts->realtime > 0 && ts->realtime != USEC_INFINITY) ||
107 (ts->monotonic > 0 && ts->monotonic != USEC_INFINITY) ||
108 (ts->boottime > 0 && ts->boottime != USEC_INFINITY));
109}
110
111usec_t triple_timestamp_by_clock(triple_timestamp *ts, clockid_t clock);
112
44a6b1b6 113usec_t timespec_load(const struct timespec *ts) _pure_;
3a730176 114nsec_t timespec_load_nsec(const struct timespec *ts) _pure_;
9a98c7a1
LP
115struct timespec *timespec_store(struct timespec *ts, usec_t u);
116
44a6b1b6 117usec_t timeval_load(const struct timeval *tv) _pure_;
9a98c7a1
LP
118struct timeval *timeval_store(struct timeval *tv, usec_t u);
119
120char *format_timestamp(char *buf, size_t l, usec_t t);
5ab99e07
LP
121char *format_timestamp_utc(char *buf, size_t l, usec_t t);
122char *format_timestamp_us(char *buf, size_t l, usec_t t);
123char *format_timestamp_us_utc(char *buf, size_t l, usec_t t);
bbb8486e 124char *format_timestamp_relative(char *buf, size_t l, usec_t t);
2fa4092c 125char *format_timespan(char *buf, size_t l, usec_t t, usec_t accuracy);
9a98c7a1
LP
126
127void dual_timestamp_serialize(FILE *f, const char *name, dual_timestamp *t);
e911de99 128int dual_timestamp_deserialize(const char *value, dual_timestamp *t);
b895a735 129int timestamp_deserialize(const char *value, usec_t *timestamp);
9a98c7a1
LP
130
131int parse_timestamp(const char *t, usec_t *usec);
132
7f602784 133int parse_sec(const char *t, usec_t *usec);
519cffec 134int parse_time(const char *t, usec_t *usec, usec_t default_unit);
9a98c7a1 135int parse_nsec(const char *t, nsec_t *nsec);
03cc26dd
LP
136
137bool ntp_synced(void);
75683450
LP
138
139int get_timezones(char ***l);
140bool timezone_is_valid(const char *name);
77ff2de9 141
3411372e 142bool clock_boottime_supported(void);
fe624c4c 143bool clock_supported(clockid_t clock);
77ff2de9 144clockid_t clock_boottime_or_monotonic(void);
babc21fd 145
a8b62610
MS
146#define xstrftime(buf, fmt, tm) \
147 assert_message_se(strftime(buf, ELEMENTSOF(buf), fmt, tm) > 0, \
148 "xstrftime: " #buf "[] must be big enough")
5c904ba5
LP
149
150int get_timezone(char **timezone);
7c67c79c
HV
151
152time_t mktime_or_timegm(struct tm *tm, bool utc);
153struct tm *localtime_or_gmtime_r(const time_t *t, struct tm *tm, bool utc);
87b8ce69
SS
154
155unsigned long usec_to_jiffies(usec_t usec);
53f555b6
LP
156
157static inline usec_t usec_add(usec_t a, usec_t b) {
158 usec_t c;
159
160 /* Adds two time values, and makes sure USEC_INFINITY as input results as USEC_INFINITY in output, and doesn't
161 * overflow. */
162
163 c = a + b;
164 if (c < a || c < b) /* overflow check */
165 return USEC_INFINITY;
166
167 return c;
168}
5d634ca8
AK
169
170static inline usec_t usec_sub(usec_t timestamp, int64_t delta) {
171 if (delta < 0)
04a1d84c 172 return usec_add(timestamp, (usec_t) (-delta));
5d634ca8 173
04a1d84c
LP
174 if (timestamp == USEC_INFINITY) /* Make sure infinity doesn't degrade */
175 return USEC_INFINITY;
176
177 if (timestamp < (usec_t) delta)
178 return 0;
179
180 return timestamp - delta;
5d634ca8 181}