mirror of
https://github.com/systemd/systemd.git
synced 2025-01-10 05:18:17 +03:00
Merge pull request #34155 from poettering/gmtime-safe
handle gmtime_r() errors more robustly
This commit is contained in:
commit
ccd62e6d4d
@ -536,8 +536,8 @@ static int write_to_syslog(
|
||||
char header_priority[2 + DECIMAL_STR_MAX(int) + 1],
|
||||
header_time[64],
|
||||
header_pid[4 + DECIMAL_STR_MAX(pid_t) + 1];
|
||||
time_t t;
|
||||
struct tm tm;
|
||||
int r;
|
||||
|
||||
if (syslog_fd < 0)
|
||||
return 0;
|
||||
@ -547,9 +547,9 @@ static int write_to_syslog(
|
||||
|
||||
xsprintf(header_priority, "<%i>", level);
|
||||
|
||||
t = (time_t) (now(CLOCK_REALTIME) / USEC_PER_SEC);
|
||||
if (!localtime_r(&t, &tm))
|
||||
return -EINVAL;
|
||||
r = localtime_or_gmtime_usec(now(CLOCK_REALTIME), /* utc= */ false, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (strftime(header_time, sizeof(header_time), "%h %e %T ", &tm) <= 0)
|
||||
return -EINVAL;
|
||||
|
@ -450,24 +450,29 @@ int os_release_support_ended(const char *support_end, bool quiet, usec_t *ret_eo
|
||||
support_end = _support_end_alloc;
|
||||
}
|
||||
|
||||
if (isempty(support_end)) /* An empty string is a explicit way to say "no EOL exists" */
|
||||
if (isempty(support_end)) { /* An empty string is a explicit way to say "no EOL exists" */
|
||||
if (ret_eol)
|
||||
*ret_eol = USEC_INFINITY;
|
||||
|
||||
return false; /* no end date defined */
|
||||
}
|
||||
|
||||
struct tm tm = {};
|
||||
const char *k = strptime(support_end, "%Y-%m-%d", &tm);
|
||||
if (!k || *k)
|
||||
return log_full_errno(quiet ? LOG_DEBUG : LOG_WARNING, SYNTHETIC_ERRNO(EINVAL),
|
||||
"Failed to parse SUPPORT_END= in os-release file, ignoring: %m");
|
||||
"Failed to parse SUPPORT_END= from os-release file, ignoring: %s", support_end);
|
||||
|
||||
time_t eol = timegm(&tm);
|
||||
if (eol == (time_t) -1)
|
||||
return log_full_errno(quiet ? LOG_DEBUG : LOG_WARNING, SYNTHETIC_ERRNO(EINVAL),
|
||||
"Failed to convert SUPPORT_END= in os-release file, ignoring: %m");
|
||||
usec_t eol;
|
||||
r = mktime_or_timegm_usec(&tm, /* utc= */ true, &eol);
|
||||
if (r < 0)
|
||||
return log_full_errno(quiet ? LOG_DEBUG : LOG_WARNING, r,
|
||||
"Failed to convert SUPPORT_END= time from os-release file, ignoring: %m");
|
||||
|
||||
if (ret_eol)
|
||||
*ret_eol = eol * USEC_PER_SEC;
|
||||
*ret_eol = eol;
|
||||
|
||||
return DIV_ROUND_UP(now(CLOCK_REALTIME), USEC_PER_SEC) > (usec_t) eol;
|
||||
return now(CLOCK_REALTIME) > eol;
|
||||
}
|
||||
|
||||
const char* os_release_pretty_name(const char *pretty_name, const char *name) {
|
||||
|
@ -332,7 +332,6 @@ char* format_timestamp_style(
|
||||
|
||||
struct tm tm;
|
||||
bool utc, us;
|
||||
time_t sec;
|
||||
size_t n;
|
||||
|
||||
assert(buf);
|
||||
@ -375,9 +374,7 @@ char* format_timestamp_style(
|
||||
return strcpy(buf, xxx[style]);
|
||||
}
|
||||
|
||||
sec = (time_t) (t / USEC_PER_SEC); /* Round down */
|
||||
|
||||
if (!localtime_or_gmtime_r(&sec, &tm, utc))
|
||||
if (localtime_or_gmtime_usec(t, utc, &tm) < 0)
|
||||
return NULL;
|
||||
|
||||
/* Start with the week day */
|
||||
@ -665,7 +662,6 @@ static int parse_timestamp_impl(
|
||||
unsigned fractional = 0;
|
||||
const char *k;
|
||||
struct tm tm, copy;
|
||||
time_t sec;
|
||||
|
||||
/* Allowed syntaxes:
|
||||
*
|
||||
@ -778,10 +774,9 @@ static int parse_timestamp_impl(
|
||||
}
|
||||
}
|
||||
|
||||
sec = (time_t) (usec / USEC_PER_SEC);
|
||||
|
||||
if (!localtime_or_gmtime_r(&sec, &tm, utc))
|
||||
return -EINVAL;
|
||||
r = localtime_or_gmtime_usec(usec, utc, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
tm.tm_isdst = isdst;
|
||||
|
||||
@ -939,11 +934,11 @@ from_tm:
|
||||
} else
|
||||
minus = gmtoff * USEC_PER_SEC;
|
||||
|
||||
sec = mktime_or_timegm(&tm, utc);
|
||||
if (sec < 0)
|
||||
return -EINVAL;
|
||||
r = mktime_or_timegm_usec(&tm, utc, &usec);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
usec = usec_add(sec * USEC_PER_SEC, fractional);
|
||||
usec = usec_add(usec, fractional);
|
||||
|
||||
finish:
|
||||
usec = usec_add(usec, plus);
|
||||
@ -1625,17 +1620,54 @@ int get_timezone(char **ret) {
|
||||
return strdup_to(ret, e);
|
||||
}
|
||||
|
||||
time_t mktime_or_timegm(struct tm *tm, bool utc) {
|
||||
int mktime_or_timegm_usec(
|
||||
struct tm *tm, /* input + normalized output */
|
||||
bool utc,
|
||||
usec_t *ret) {
|
||||
|
||||
time_t t;
|
||||
|
||||
assert(tm);
|
||||
|
||||
return utc ? timegm(tm) : mktime(tm);
|
||||
if (tm->tm_year < 69) /* early check for negative (i.e. before 1970) time_t (Note that in some timezones the epoch is in the year 1969!)*/
|
||||
return -ERANGE;
|
||||
if ((usec_t) tm->tm_year > CONST_MIN(USEC_INFINITY / USEC_PER_YEAR, (usec_t) TIME_T_MAX / (365U * 24U * 60U * 60U)) - 1900) /* early check for possible overrun of usec_t or time_t */
|
||||
return -ERANGE;
|
||||
|
||||
/* timegm()/mktime() is a bit weird to use, since it returns -1 in two cases: on error as well as a
|
||||
* valid time indicating one second before the UNIX epoch. Let's treat both cases the same here, and
|
||||
* return -ERANGE for anything negative, since usec_t is unsigned, and we can thus not express
|
||||
* negative times anyway. */
|
||||
|
||||
t = utc ? timegm(tm) : mktime(tm);
|
||||
if (t < 0) /* Refuse negative times and errors */
|
||||
return -ERANGE;
|
||||
if ((usec_t) t >= USEC_INFINITY / USEC_PER_SEC) /* Never return USEC_INFINITY by accident (or overflow) */
|
||||
return -ERANGE;
|
||||
|
||||
if (ret)
|
||||
*ret = (usec_t) t * USEC_PER_SEC;
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct tm *localtime_or_gmtime_r(const time_t *t, struct tm *tm, bool utc) {
|
||||
assert(t);
|
||||
assert(tm);
|
||||
int localtime_or_gmtime_usec(
|
||||
usec_t t,
|
||||
bool utc,
|
||||
struct tm *ret) {
|
||||
|
||||
return utc ? gmtime_r(t, tm) : localtime_r(t, tm);
|
||||
t /= USEC_PER_SEC; /* Round down */
|
||||
if (t > (usec_t) TIME_T_MAX)
|
||||
return -ERANGE;
|
||||
time_t sec = (time_t) t;
|
||||
|
||||
struct tm buf = {};
|
||||
if (!(utc ? gmtime_r(&sec, &buf) : localtime_r(&sec, &buf)))
|
||||
return -EINVAL;
|
||||
|
||||
if (ret)
|
||||
*ret = buf;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static uint32_t sysconf_clock_ticks_cached(void) {
|
||||
|
@ -177,8 +177,8 @@ usec_t usec_shift_clock(usec_t, clockid_t from, clockid_t to);
|
||||
|
||||
int get_timezone(char **ret);
|
||||
|
||||
time_t mktime_or_timegm(struct tm *tm, bool utc);
|
||||
struct tm *localtime_or_gmtime_r(const time_t *t, struct tm *tm, bool utc);
|
||||
int mktime_or_timegm_usec(struct tm *tm, bool utc, usec_t *ret);
|
||||
int localtime_or_gmtime_usec(usec_t t, bool utc, struct tm *ret);
|
||||
|
||||
uint32_t usec_to_jiffies(usec_t usec);
|
||||
usec_t jiffies_to_usec(uint32_t jiffies);
|
||||
|
@ -342,14 +342,15 @@ static int get_firmware_date(usec_t *ret) {
|
||||
.tm_mon = m,
|
||||
.tm_year = y,
|
||||
};
|
||||
time_t v = timegm(&tm);
|
||||
if (v == (time_t) -1)
|
||||
return -errno;
|
||||
|
||||
usec_t v;
|
||||
r = mktime_or_timegm_usec(&tm, /* utc= */ true, &v);
|
||||
if (r < 0)
|
||||
return r;
|
||||
if (tm.tm_mday != (int) d || tm.tm_mon != (int) m || tm.tm_year != (int) y)
|
||||
return -EINVAL; /* date was not normalized? (e.g. "30th of feb") */
|
||||
|
||||
*ret = (usec_t) v * USEC_PER_SEC;
|
||||
|
||||
*ret = v;
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -384,7 +384,6 @@ int curl_parse_http_time(const char *t, usec_t *ret) {
|
||||
_cleanup_(freelocalep) locale_t loc = (locale_t) 0;
|
||||
const char *e;
|
||||
struct tm tm;
|
||||
time_t v;
|
||||
|
||||
assert(t);
|
||||
assert(ret);
|
||||
@ -404,10 +403,5 @@ int curl_parse_http_time(const char *t, usec_t *ret) {
|
||||
if (!e || *e != 0)
|
||||
return -EINVAL;
|
||||
|
||||
v = timegm(&tm);
|
||||
if (v == (time_t) -1)
|
||||
return -EINVAL;
|
||||
|
||||
*ret = (usec_t) v * USEC_PER_SEC;
|
||||
return 0;
|
||||
return mktime_or_timegm_usec(&tm, /* usec= */ true, ret);
|
||||
}
|
||||
|
@ -127,7 +127,6 @@ void server_forward_syslog(Server *s, int priority, const char *identifier, cons
|
||||
char header_priority[DECIMAL_STR_MAX(priority) + 3], header_time[64],
|
||||
header_pid[STRLEN("[]: ") + DECIMAL_STR_MAX(pid_t) + 1];
|
||||
int n = 0;
|
||||
time_t t;
|
||||
struct tm tm;
|
||||
_cleanup_free_ char *ident_buf = NULL;
|
||||
|
||||
@ -144,8 +143,7 @@ void server_forward_syslog(Server *s, int priority, const char *identifier, cons
|
||||
iovec[n++] = IOVEC_MAKE_STRING(header_priority);
|
||||
|
||||
/* Second: timestamp */
|
||||
t = tv ? tv->tv_sec : ((time_t) (now(CLOCK_REALTIME) / USEC_PER_SEC));
|
||||
if (!localtime_r(&t, &tm))
|
||||
if (localtime_or_gmtime_usec(tv ? tv->tv_sec * USEC_PER_SEC : now(CLOCK_REALTIME), /* utc= */ false, &tm) < 0)
|
||||
return;
|
||||
if (strftime(header_time, sizeof(header_time), "%h %e %T ", &tm) <= 0)
|
||||
return;
|
||||
|
@ -793,12 +793,14 @@ static char* format_location(uint32_t latitude, uint32_t longitude, uint32_t alt
|
||||
|
||||
static int format_timestamp_dns(char *buf, size_t l, time_t sec) {
|
||||
struct tm tm;
|
||||
int r;
|
||||
|
||||
assert(buf);
|
||||
assert(l > STRLEN("YYYYMMDDHHmmSS"));
|
||||
|
||||
if (!gmtime_r(&sec, &tm))
|
||||
return -EINVAL;
|
||||
r = localtime_or_gmtime_usec(sec * USEC_PER_SEC, /* utc= */ true, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (strftime(buf, l, "%Y%m%d%H%M%S", &tm) <= 0)
|
||||
return -EINVAL;
|
||||
|
@ -576,9 +576,13 @@ static int calendarspec_from_time_t(CalendarSpec *c, time_t time) {
|
||||
struct tm tm;
|
||||
int r;
|
||||
|
||||
if (!gmtime_r(&time, &tm))
|
||||
if ((usec_t) time > USEC_INFINITY / USEC_PER_SEC)
|
||||
return -ERANGE;
|
||||
|
||||
r = localtime_or_gmtime_usec((usec_t) time * USEC_PER_SEC, /* utc= */ true, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (tm.tm_year > INT_MAX - 1900)
|
||||
return -ERANGE;
|
||||
|
||||
@ -1094,12 +1098,12 @@ int calendar_spec_from_string(const char *p, CalendarSpec **ret) {
|
||||
}
|
||||
|
||||
static int find_end_of_month(const struct tm *tm, bool utc, int day) {
|
||||
struct tm t = *tm;
|
||||
struct tm t = *ASSERT_PTR(tm);
|
||||
|
||||
t.tm_mon++;
|
||||
t.tm_mday = 1 - day;
|
||||
|
||||
if (mktime_or_timegm(&t, utc) < 0 ||
|
||||
if (mktime_or_timegm_usec(&t, utc, /* ret= */ NULL) < 0 ||
|
||||
t.tm_mon != tm->tm_mon)
|
||||
return -1;
|
||||
|
||||
@ -1171,8 +1175,8 @@ static int find_matching_component(
|
||||
}
|
||||
|
||||
static int tm_within_bounds(struct tm *tm, bool utc) {
|
||||
struct tm t;
|
||||
int cmp;
|
||||
int r;
|
||||
|
||||
assert(tm);
|
||||
|
||||
/*
|
||||
@ -1183,9 +1187,10 @@ static int tm_within_bounds(struct tm *tm, bool utc) {
|
||||
if (tm->tm_year + 1900 > MAX_YEAR)
|
||||
return -ERANGE;
|
||||
|
||||
t = *tm;
|
||||
if (mktime_or_timegm(&t, utc) < 0)
|
||||
return negative_errno();
|
||||
struct tm t = *tm;
|
||||
r = mktime_or_timegm_usec(&t, utc, /* ret= */ NULL);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
/*
|
||||
* Did any normalization take place? If so, it was out of bounds before.
|
||||
@ -1194,6 +1199,7 @@ static int tm_within_bounds(struct tm *tm, bool utc) {
|
||||
* out of bounds. Normalization has occurred implies find_matching_component() > 0,
|
||||
* other sub time units are already reset in find_next().
|
||||
*/
|
||||
int cmp;
|
||||
if ((cmp = CMP(t.tm_year, tm->tm_year)) != 0)
|
||||
t.tm_mon = 0;
|
||||
else if ((cmp = CMP(t.tm_mon, tm->tm_mon)) != 0)
|
||||
@ -1222,7 +1228,7 @@ static bool matches_weekday(int weekdays_bits, const struct tm *tm, bool utc) {
|
||||
return true;
|
||||
|
||||
t = *tm;
|
||||
if (mktime_or_timegm(&t, utc) < 0)
|
||||
if (mktime_or_timegm_usec(&t, utc, /* ret= */ NULL) < 0)
|
||||
return false;
|
||||
|
||||
k = t.tm_wday == 0 ? 6 : t.tm_wday - 1;
|
||||
@ -1248,7 +1254,7 @@ static int find_next(const CalendarSpec *spec, struct tm *tm, usec_t *usec) {
|
||||
|
||||
for (unsigned iteration = 0; iteration < MAX_CALENDAR_ITERATIONS; iteration++) {
|
||||
/* Normalize the current date */
|
||||
(void) mktime_or_timegm(&c, spec->utc);
|
||||
(void) mktime_or_timegm_usec(&c, spec->utc, /* ret= */ NULL);
|
||||
c.tm_isdst = spec->dst;
|
||||
|
||||
c.tm_year += 1900;
|
||||
@ -1354,10 +1360,9 @@ static int find_next(const CalendarSpec *spec, struct tm *tm, usec_t *usec) {
|
||||
}
|
||||
|
||||
static int calendar_spec_next_usec_impl(const CalendarSpec *spec, usec_t usec, usec_t *ret_next) {
|
||||
struct tm tm;
|
||||
time_t t;
|
||||
int r;
|
||||
usec_t tm_usec;
|
||||
struct tm tm;
|
||||
int r;
|
||||
|
||||
assert(spec);
|
||||
|
||||
@ -1365,20 +1370,22 @@ static int calendar_spec_next_usec_impl(const CalendarSpec *spec, usec_t usec, u
|
||||
return -EINVAL;
|
||||
|
||||
usec++;
|
||||
t = (time_t) (usec / USEC_PER_SEC);
|
||||
assert_se(localtime_or_gmtime_r(&t, &tm, spec->utc));
|
||||
r = localtime_or_gmtime_usec(usec, spec->utc, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
tm_usec = usec % USEC_PER_SEC;
|
||||
|
||||
r = find_next(spec, &tm, &tm_usec);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
t = mktime_or_timegm(&tm, spec->utc);
|
||||
if (t < 0)
|
||||
return -EINVAL;
|
||||
usec_t t;
|
||||
r = mktime_or_timegm_usec(&tm, spec->utc, &t);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (ret_next)
|
||||
*ret_next = (usec_t) t * USEC_PER_SEC + tm_usec;
|
||||
*ret_next = t + tm_usec;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -53,11 +53,13 @@ int clock_is_localtime(const char *adjtime_path) {
|
||||
}
|
||||
|
||||
int clock_set_timezone(int *ret_minutesdelta) {
|
||||
struct timespec ts;
|
||||
struct tm tm;
|
||||
int r;
|
||||
|
||||
r = localtime_or_gmtime_usec(now(CLOCK_REALTIME), /* utc= */ false, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
assert_se(clock_gettime(CLOCK_REALTIME, &ts) == 0);
|
||||
assert_se(localtime_r(&ts.tv_sec, &tm));
|
||||
int minutesdelta = tm.tm_gmtoff / 60;
|
||||
|
||||
struct timezone tz = {
|
||||
|
@ -368,6 +368,7 @@ static int output_timestamp_realtime(
|
||||
usec_t usec) {
|
||||
|
||||
char buf[CONST_MAX(FORMAT_TIMESTAMP_MAX, 64U)];
|
||||
int r;
|
||||
|
||||
assert(f);
|
||||
assert(j);
|
||||
@ -375,65 +376,76 @@ static int output_timestamp_realtime(
|
||||
if (!VALID_REALTIME(usec))
|
||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "No valid realtime timestamp available.");
|
||||
|
||||
if (IN_SET(mode, OUTPUT_SHORT_FULL, OUTPUT_WITH_UNIT)) {
|
||||
const char *k;
|
||||
switch (mode) {
|
||||
|
||||
if (flags & OUTPUT_UTC)
|
||||
k = format_timestamp_style(buf, sizeof(buf), usec, TIMESTAMP_UTC);
|
||||
else
|
||||
k = format_timestamp(buf, sizeof(buf), usec);
|
||||
if (!k)
|
||||
case OUTPUT_SHORT_FULL:
|
||||
case OUTPUT_WITH_UNIT: {
|
||||
if (!format_timestamp_style(buf, sizeof(buf), usec, flags & OUTPUT_UTC ? TIMESTAMP_UTC : TIMESTAMP_PRETTY))
|
||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL),
|
||||
"Failed to format timestamp: %" PRIu64, usec);
|
||||
break;
|
||||
}
|
||||
|
||||
} else {
|
||||
case OUTPUT_SHORT_UNIX:
|
||||
xsprintf(buf, "%10" PRI_USEC ".%06" PRI_USEC, usec / USEC_PER_SEC, usec % USEC_PER_SEC);
|
||||
break;
|
||||
|
||||
case OUTPUT_SHORT:
|
||||
case OUTPUT_SHORT_PRECISE:
|
||||
case OUTPUT_SHORT_ISO:
|
||||
case OUTPUT_SHORT_ISO_PRECISE: {
|
||||
struct tm tm;
|
||||
time_t t;
|
||||
size_t tail = 0;
|
||||
|
||||
t = (time_t) (usec / USEC_PER_SEC);
|
||||
|
||||
switch (mode) {
|
||||
|
||||
case OUTPUT_SHORT_UNIX:
|
||||
xsprintf(buf, "%10"PRI_TIME".%06"PRIu64, t, usec % USEC_PER_SEC);
|
||||
break;
|
||||
|
||||
case OUTPUT_SHORT_ISO:
|
||||
case OUTPUT_SHORT_ISO_PRECISE: {
|
||||
size_t tail = strftime(buf, sizeof(buf), "%Y-%m-%dT%H:%M:%S",
|
||||
localtime_or_gmtime_r(&t, &tm, flags & OUTPUT_UTC));
|
||||
if (tail == 0)
|
||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "Failed to format ISO time.");
|
||||
|
||||
/* No usec in strftime, need to append */
|
||||
if (mode == OUTPUT_SHORT_ISO_PRECISE) {
|
||||
assert_se(snprintf_ok(buf + tail, ELEMENTSOF(buf) - tail, ".%06"PRI_USEC, usec % USEC_PER_SEC));
|
||||
tail += 7;
|
||||
}
|
||||
|
||||
int h = tm.tm_gmtoff / 60 / 60;
|
||||
int m = labs((tm.tm_gmtoff / 60) % 60);
|
||||
snprintf(buf + tail, ELEMENTSOF(buf) - tail, "%+03d:%02d", h, m);
|
||||
break;
|
||||
r = localtime_or_gmtime_usec(usec, flags & OUTPUT_UTC, &tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to convert timestamp to calendar time, generating fallback timestamp: %m");
|
||||
else {
|
||||
tail = strftime(
|
||||
buf, sizeof(buf),
|
||||
IN_SET(mode, OUTPUT_SHORT_ISO, OUTPUT_SHORT_ISO_PRECISE) ? "%Y-%m-%dT%H:%M:%S" : "%b %d %H:%M:%S",
|
||||
&tm);
|
||||
if (tail <= 0)
|
||||
log_debug("Failed to format calendar time, generating fallback timestamp.");
|
||||
}
|
||||
|
||||
case OUTPUT_SHORT:
|
||||
case OUTPUT_SHORT_PRECISE:
|
||||
if (tail <= 0) {
|
||||
/* Generate fallback timestamp if regular formatting didn't work. (This might happen on systems where time_t is 32bit) */
|
||||
|
||||
if (strftime(buf, sizeof(buf), "%b %d %H:%M:%S",
|
||||
localtime_or_gmtime_r(&t, &tm, flags & OUTPUT_UTC)) <= 0)
|
||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "Failed to format syslog time.");
|
||||
static const char *const xxx[_OUTPUT_MODE_MAX] = {
|
||||
[OUTPUT_SHORT] = "XXX XX XX:XX:XX",
|
||||
[OUTPUT_SHORT_PRECISE] = "XXX XX XX:XX:XX.XXXXXX",
|
||||
[OUTPUT_SHORT_ISO] = "XXXX-XX-XXTXX:XX:XX+XX:XX",
|
||||
[OUTPUT_SHORT_ISO_PRECISE] = "XXXX-XX-XXTXX:XX:XX.XXXXXX+XX:XX",
|
||||
};
|
||||
|
||||
if (mode == OUTPUT_SHORT_PRECISE) {
|
||||
assert(sizeof(buf) > strlen(buf));
|
||||
if (!snprintf_ok(buf + strlen(buf), sizeof(buf) - strlen(buf), ".%06"PRIu64, usec % USEC_PER_SEC))
|
||||
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "Failed to format precise time.");
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
assert_not_reached();
|
||||
fputs(xxx[mode], f);
|
||||
return strlen(xxx[mode]);
|
||||
}
|
||||
|
||||
assert(tail <= sizeof(buf));
|
||||
|
||||
/* No usec in strftime, need to append */
|
||||
if (IN_SET(mode, OUTPUT_SHORT_ISO_PRECISE, OUTPUT_SHORT_PRECISE)) {
|
||||
assert_se(snprintf_ok(buf + tail, sizeof(buf) - tail, ".%06" PRI_USEC, usec % USEC_PER_SEC));
|
||||
|
||||
tail += 7;
|
||||
|
||||
assert(tail <= sizeof(buf));
|
||||
}
|
||||
|
||||
if (IN_SET(mode, OUTPUT_SHORT_ISO, OUTPUT_SHORT_ISO_PRECISE)) {
|
||||
int h = tm.tm_gmtoff / 60 / 60,
|
||||
m = abs((int) ((tm.tm_gmtoff / 60) % 60));
|
||||
|
||||
assert_se(snprintf_ok(buf + tail, sizeof(buf) - tail, "%+03d:%02d", h, m));
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
assert_not_reached();
|
||||
}
|
||||
|
||||
fputs(buf, f);
|
||||
|
@ -58,6 +58,8 @@ int talk_initctl(char rl) {
|
||||
}
|
||||
|
||||
int parse_shutdown_time_spec(const char *t, usec_t *ret) {
|
||||
int r;
|
||||
|
||||
assert(t);
|
||||
assert(ret);
|
||||
|
||||
@ -73,9 +75,6 @@ int parse_shutdown_time_spec(const char *t, usec_t *ret) {
|
||||
} else {
|
||||
char *e = NULL;
|
||||
long hour, minute;
|
||||
struct tm tm = {};
|
||||
time_t s;
|
||||
usec_t n;
|
||||
|
||||
errno = 0;
|
||||
hour = strtol(t, &e, 10);
|
||||
@ -86,22 +85,26 @@ int parse_shutdown_time_spec(const char *t, usec_t *ret) {
|
||||
if (errno > 0 || *e != 0 || minute < 0 || minute > 59)
|
||||
return -EINVAL;
|
||||
|
||||
n = now(CLOCK_REALTIME);
|
||||
s = (time_t) (n / USEC_PER_SEC);
|
||||
usec_t n = now(CLOCK_REALTIME);
|
||||
struct tm tm = {};
|
||||
|
||||
assert_se(localtime_r(&s, &tm));
|
||||
r = localtime_or_gmtime_usec(n, /* utc= */ false, &tm);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
tm.tm_hour = (int) hour;
|
||||
tm.tm_min = (int) minute;
|
||||
tm.tm_sec = 0;
|
||||
|
||||
s = mktime(&tm);
|
||||
assert(s >= 0);
|
||||
usec_t s;
|
||||
r = mktime_or_timegm_usec(&tm, /* utc= */ false, &s);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*ret = (usec_t) s * USEC_PER_SEC;
|
||||
while (s <= n)
|
||||
s += USEC_PER_DAY;
|
||||
|
||||
while (*ret <= n)
|
||||
*ret += USEC_PER_DAY;
|
||||
*ret = s;
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@ -721,7 +721,7 @@ static void test_parse_timestamp_impl(const char *tz) {
|
||||
test_parse_timestamp_one("1996-12-20T00:39:57Z", 0, 851042397 * USEC_PER_SEC + 000000);
|
||||
test_parse_timestamp_one("1990-12-31T23:59:60Z", 0, 662688000 * USEC_PER_SEC + 000000);
|
||||
test_parse_timestamp_one("1990-12-31T15:59:60-08:00", 0, 662688000 * USEC_PER_SEC + 000000);
|
||||
assert_se(parse_timestamp("1937-01-01T12:00:27.87+00:20", NULL) == -EINVAL); /* we don't support pre-epoch timestamps */
|
||||
assert_se(parse_timestamp("1937-01-01T12:00:27.87+00:20", NULL) == -ERANGE); /* we don't support pre-epoch timestamps */
|
||||
/* We accept timestamps without seconds as well */
|
||||
test_parse_timestamp_one("1996-12-20T00:39Z", 0, (851042397 - 57) * USEC_PER_SEC + 000000);
|
||||
test_parse_timestamp_one("1990-12-31T15:59-08:00", 0, (662688000-60) * USEC_PER_SEC + 000000);
|
||||
@ -1170,6 +1170,33 @@ TEST(timezone_offset_change) {
|
||||
tzset();
|
||||
}
|
||||
|
||||
static usec_t absdiff(usec_t a, usec_t b) {
|
||||
return a > b ? a - b : b - a;
|
||||
}
|
||||
|
||||
TEST(mktime_or_timegm_usec) {
|
||||
|
||||
usec_t n = now(CLOCK_REALTIME), m;
|
||||
struct tm tm;
|
||||
|
||||
assert_se(localtime_or_gmtime_usec(n, /* utc= */ false, &tm) >= 0);
|
||||
assert_se(mktime_or_timegm_usec(&tm, /* utc= */ false, &m) >= 0);
|
||||
assert_se(absdiff(n, m) < 2 * USEC_PER_DAY);
|
||||
|
||||
assert_se(localtime_or_gmtime_usec(n, /* utc= */ true, &tm) >= 0);
|
||||
assert_se(mktime_or_timegm_usec(&tm, /* utc= */ true, &m) >= 0);
|
||||
assert_se(absdiff(n, m) < USEC_PER_SEC);
|
||||
|
||||
/* This definitely should fail, because we refuse dates before the UNIX epoch */
|
||||
tm = (struct tm) {
|
||||
.tm_mday = 15,
|
||||
.tm_mon = 11,
|
||||
.tm_year = 1969 - 1900,
|
||||
};
|
||||
|
||||
assert_se(mktime_or_timegm_usec(&tm, /* utc= */ true, NULL) == -ERANGE);
|
||||
}
|
||||
|
||||
static int intro(void) {
|
||||
/* Tests have hard-coded results that do not expect a specific timezone to be set by the caller */
|
||||
assert_se(unsetenv("TZ") >= 0);
|
||||
|
@ -9,24 +9,23 @@
|
||||
#include "fd-util.h"
|
||||
#include "hwclock-util.h"
|
||||
|
||||
int hwclock_get(struct tm *ret) {
|
||||
int hwclock_get(struct tm *tm /* input + output! */) {
|
||||
_cleanup_close_ int fd = -EBADF;
|
||||
|
||||
assert(ret);
|
||||
assert(tm);
|
||||
|
||||
fd = open("/dev/rtc", O_RDONLY|O_CLOEXEC);
|
||||
if (fd < 0)
|
||||
return -errno;
|
||||
|
||||
/* This leaves the timezone fields of struct ret uninitialized! */
|
||||
if (ioctl(fd, RTC_RD_TIME, ret) < 0)
|
||||
if (ioctl(fd, RTC_RD_TIME, tm) < 0)
|
||||
/* Some drivers return -EINVAL in case the time could not be kept, i.e. power loss
|
||||
* happened. Let's turn that into a clearly recognizable error */
|
||||
return errno == EINVAL ? -ENODATA : -errno;
|
||||
|
||||
/* We don't know daylight saving, so we reset this in order not
|
||||
* to confuse mktime(). */
|
||||
ret->tm_isdst = -1;
|
||||
/* We don't know daylight saving, so we reset this in order not to confuse mktime(). */
|
||||
tm->tm_isdst = -1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -3,5 +3,5 @@
|
||||
|
||||
#include <time.h>
|
||||
|
||||
int hwclock_get(struct tm *ret);
|
||||
int hwclock_get(struct tm *tm);
|
||||
int hwclock_set(const struct tm *tm);
|
||||
|
@ -55,7 +55,7 @@ static int print_status_info(const StatusInfo *i) {
|
||||
char a[LINE_MAX];
|
||||
TableCell *cell;
|
||||
struct tm tm;
|
||||
time_t sec;
|
||||
usec_t t;
|
||||
size_t n;
|
||||
int r;
|
||||
|
||||
@ -84,22 +84,38 @@ static int print_status_info(const StatusInfo *i) {
|
||||
tzset();
|
||||
|
||||
if (i->time != 0) {
|
||||
sec = (time_t) (i->time / USEC_PER_SEC);
|
||||
t = i->time;
|
||||
have_time = true;
|
||||
} else if (IN_SET(arg_transport, BUS_TRANSPORT_LOCAL, BUS_TRANSPORT_MACHINE)) {
|
||||
sec = time(NULL);
|
||||
t = now(CLOCK_REALTIME);
|
||||
have_time = true;
|
||||
} else
|
||||
log_warning("Could not get time from timedated and not operating locally, ignoring.");
|
||||
|
||||
n = have_time ? strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S %Z", localtime_r(&sec, &tm)) : 0;
|
||||
if (have_time) {
|
||||
r = localtime_or_gmtime_usec(t, /* utc= */ false, &tm);
|
||||
if (r < 0) {
|
||||
log_warning_errno(r, "Failed to convert system time to local time, ignoring: %m");
|
||||
n = 0;
|
||||
} else
|
||||
n = strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S %Z", &tm);
|
||||
} else
|
||||
n = 0;
|
||||
r = table_add_many(table,
|
||||
TABLE_FIELD, "Local time",
|
||||
TABLE_STRING, n > 0 ? a : "n/a");
|
||||
if (r < 0)
|
||||
return table_log_add_error(r);
|
||||
|
||||
n = have_time ? strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S UTC", gmtime_r(&sec, &tm)) : 0;
|
||||
if (have_time) {
|
||||
r = localtime_or_gmtime_usec(t, /* utc= */ true, &tm);
|
||||
if (r < 0) {
|
||||
log_warning_errno(r, "Failed to convert system time to universal time, ignoring: %m");
|
||||
n = 0;
|
||||
} else
|
||||
n = strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S UTC", &tm);
|
||||
} else
|
||||
n = 0;
|
||||
r = table_add_many(table,
|
||||
TABLE_FIELD, "Universal time",
|
||||
TABLE_STRING, n > 0 ? a : "n/a");
|
||||
@ -107,10 +123,12 @@ static int print_status_info(const StatusInfo *i) {
|
||||
return table_log_add_error(r);
|
||||
|
||||
if (i->rtc_time > 0) {
|
||||
time_t rtc_sec;
|
||||
|
||||
rtc_sec = (time_t) (i->rtc_time / USEC_PER_SEC);
|
||||
n = strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S", gmtime_r(&rtc_sec, &tm));
|
||||
r = localtime_or_gmtime_usec(i->rtc_time, /* utc= */ true, &tm);
|
||||
if (r < 0) {
|
||||
log_warning_errno(r, "Failed to convert RTC time to universal time, ignoring: %m");
|
||||
n = 0;
|
||||
} else
|
||||
n = strftime(a, sizeof a, "%a %Y-%m-%d %H:%M:%S", &tm);
|
||||
} else
|
||||
n = 0;
|
||||
r = table_add_many(table,
|
||||
@ -122,8 +140,15 @@ static int print_status_info(const StatusInfo *i) {
|
||||
r = table_add_cell(table, NULL, TABLE_FIELD, "Time zone");
|
||||
if (r < 0)
|
||||
return table_log_add_error(r);
|
||||
|
||||
n = have_time ? strftime(a, sizeof a, "%Z, %z", localtime_r(&sec, &tm)) : 0;
|
||||
if (have_time) {
|
||||
r = localtime_or_gmtime_usec(t, /* utc= */ false, &tm);
|
||||
if (r < 0) {
|
||||
log_warning_errno(r, "Failed to determine timezone from system time, ignoring: %m");
|
||||
n = 0;
|
||||
} else
|
||||
n = strftime(a, sizeof a, "%Z, %z", &tm);
|
||||
} else
|
||||
n = 0;
|
||||
r = table_add_cell_stringf(table, NULL, "%s (%s)", strna(i->timezone), n > 0 ? a : "n/a");
|
||||
if (r < 0)
|
||||
return table_log_add_error(r);
|
||||
|
@ -601,8 +601,11 @@ static int property_get_rtc_time(
|
||||
log_debug("/dev/rtc has no valid time, power loss probably occurred?");
|
||||
else if (r < 0)
|
||||
return sd_bus_error_set_errnof(error, r, "Failed to read RTC: %m");
|
||||
else
|
||||
t = (usec_t) timegm(&tm) * USEC_PER_SEC;
|
||||
else {
|
||||
r = mktime_or_timegm_usec(&tm, /* utc= */ true, &t);
|
||||
if (r < 0)
|
||||
log_warning_errno(r, "Failed to convert RTC time to UNIX time, ignoring: %m");
|
||||
}
|
||||
|
||||
return sd_bus_message_append(reply, "t", t);
|
||||
}
|
||||
@ -713,16 +716,17 @@ static int method_set_timezone(sd_bus_message *m, void *userdata, sd_bus_error *
|
||||
log_debug_errno(r, "Failed to tell kernel about timezone, ignoring: %m");
|
||||
|
||||
if (c->local_rtc) {
|
||||
struct timespec ts;
|
||||
struct tm tm;
|
||||
|
||||
/* 4. Sync RTC from system clock, with the new delta */
|
||||
assert_se(clock_gettime(CLOCK_REALTIME, &ts) == 0);
|
||||
assert_se(localtime_r(&ts.tv_sec, &tm));
|
||||
|
||||
r = hwclock_set(&tm);
|
||||
r = localtime_or_gmtime_usec(now(CLOCK_REALTIME), /* utc= */ false, &tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to sync time to hardware clock, ignoring: %m");
|
||||
log_debug_errno(r, "Failed to convert system time to calendar time, ignoring: %m");
|
||||
else {
|
||||
r = hwclock_set(&tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to sync time to hardware clock, ignoring: %m");
|
||||
}
|
||||
}
|
||||
|
||||
log_struct(LOG_INFO,
|
||||
@ -787,32 +791,46 @@ static int method_set_local_rtc(sd_bus_message *m, void *userdata, sd_bus_error
|
||||
assert_se(clock_gettime(CLOCK_REALTIME, &ts) == 0);
|
||||
|
||||
if (fix_system) {
|
||||
struct tm tm;
|
||||
struct tm tm = {
|
||||
.tm_isdst = -1,
|
||||
};
|
||||
|
||||
/* Sync system clock from RTC; first, initialize the timezone fields of struct tm. */
|
||||
localtime_or_gmtime_r(&ts.tv_sec, &tm, !c->local_rtc);
|
||||
r = localtime_or_gmtime_usec(timespec_load(&ts), !c->local_rtc, &tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to determine current timezone, ignoring: %m");
|
||||
|
||||
/* Override the main fields of struct tm, but not the timezone fields */
|
||||
r = hwclock_get(&tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to get hardware clock, ignoring: %m");
|
||||
else {
|
||||
usec_t t;
|
||||
/* And set the system clock with this */
|
||||
ts.tv_sec = mktime_or_timegm(&tm, !c->local_rtc);
|
||||
|
||||
if (clock_settime(CLOCK_REALTIME, &ts) < 0)
|
||||
log_debug_errno(errno, "Failed to update system clock, ignoring: %m");
|
||||
r = mktime_or_timegm_usec(&tm, !c->local_rtc, &t);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to convert calendar time to system time, ignoring: %m");
|
||||
else {
|
||||
/* We leave the subsecond offset as is! */
|
||||
ts.tv_sec = t / USEC_PER_SEC;
|
||||
|
||||
if (clock_settime(CLOCK_REALTIME, &ts) < 0)
|
||||
log_debug_errno(errno, "Failed to update system clock, ignoring: %m");
|
||||
}
|
||||
}
|
||||
|
||||
} else {
|
||||
struct tm tm;
|
||||
|
||||
/* Sync RTC from system clock */
|
||||
localtime_or_gmtime_r(&ts.tv_sec, &tm, !c->local_rtc);
|
||||
|
||||
r = hwclock_set(&tm);
|
||||
r = localtime_or_gmtime_usec(timespec_load(&ts), !c->local_rtc, &tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to sync time to hardware clock, ignoring: %m");
|
||||
log_debug_errno(r, "Failed to convert time to calendar time, ignoring: %m");
|
||||
else {
|
||||
r = hwclock_set(&tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to sync time to hardware clock, ignoring: %m");
|
||||
}
|
||||
}
|
||||
|
||||
log_info("RTC configured to %s time.", c->local_rtc ? "local" : "UTC");
|
||||
@ -826,7 +844,6 @@ static int method_set_local_rtc(sd_bus_message *m, void *userdata, sd_bus_error
|
||||
|
||||
static int method_set_time(sd_bus_message *m, void *userdata, sd_bus_error *error) {
|
||||
sd_bus *bus = sd_bus_message_get_bus(m);
|
||||
char buf[FORMAT_TIMESTAMP_MAX];
|
||||
int relative, interactive, r;
|
||||
Context *c = ASSERT_PTR(userdata);
|
||||
int64_t utc;
|
||||
@ -901,16 +918,19 @@ static int method_set_time(sd_bus_message *m, void *userdata, sd_bus_error *erro
|
||||
}
|
||||
|
||||
/* Sync down to RTC */
|
||||
localtime_or_gmtime_r(&ts.tv_sec, &tm, !c->local_rtc);
|
||||
|
||||
r = hwclock_set(&tm);
|
||||
r = localtime_or_gmtime_usec(timespec_load(&ts), !c->local_rtc, &tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to update hardware clock, ignoring: %m");
|
||||
log_debug_errno(r, "Failed to convert timestamp to calendar time, ignoring: %m");
|
||||
else {
|
||||
r = hwclock_set(&tm);
|
||||
if (r < 0)
|
||||
log_debug_errno(r, "Failed to update hardware clock, ignoring: %m");
|
||||
}
|
||||
|
||||
log_struct(LOG_INFO,
|
||||
"MESSAGE_ID=" SD_MESSAGE_TIME_CHANGE_STR,
|
||||
"REALTIME="USEC_FMT, timespec_load(&ts),
|
||||
LOG_MESSAGE("Changed local time to %s", strnull(format_timestamp(buf, sizeof(buf), timespec_load(&ts)))));
|
||||
LOG_MESSAGE("Changed local time to %s", strnull(FORMAT_TIMESTAMP(timespec_load(&ts)))));
|
||||
|
||||
return sd_bus_reply_method_return(m, NULL);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user