summaryrefslogtreecommitdiff
path: root/src/shared/logs-show.c
diff options
context:
space:
mode:
authorDaniel Braunwarth <daniel@braunwarth.dev>2022-09-22 18:35:19 +0200
committerDaniel Braunwarth <daniel@braunwarth.dev>2022-09-23 10:07:03 +0200
commit893bcd3d074022bd52618cb682152cb822878636 (patch)
treefd636b0ef90ba4d39163f92f1a3032a11237dcc8 /src/shared/logs-show.c
parent275e6be052e690adcad5d2a557acb9dcb5bedbc6 (diff)
downloadsystemd-893bcd3d074022bd52618cb682152cb822878636.tar.gz
shared/logs-show: add new --output= format "short-delta"
This new output formatting option is similar to "short-monotonic" but also shows the time delta between two messages. This fixes #24641.
Diffstat (limited to 'src/shared/logs-show.c')
-rw-r--r--src/shared/logs-show.c94
1 files changed, 80 insertions, 14 deletions
diff --git a/src/shared/logs-show.c b/src/shared/logs-show.c
index 23de0f456e..0ebc66597d 100644
--- a/src/shared/logs-show.c
+++ b/src/shared/logs-show.c
@@ -317,15 +317,47 @@ static bool print_multiline(
return ellipsized;
}
-static int output_timestamp_monotonic(FILE *f, const dual_timestamp *ts) {
+static int output_timestamp_monotonic(
+ FILE *f, OutputMode mode,
+ const dual_timestamp *ts,
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
+
+ int written_chars = 0;
+
assert(f);
assert(ts);
+ assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
if (!VALID_MONOTONIC(ts->monotonic))
return log_error_errno(SYNTHETIC_ERRNO(EINVAL), "No valid monotonic timestamp available");
- fprintf(f, "[%5"PRI_USEC".%06"PRI_USEC"]", ts->monotonic / USEC_PER_SEC, ts->monotonic % USEC_PER_SEC);
- return 1 + 5 + 1 + 6 + 1;
+ written_chars += fprintf(f, "[%5"PRI_USEC".%06"PRI_USEC, ts->monotonic / USEC_PER_SEC, ts->monotonic % USEC_PER_SEC);
+
+ if (mode == OUTPUT_SHORT_DELTA) {
+ uint64_t delta;
+ bool reliable_ts = true;
+
+ if (VALID_MONOTONIC(previous_ts->monotonic) && sd_id128_equal(*boot_id, *previous_boot_id))
+ delta = usec_sub_unsigned(ts->monotonic, previous_ts->monotonic);
+ else if (VALID_REALTIME(ts->realtime) && VALID_REALTIME(previous_ts->realtime)) {
+ delta = usec_sub_unsigned(ts->realtime, previous_ts->realtime);
+ reliable_ts = false;
+ } else {
+ written_chars += fprintf(f, "%16s", "");
+ goto finish;
+ }
+
+ written_chars += fprintf(f, " <%5"PRI_USEC".%06"PRI_USEC"%s>", delta / USEC_PER_SEC, delta % USEC_PER_SEC, reliable_ts ? " " : "*");
+ }
+
+finish:
+ written_chars += fprintf(f, "%s", "]");
+
+ return written_chars;
}
static int output_timestamp_realtime(
@@ -426,7 +458,9 @@ static int output_short(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
int r;
const void *data;
@@ -459,6 +493,8 @@ static int output_short(
assert(j);
assert(ts);
assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
/* Set the threshold to one bigger than the actual print
* threshold, so that if the line is actually longer than what
@@ -493,8 +529,8 @@ static int output_short(
audit = streq_ptr(transport, "audit");
- if (mode == OUTPUT_SHORT_MONOTONIC)
- r = output_timestamp_monotonic(f, ts);
+ if (IN_SET(mode, OUTPUT_SHORT_MONOTONIC, OUTPUT_SHORT_DELTA))
+ r = output_timestamp_monotonic(f, mode, ts, boot_id, previous_ts, previous_boot_id);
else
r = output_timestamp_realtime(f, j, mode, flags, ts);
if (r < 0)
@@ -629,7 +665,9 @@ static int output_verbose(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
const void *data;
size_t length;
@@ -642,6 +680,8 @@ static int output_verbose(
assert(j);
assert(ts);
assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
sd_journal_set_data_threshold(j, 0);
@@ -727,7 +767,9 @@ static int output_export(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
_cleanup_free_ char *cursor = NULL;
const void *data;
@@ -737,6 +779,8 @@ static int output_export(
assert(j);
assert(ts);
assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
sd_journal_set_data_threshold(j, 0);
@@ -962,7 +1006,9 @@ static int output_json(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
char sid[SD_ID128_STRING_MAX], usecbuf[DECIMAL_STR_MAX(usec_t)];
_cleanup_(json_variant_unrefp) JsonVariant *object = NULL;
@@ -976,6 +1022,8 @@ static int output_json(
assert(j);
assert(ts);
assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
(void) sd_journal_set_data_threshold(j, flags & OUTPUT_SHOW_ALL ? 0 : JSON_THRESHOLD);
@@ -1158,7 +1206,9 @@ static int output_cat(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) {
int r, prio = LOG_INFO;
const char *field;
@@ -1167,6 +1217,8 @@ static int output_cat(
assert(f);
assert(ts);
assert(boot_id);
+ assert(previous_ts);
+ assert(previous_boot_id);
(void) sd_journal_set_data_threshold(j, 0);
@@ -1259,13 +1311,16 @@ static int (*output_funcs[_OUTPUT_MODE_MAX])(
const Set *output_fields,
const size_t highlight[2],
const dual_timestamp *ts,
- const sd_id128_t *boot_id) = {
+ const sd_id128_t *boot_id,
+ const dual_timestamp *previous_ts,
+ const sd_id128_t *previous_boot_id) = {
[OUTPUT_SHORT] = output_short,
[OUTPUT_SHORT_ISO] = output_short,
[OUTPUT_SHORT_ISO_PRECISE] = output_short,
[OUTPUT_SHORT_PRECISE] = output_short,
[OUTPUT_SHORT_MONOTONIC] = output_short,
+ [OUTPUT_SHORT_DELTA] = output_short,
[OUTPUT_SHORT_UNIX] = output_short,
[OUTPUT_SHORT_FULL] = output_short,
[OUTPUT_VERBOSE] = output_verbose,
@@ -1286,7 +1341,9 @@ int show_journal_entry(
OutputFlags flags,
char **output_fields,
const size_t highlight[2],
- bool *ellipsized) {
+ bool *ellipsized,
+ dual_timestamp *previous_ts,
+ sd_id128_t *previous_boot_id) {
_cleanup_set_free_ Set *fields = NULL;
dual_timestamp ts = DUAL_TIMESTAMP_NULL;
@@ -1295,6 +1352,8 @@ int show_journal_entry(
assert(mode >= 0);
assert(mode < _OUTPUT_MODE_MAX);
+ assert(previous_ts);
+ assert(previous_boot_id);
if (n_columns <= 0)
n_columns = columns();
@@ -1311,7 +1370,11 @@ int show_journal_entry(
if (r < 0)
return log_error_errno(r, "Failed to get journal fields: %m");
- r = output_funcs[mode](f, j, mode, n_columns, flags, fields, highlight, &ts, &boot_id);
+ r = output_funcs[mode](f, j, mode, n_columns, flags, fields, highlight, &ts, &boot_id, previous_ts, previous_boot_id);
+
+ /* Store timestamp and boot ID for next iteration */
+ *previous_ts = ts;
+ *previous_boot_id = boot_id;
if (ellipsized && r > 0)
*ellipsized = true;
@@ -1348,6 +1411,8 @@ int show_journal(
unsigned line = 0;
bool need_seek = false;
int warn_cutoff = flags & OUTPUT_WARN_CUTOFF;
+ dual_timestamp previous_ts = DUAL_TIMESTAMP_NULL;
+ sd_id128_t previous_boot_id = SD_ID128_NULL;
assert(j);
assert(mode >= 0);
@@ -1396,7 +1461,8 @@ int show_journal(
line++;
maybe_print_begin_newline(f, &flags);
- r = show_journal_entry(f, j, mode, n_columns, flags, NULL, NULL, ellipsized);
+ r = show_journal_entry(f, j, mode, n_columns, flags, NULL, NULL, ellipsized,
+ &previous_ts, &previous_boot_id);
if (r < 0)
return r;
}