diff options
| -rw-r--r-- | src/journal/journal-internal.h | 2 | ||||
| -rw-r--r-- | src/journal/journalctl.c | 27 | ||||
| -rw-r--r-- | src/journal/sd-journal.c | 49 | 
3 files changed, 63 insertions, 15 deletions
| diff --git a/src/journal/journal-internal.h b/src/journal/journal-internal.h index 1221799c1d..06847402e0 100644 --- a/src/journal/journal-internal.h +++ b/src/journal/journal-internal.h @@ -121,7 +121,7 @@ struct sd_journal {          Hashmap *directories_by_path;          Hashmap *directories_by_wd; -        Set *errors; +        Hashmap *errors;  };  char *journal_make_match_string(sd_journal *j); diff --git a/src/journal/journalctl.c b/src/journal/journalctl.c index 30ae6e39c3..98a852cb50 100644 --- a/src/journal/journalctl.c +++ b/src/journal/journalctl.c @@ -1708,33 +1708,50 @@ static int access_check_var_log_journal(sd_journal *j) {  static int access_check(sd_journal *j) {          Iterator it;          void *code; +        char *path;          int r = 0;          assert(j); -        if (set_isempty(j->errors)) { +        if (hashmap_isempty(j->errors)) {                  if (ordered_hashmap_isempty(j->files))                          log_notice("No journal files were found.");                  return 0;          } -        if (set_contains(j->errors, INT_TO_PTR(-EACCES))) { +        if (hashmap_contains(j->errors, INT_TO_PTR(-EACCES))) {                  (void) access_check_var_log_journal(j);                  if (ordered_hashmap_isempty(j->files))                          r = log_error_errno(EACCES, "No journal files were opened due to insufficient permissions.");          } -        SET_FOREACH(code, j->errors, it) { +        HASHMAP_FOREACH_KEY(path, code, j->errors, it) {                  int err;                  err = abs(PTR_TO_INT(code)); -                if (err == EACCES) +                switch (err) { +                case EACCES:                          continue; -                log_warning_errno(err, "An error was encountered while opening journal files, ignoring: %m"); +                case ENODATA: +                        log_warning_errno(err, "Journal file %s is truncated, ignoring file.", path); +                        break; + +                case EPROTONOSUPPORT: +                        log_warning_errno(err, "Journal file %s uses an unsupported feature, ignoring file.", path); +                        break; + +                case EBADMSG: +                        log_warning_errno(err, "Journal file %s corrupted, ignoring file.", path); +                        break; + +                default: +                        log_warning_errno(err, "An error was encountered while opening journal file %s, ignoring file.", path); +                        break; +                }          }          return r; diff --git a/src/journal/sd-journal.c b/src/journal/sd-journal.c index b5e1d05fa6..5cde7f17f7 100644 --- a/src/journal/sd-journal.c +++ b/src/journal/sd-journal.c @@ -73,19 +73,46 @@ static bool journal_pid_changed(sd_journal *j) {          return j->original_pid != getpid();  } -/* We return an error here only if we didn't manage to -   memorize the real error. */ -static int set_put_error(sd_journal *j, int r) { +static int journal_put_error(sd_journal *j, int r, const char *path) { +        char *copy;          int k; +        /* Memorize an error we encountered, and store which +         * file/directory it was generated from. Note that we store +         * only *one* path per error code, as the error code is the +         * key into the hashmap, and the path is the value. This means +         * we keep track only of all error kinds, but not of all error +         * locations. This has the benefit that the hashmap cannot +         * grow beyond bounds. +         * +         * We return an error here only if we didn't manage to +         * memorize the real error. */ +          if (r >= 0)                  return r; -        k = set_ensure_allocated(&j->errors, NULL); +        k = hashmap_ensure_allocated(&j->errors, NULL);          if (k < 0)                  return k; -        return set_put(j->errors, INT_TO_PTR(r)); +        if (path) { +                copy = strdup(path); +                if (!copy) +                        return -ENOMEM; +        } else +                copy = NULL; + +        k = hashmap_put(j->errors, INT_TO_PTR(r), copy); +        if (k < 0) { +                free(copy); + +                if (k == -EEXIST) +                        return 0; + +                return k; +        } + +        return 0;  }  static void detach_location(sd_journal *j) { @@ -1246,7 +1273,7 @@ static int add_any_file(sd_journal *j, const char *path) {          return 0;  fail: -        k = set_put_error(j, r); +        k = journal_put_error(j, r, path);          if (k < 0)                  return k; @@ -1402,7 +1429,7 @@ static int add_directory(sd_journal *j, const char *prefix, const char *dirname)          return 0;  fail: -        k = set_put_error(j, r); +        k = journal_put_error(j, r, path ?: dirname);          if (k < 0)                  return k; @@ -1493,7 +1520,7 @@ static int add_root_directory(sd_journal *j, const char *p, bool missing_ok) {          return 0;  fail: -        k = set_put_error(j, r); +        k = journal_put_error(j, r, p);          if (k < 0)                  return k; @@ -1738,6 +1765,7 @@ fail:  _public_ void sd_journal_close(sd_journal *j) {          Directory *d;          JournalFile *f; +        char *p;          if (!j)                  return; @@ -1765,10 +1793,13 @@ _public_ void sd_journal_close(sd_journal *j) {                  mmap_cache_unref(j->mmap);          } +        while ((p = hashmap_steal_first(j->errors))) +                free(p); +        hashmap_free(j->errors); +          free(j->path);          free(j->prefix);          free(j->unique_field); -        set_free(j->errors);          free(j);  } | 
