diff options
| -rw-r--r-- | man/systemctl.xml | 2 | ||||
| -rw-r--r-- | src/analyze/analyze-verify.c | 60 | ||||
| -rw-r--r-- | src/core/dbus.c | 66 | ||||
| -rw-r--r-- | src/core/dbus.h | 2 | ||||
| -rw-r--r-- | src/core/manager.c | 4 | ||||
| -rw-r--r-- | src/core/service.c | 14 | ||||
| -rw-r--r-- | src/core/service.h | 1 | ||||
| -rw-r--r-- | src/core/socket.c | 12 | ||||
| -rw-r--r-- | src/journal/compress.c | 31 | ||||
| -rw-r--r-- | src/journal/compress.h | 13 | ||||
| -rw-r--r-- | src/journal/journal-file.c | 2 | ||||
| -rw-r--r-- | src/journal/sd-journal.c | 8 | ||||
| -rw-r--r-- | src/journal/test-compress-benchmark.c | 7 | ||||
| -rw-r--r-- | src/journal/test-compress.c | 121 | ||||
| -rw-r--r-- | src/libsystemd-network/sd-lldp.c | 88 | ||||
| -rw-r--r-- | src/libsystemd/sd-event/sd-event.c | 39 | ||||
| -rw-r--r-- | src/resolve/resolved-dns-transaction.c | 6 | ||||
| -rw-r--r-- | src/shared/logs-show.c | 3 | 
18 files changed, 351 insertions, 128 deletions
| diff --git a/man/systemctl.xml b/man/systemctl.xml index 1fb056874c..cf914f4cb2 100644 --- a/man/systemctl.xml +++ b/man/systemctl.xml @@ -832,7 +832,7 @@ kobject-uevent 1 systemd-udevd-kernel.socket systemd-udevd.service              output. If you are looking for computer-parsable output,              use <command>show</command> instead. By default, this              function only shows 10 lines of output and ellipsizes -            lines to fit in the terminal window. This can be changes +            lines to fit in the terminal window. This can be changed              with <option>--lines</option> and <option>--full</option>,              see above. In addition, <command>journalctl              --unit=<replaceable>NAME</replaceable></command> or diff --git a/src/analyze/analyze-verify.c b/src/analyze/analyze-verify.c index 6e460697db..3c9766da04 100644 --- a/src/analyze/analyze-verify.c +++ b/src/analyze/analyze-verify.c @@ -30,6 +30,47 @@  #include "pager.h"  #include "path-util.h"  #include "strv.h" +#include "unit-name.h" + +static int prepare_filename(const char *filename, char **ret) { +        int r; +        const char *name; +        _cleanup_free_ char *abspath = NULL; +        _cleanup_free_ char *dir = NULL; +        _cleanup_free_ char *with_instance = NULL; +        char *c; + +        assert(filename); +        assert(ret); + +        r = path_make_absolute_cwd(filename, &abspath); +        if (r < 0) +                return r; + +        name = basename(abspath); +        if (!unit_name_is_valid(name, UNIT_NAME_ANY)) +                return -EINVAL; + +        if (unit_name_is_valid(name, UNIT_NAME_TEMPLATE)) { +                r = unit_name_replace_instance(name, "i", &with_instance); +                if (r < 0) +                        return r; +        } + +        dir = dirname_malloc(abspath); +        if (!dir) +                return -ENOMEM; + +        if (with_instance) +                c = path_join(NULL, dir, with_instance); +        else +                c = path_join(NULL, dir, name); +        if (!c) +                return -ENOMEM; + +        *ret = c; +        return 0; +}  static int generate_path(char **var, char **filenames) {          char **filename; @@ -233,18 +274,19 @@ int verify_units(char **filenames, ManagerRunningAs running_as, bool check_man)          log_debug("Loading remaining units from the command line...");          STRV_FOREACH(filename, filenames) { -                char fname[UNIT_NAME_MAX + 2 + 1] = "./"; +                _cleanup_free_ char *prepared = NULL;                  log_debug("Handling %s...", *filename); -                /* manager_load_unit does not like pure basenames, so prepend -                 * the local directory, but only for valid names. manager_load_unit -                 * will print the error for other ones. */ -                if (!strchr(*filename, '/') && strlen(*filename) <= UNIT_NAME_MAX) { -                        strncat(fname + 2, *filename, UNIT_NAME_MAX); -                        k = manager_load_unit(m, NULL, fname, &err, &units[count]); -                } else -                        k = manager_load_unit(m, NULL, *filename, &err, &units[count]); +                k = prepare_filename(*filename, &prepared); +                if (k < 0) { +                        log_error_errno(k, "Failed to prepare filename %s: %m", *filename); +                        if (r == 0) +                                r = k; +                        continue; +                } + +                k = manager_load_unit(m, NULL, prepared, &err, &units[count]);                  if (k < 0) {                          log_error_errno(k, "Failed to load %s: %m", *filename);                          if (r == 0) diff --git a/src/core/dbus.c b/src/core/dbus.c index e7ee216f0e..1d89b9e250 100644 --- a/src/core/dbus.c +++ b/src/core/dbus.c @@ -734,9 +734,11 @@ static int bus_on_connection(sd_event_source *s, int fd, uint32_t revents, void          return 0;  } -static int bus_list_names(Manager *m, sd_bus *bus) { +int manager_sync_bus_names(Manager *m, sd_bus *bus) {          _cleanup_strv_free_ char **names = NULL; -        char **i; +        const char *name; +        Iterator i; +        Unit *u;          int r;          assert(m); @@ -746,15 +748,55 @@ static int bus_list_names(Manager *m, sd_bus *bus) {          if (r < 0)                  return log_error_errno(r, "Failed to get initial list of names: %m"); -        /* This is a bit hacky, we say the owner of the name is the -         * name itself, because we don't want the extra traffic to -         * figure out the real owner. */ -        STRV_FOREACH(i, names) { -                Unit *u; +        /* We have to synchronize the current bus names with the +         * list of active services. To do this, walk the list of +         * all units with bus names. */ +        HASHMAP_FOREACH_KEY(u, name, m->watch_bus, i) { +                Service *s = SERVICE(u); + +                assert(s); -                u = hashmap_get(m->watch_bus, *i); -                if (u) -                        UNIT_VTABLE(u)->bus_name_owner_change(u, *i, NULL, *i); +                if (!streq_ptr(s->bus_name, name)) { +                        log_unit_warning(u, "Bus name has changed from %s → %s, ignoring.", s->bus_name, name); +                        continue; +                } + +                /* Check if a service's bus name is in the list of currently +                 * active names */ +                if (strv_contains(names, name)) { +                        _cleanup_(sd_bus_creds_unrefp) sd_bus_creds *creds = NULL; +                        const char *unique; + +                        /* If it is, determine its current owner */ +                        r = sd_bus_get_name_creds(bus, name, SD_BUS_CREDS_UNIQUE_NAME, &creds); +                        if (r < 0) { +                                log_error_errno(r, "Failed to get bus name owner %s: %m", name); +                                continue; +                        } + +                        r = sd_bus_creds_get_unique_name(creds, &unique); +                        if (r < 0) { +                                log_error_errno(r, "Failed to get unique name for %s: %m", name); +                                continue; +                        } + +                        /* Now, let's compare that to the previous bus owner, and +                         * if it's still the same, all is fine, so just don't +                         * bother the service. Otherwise, the name has apparently +                         * changed, so synthesize a name owner changed signal. */ + +                        if (!streq_ptr(unique, s->bus_name_owner)) +                                UNIT_VTABLE(u)->bus_name_owner_change(u, name, s->bus_name_owner, unique); +                } else { +                        /* So, the name we're watching is not on the bus. +                         * This either means it simply hasn't appeared yet, +                         * or it was lost during the daemon reload. +                         * Check if the service has a stored name owner, +                         * and synthesize a name loss signal in this case. */ + +                        if (s->bus_name_owner) +                                UNIT_VTABLE(u)->bus_name_owner_change(u, name, s->bus_name_owner, NULL); +                }          }          return 0; @@ -808,7 +850,9 @@ static int bus_setup_api(Manager *m, sd_bus *bus) {          if (r < 0)                  return log_error_errno(r, "Failed to register name: %m"); -        bus_list_names(m, bus); +        r = manager_sync_bus_names(m, bus); +        if (r < 0) +                return r;          log_debug("Successfully connected to API bus.");          return 0; diff --git a/src/core/dbus.h b/src/core/dbus.h index 4f06ad11c4..ff761668f3 100644 --- a/src/core/dbus.h +++ b/src/core/dbus.h @@ -34,6 +34,8 @@ void bus_track_serialize(sd_bus_track *t, FILE *f);  int bus_track_deserialize_item(char ***l, const char *line);  int bus_track_coldplug(Manager *m, sd_bus_track **t, char ***l); +int manager_sync_bus_names(Manager *m, sd_bus *bus); +  int bus_foreach_bus(Manager *m, sd_bus_track *subscribed2, int (*send_message)(sd_bus *bus, void *userdata), void *userdata);  int bus_verify_manage_units_async(Manager *m, sd_bus_message *call, sd_bus_error *error); diff --git a/src/core/manager.c b/src/core/manager.c index e65616adc1..ffe27be743 100644 --- a/src/core/manager.c +++ b/src/core/manager.c @@ -2574,6 +2574,10 @@ int manager_reload(Manager *m) {          /* Third, fire things up! */          manager_coldplug(m); +        /* Sync current state of bus names with our set of listening units */ +        if (m->api_bus) +                manager_sync_bus_names(m, m->api_bus); +          assert(m->n_reloading > 0);          m->n_reloading--; diff --git a/src/core/service.c b/src/core/service.c index 41a729c421..c5b689a35c 100644 --- a/src/core/service.c +++ b/src/core/service.c @@ -323,6 +323,8 @@ static void service_done(Unit *u) {                  s->bus_name = mfree(s->bus_name);          } +        s->bus_name_owner = mfree(s->bus_name_owner); +          s->bus_endpoint_fd = safe_close(s->bus_endpoint_fd);          service_close_socket_fd(s);          service_connection_unref(s); @@ -2122,6 +2124,7 @@ static int service_serialize(Unit *u, FILE *f, FDSet *fds) {          unit_serialize_item(u, f, "main-pid-known", yes_no(s->main_pid_known));          unit_serialize_item(u, f, "bus-name-good", yes_no(s->bus_name_good)); +        unit_serialize_item(u, f, "bus-name-owner", s->bus_name_owner);          r = unit_serialize_item_escaped(u, f, "status-text", s->status_text);          if (r < 0) @@ -2249,6 +2252,10 @@ static int service_deserialize_item(Unit *u, const char *key, const char *value,                          log_unit_debug(u, "Failed to parse bus-name-good value: %s", value);                  else                          s->bus_name_good = b; +        } else if (streq(key, "bus-name-owner")) { +                r = free_and_strdup(&s->bus_name_owner, value); +                if (r < 0) +                        log_unit_error_errno(u, r, "Unable to deserialize current bus owner %s: %m", value);          } else if (streq(key, "status-text")) {                  char *t; @@ -3134,6 +3141,13 @@ static void service_bus_name_owner_change(          s->bus_name_good = !!new_owner; +        /* Track the current owner, so we can reconstruct changes after a daemon reload */ +        r = free_and_strdup(&s->bus_name_owner, new_owner); +        if (r < 0) { +                log_unit_error_errno(u, r, "Unable to set new bus name owner %s: %m", new_owner); +                return; +        } +          if (s->type == SERVICE_DBUS) {                  /* service_enter_running() will figure out what to diff --git a/src/core/service.h b/src/core/service.h index d0faad88e0..19efbccfc7 100644 --- a/src/core/service.h +++ b/src/core/service.h @@ -172,6 +172,7 @@ struct Service {          bool reset_cpu_usage:1;          char *bus_name; +        char *bus_name_owner; /* unique name of the current owner */          char *status_text;          int status_errno; diff --git a/src/core/socket.c b/src/core/socket.c index 7beec3644e..d6b0c963e8 100644 --- a/src/core/socket.c +++ b/src/core/socket.c @@ -156,14 +156,16 @@ static void socket_done(Unit *u) {          s->tcp_congestion = mfree(s->tcp_congestion);          s->bind_to_device = mfree(s->bind_to_device); -        free(s->smack); -        free(s->smack_ip_in); -        free(s->smack_ip_out); +        s->smack = mfree(s->smack); +        s->smack_ip_in = mfree(s->smack_ip_in); +        s->smack_ip_out = mfree(s->smack_ip_out);          strv_free(s->symlinks); -        free(s->user); -        free(s->group); +        s->user = mfree(s->user); +        s->group = mfree(s->group); + +        s->fdname = mfree(s->fdname);          s->timer_event_source = sd_event_source_unref(s->timer_event_source);  } diff --git a/src/journal/compress.c b/src/journal/compress.c index 1a3d2cdd80..1828165894 100644 --- a/src/journal/compress.c +++ b/src/journal/compress.c @@ -58,7 +58,8 @@ static const char* const object_compressed_table[_OBJECT_COMPRESSED_MAX] = {  DEFINE_STRING_TABLE_LOOKUP(object_compressed, int); -int compress_blob_xz(const void *src, uint64_t src_size, void *dst, size_t *dst_size) { +int compress_blob_xz(const void *src, uint64_t src_size, +                     void *dst, size_t dst_alloc_size, size_t *dst_size) {  #ifdef HAVE_XZ          static const lzma_options_lzma opt = {                  1u << 20u, NULL, 0, LZMA_LC_DEFAULT, LZMA_LP_DEFAULT, @@ -74,6 +75,7 @@ int compress_blob_xz(const void *src, uint64_t src_size, void *dst, size_t *dst_          assert(src);          assert(src_size > 0);          assert(dst); +        assert(dst_alloc_size > 0);          assert(dst_size);          /* Returns < 0 if we couldn't compress the data or the @@ -83,7 +85,7 @@ int compress_blob_xz(const void *src, uint64_t src_size, void *dst, size_t *dst_                  return -ENOBUFS;          ret = lzma_stream_buffer_encode((lzma_filter*) filters, LZMA_CHECK_NONE, NULL, -                                        src, src_size, dst, &out_pos, src_size - 1); +                                        src, src_size, dst, &out_pos, dst_alloc_size);          if (ret != LZMA_OK)                  return -ENOBUFS; @@ -94,13 +96,15 @@ int compress_blob_xz(const void *src, uint64_t src_size, void *dst, size_t *dst_  #endif  } -int compress_blob_lz4(const void *src, uint64_t src_size, void *dst, size_t *dst_size) { +int compress_blob_lz4(const void *src, uint64_t src_size, +                      void *dst, size_t dst_alloc_size, size_t *dst_size) {  #ifdef HAVE_LZ4          int r;          assert(src);          assert(src_size > 0);          assert(dst); +        assert(dst_alloc_size > 0);          assert(dst_size);          /* Returns < 0 if we couldn't compress the data or the @@ -109,7 +113,7 @@ int compress_blob_lz4(const void *src, uint64_t src_size, void *dst, size_t *dst          if (src_size < 9)                  return -ENOBUFS; -        r = LZ4_compress_limitedOutput(src, dst + 8, src_size, src_size - 8 - 1); +        r = LZ4_compress_limitedOutput(src, dst + 8, src_size, (int) dst_alloc_size - 8);          if (r <= 0)                  return -ENOBUFS; @@ -306,6 +310,7 @@ int decompress_startswith_lz4(const void *src, uint64_t src_size,           * prefix */          int r; +        size_t size;          assert(src);          assert(src_size > 0); @@ -322,10 +327,18 @@ int decompress_startswith_lz4(const void *src, uint64_t src_size,          r = LZ4_decompress_safe_partial(src + 8, *buffer, src_size - 8,                                          prefix_len + 1, *buffer_size); +        if (r >= 0) +                size = (unsigned) r; +        else { +                /* lz4 always tries to decode full "sequence", so in +                 * pathological cases might need to decompress the +                 * full field. */ +                r = decompress_blob_lz4(src, src_size, buffer, buffer_size, &size, 0); +                if (r < 0) +                        return r; +        } -        if (r < 0) -                return -EBADMSG; -        if ((unsigned) r >= prefix_len + 1) +        if (size >= prefix_len + 1)                  return memcmp(*buffer, prefix, prefix_len) == 0 &&                          ((const uint8_t*) *buffer)[prefix_len] == extra;          else @@ -438,7 +451,7 @@ int compress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {          _cleanup_(LZ4F_freeCompressionContextp) LZ4F_compressionContext_t ctx = NULL;          _cleanup_free_ char *buf = NULL;          char *src = NULL; -        size_t size, n, total_in = 0, total_out = 0, offset = 0, frame_size; +        size_t size, n, total_in = 0, total_out, offset = 0, frame_size;          struct stat st;          int r;          static const LZ4F_compressOptions_t options = { @@ -461,7 +474,7 @@ int compress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {          if (!buf)                  return -ENOMEM; -        n = offset = LZ4F_compressBegin(ctx, buf, size, &preferences); +        n = offset = total_out = LZ4F_compressBegin(ctx, buf, size, &preferences);          if (LZ4F_isError(n))                  return -EINVAL; diff --git a/src/journal/compress.h b/src/journal/compress.h index 9a065eb763..758598730a 100644 --- a/src/journal/compress.h +++ b/src/journal/compress.h @@ -28,17 +28,20 @@  const char* object_compressed_to_string(int compression);  int object_compressed_from_string(const char *compression); -int compress_blob_xz(const void *src, uint64_t src_size, void *dst, size_t *dst_size); -int compress_blob_lz4(const void *src, uint64_t src_size, void *dst, size_t *dst_size); +int compress_blob_xz(const void *src, uint64_t src_size, +                     void *dst, size_t dst_alloc_size, size_t *dst_size); +int compress_blob_lz4(const void *src, uint64_t src_size, +                      void *dst, size_t dst_alloc_size, size_t *dst_size); -static inline int compress_blob(const void *src, uint64_t src_size, void *dst, size_t *dst_size) { +static inline int compress_blob(const void *src, uint64_t src_size, +                                void *dst, size_t dst_alloc_size, size_t *dst_size) {          int r;  #ifdef HAVE_LZ4 -        r = compress_blob_lz4(src, src_size, dst, dst_size); +        r = compress_blob_lz4(src, src_size, dst, dst_alloc_size, dst_size);          if (r == 0)                  return OBJECT_COMPRESSED_LZ4;  #else -        r = compress_blob_xz(src, src_size, dst, dst_size); +        r = compress_blob_xz(src, src_size, dst, dst_alloc_size, dst_size);          if (r == 0)                  return OBJECT_COMPRESSED_XZ;  #endif diff --git a/src/journal/journal-file.c b/src/journal/journal-file.c index 6f09301521..9e362bacae 100644 --- a/src/journal/journal-file.c +++ b/src/journal/journal-file.c @@ -1084,7 +1084,7 @@ static int journal_file_append_data(          if (JOURNAL_FILE_COMPRESS(f) && size >= COMPRESSION_SIZE_THRESHOLD) {                  size_t rsize = 0; -                compression = compress_blob(data, size, o->data.payload, &rsize); +                compression = compress_blob(data, size, o->data.payload, size - 1, &rsize);                  if (compression >= 0) {                          o->object.size = htole64(offsetof(Object, data.payload) + rsize); diff --git a/src/journal/sd-journal.c b/src/journal/sd-journal.c index 5cde7f17f7..cd5160154a 100644 --- a/src/journal/sd-journal.c +++ b/src/journal/sd-journal.c @@ -1940,10 +1940,14 @@ _public_ int sd_journal_get_data(sd_journal *j, const char *field, const void **                  compression = o->object.flags & OBJECT_COMPRESSION_MASK;                  if (compression) {  #if defined(HAVE_XZ) || defined(HAVE_LZ4) -                        if (decompress_startswith(compression, +                        r = decompress_startswith(compression,                                                    o->data.payload, l,                                                    &f->compress_buffer, &f->compress_buffer_size, -                                                  field, field_length, '=')) { +                                                  field, field_length, '='); +                        if (r < 0) +                                log_debug_errno(r, "Cannot decompress %s object of length %zu at offset "OFSfmt": %m", +                                                object_compressed_to_string(compression), l, p); +                        else if (r > 0) {                                  size_t rsize; diff --git a/src/journal/test-compress-benchmark.c b/src/journal/test-compress-benchmark.c index 93ea9c6318..baed0d82a4 100644 --- a/src/journal/test-compress-benchmark.c +++ b/src/journal/test-compress-benchmark.c @@ -27,7 +27,8 @@  #include "string-util.h"  #include "util.h" -typedef int (compress_t)(const void *src, uint64_t src_size, void *dst, size_t *dst_size); +typedef int (compress_t)(const void *src, uint64_t src_size, void *dst, +                         size_t dst_alloc_size, size_t *dst_size);  typedef int (decompress_t)(const void *src, uint64_t src_size,                             void **dst, size_t *dst_alloc_size, size_t* dst_size, size_t dst_max); @@ -111,8 +112,8 @@ static void test_compress_decompress(const char* label, const char* type,                  memzero(buf, MIN(size + 1000, MAX_SIZE)); -                r = compress(text, size, buf, &j); -                /* assume compression must be successful except for small inputs */ +                r = compress(text, size, buf, size, &j); +                /* assume compression must be successful except for small or random inputs */                  assert_se(r == 0 || (size < 2048 && r == -ENOBUFS) || streq(type, "random"));                  /* check for overwrites */ diff --git a/src/journal/test-compress.c b/src/journal/test-compress.c index b9d90a8988..68c9a4d76c 100644 --- a/src/journal/test-compress.c +++ b/src/journal/test-compress.c @@ -17,6 +17,10 @@    along with systemd; If not, see <http://www.gnu.org/licenses/>.  ***/ +#ifdef HAVE_LZ4 +#include <lz4.h> +#endif +  #include "alloc-util.h"  #include "compress.h"  #include "fd-util.h" @@ -38,7 +42,7 @@  #endif  typedef int (compress_blob_t)(const void *src, uint64_t src_size, -                              void *dst, size_t *dst_size); +                              void *dst, size_t dst_alloc_size, size_t *dst_size);  typedef int (decompress_blob_t)(const void *src, uint64_t src_size,                                  void **dst, size_t *dst_alloc_size,                                  size_t* dst_size, size_t dst_max); @@ -57,15 +61,14 @@ static void test_compress_decompress(int compression,                                       size_t data_len,                                       bool may_fail) {          char compressed[512]; -        size_t csize = 512; -        size_t usize = 0; +        size_t csize, usize = 0;          _cleanup_free_ char *decompressed = NULL;          int r;          log_info("/* testing %s %s blob compression/decompression */",                   object_compressed_to_string(compression), data); -        r = compress(data, data_len, compressed, &csize); +        r = compress(data, data_len, compressed, sizeof(compressed), &csize);          if (r == -ENOBUFS) {                  log_info_errno(r, "compression failed: %m");                  assert_se(may_fail); @@ -101,43 +104,45 @@ static void test_decompress_startswith(int compression,                                         size_t data_len,                                         bool may_fail) { -        char compressed[512]; -        size_t csize = 512; -        size_t usize = 0; -        _cleanup_free_ char *decompressed = NULL; +        char *compressed; +        _cleanup_free_ char *compressed1 = NULL, *compressed2 = NULL, *decompressed = NULL; +        size_t csize, usize = 0, len;          int r; -        log_info("/* testing decompress_startswith with %s on %s text*/", +        log_info("/* testing decompress_startswith with %s on %.20s text*/",                   object_compressed_to_string(compression), data); -        r = compress(data, data_len, compressed, &csize); +#define BUFSIZE_1 512 +#define BUFSIZE_2 20000 + +        compressed = compressed1 = malloc(BUFSIZE_1); +        assert_se(compressed1); +        r = compress(data, data_len, compressed, BUFSIZE_1, &csize);          if (r == -ENOBUFS) {                  log_info_errno(r, "compression failed: %m");                  assert_se(may_fail); -                return; + +                compressed = compressed2 = malloc(BUFSIZE_2); +                assert_se(compressed2); +                r = compress(data, data_len, compressed, BUFSIZE_2, &csize); +                assert(r == 0);          }          assert_se(r == 0); -        assert_se(decompress_sw(compressed, -                                csize, -                                (void **) &decompressed, -                                &usize, -                                data, strlen(data), '\0') > 0); -        assert_se(decompress_sw(compressed, -                                csize, -                                (void **) &decompressed, -                                &usize, -                                data, strlen(data), 'w') == 0); -        assert_se(decompress_sw(compressed, -                                csize, -                                (void **) &decompressed, -                                &usize, -                                "barbarbar", 9, ' ') == 0); -        assert_se(decompress_sw(compressed, -                                csize, -                                (void **) &decompressed, -                                &usize, -                                data, strlen(data), '\0') > 0); +        len = strlen(data); + +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, data, len, '\0'); +        assert_se(r > 0); +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, data, len, 'w'); +        assert_se(r == 0); +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, "barbarbar", 9, ' '); +        assert_se(r == 0); +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, data, len - 1, data[len-1]); +        assert_se(r > 0); +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, data, len - 1, 'w'); +        assert_se(r == 0); +        r = decompress_sw(compressed, csize, (void **) &decompressed, &usize, data, len, '\0'); +        assert_se(r > 0);  }  static void test_compress_stream(int compression, @@ -199,6 +204,44 @@ static void test_compress_stream(int compression,          assert_se(unlink(pattern2) == 0);  } +#ifdef HAVE_LZ4 +static void test_lz4_decompress_partial(void) { +        char buf[20000]; +        size_t buf_size = sizeof(buf), compressed; +        int r; +        _cleanup_free_ char *huge = NULL; + +#define HUGE_SIZE (4096*1024) +        huge = malloc(HUGE_SIZE); +        memset(huge, 'x', HUGE_SIZE); +        memcpy(huge, "HUGE=", 5); + +        r = LZ4_compress_limitedOutput(huge, buf, HUGE_SIZE, buf_size); +        assert_se(r >= 0); +        compressed = r; +        log_info("Compressed %i → %zu", HUGE_SIZE, compressed); + +        r = LZ4_decompress_safe(buf, huge, r, HUGE_SIZE); +        assert_se(r >= 0); +        log_info("Decompressed → %i", r); + +        r = LZ4_decompress_safe_partial(buf, huge, +                                        compressed, +                                        12, HUGE_SIZE); +        assert_se(r >= 0); +        log_info("Decompressed partial %i/%i → %i", 12, HUGE_SIZE, r); + +        /* We expect this to fail, because that's how current lz4 works. If this +         * call succeeds, then lz4 has been fixed, and we need to change our code. +         */ +        r = LZ4_decompress_safe_partial(buf, huge, +                                        compressed, +                                        12, HUGE_SIZE-1); +        assert_se(r < 0); +        log_info("Decompressed partial %i/%i → %i", 12, HUGE_SIZE-1, r); +} +#endif +  int main(int argc, char *argv[]) {          const char text[] =                  "text\0foofoofoofoo AAAA aaaaaaaaa ghost busters barbarbar FFF" @@ -206,6 +249,11 @@ int main(int argc, char *argv[]) {          char data[512] = "random\0"; +        char huge[4096*1024]; +        memset(huge, 'x', sizeof(huge)); +        memcpy(huge, "HUGE=", 5); +        char_array_0(huge); +          log_set_max_level(LOG_DEBUG);          random_bytes(data + 7, sizeof(data) - 7); @@ -215,12 +263,17 @@ int main(int argc, char *argv[]) {                                   text, sizeof(text), false);          test_compress_decompress(OBJECT_COMPRESSED_XZ, compress_blob_xz, decompress_blob_xz,                                   data, sizeof(data), true); +          test_decompress_startswith(OBJECT_COMPRESSED_XZ,                                     compress_blob_xz, decompress_startswith_xz,                                     text, sizeof(text), false);          test_decompress_startswith(OBJECT_COMPRESSED_XZ,                                     compress_blob_xz, decompress_startswith_xz,                                     data, sizeof(data), true); +        test_decompress_startswith(OBJECT_COMPRESSED_XZ, +                                   compress_blob_xz, decompress_startswith_xz, +                                   huge, sizeof(huge), true); +          test_compress_stream(OBJECT_COMPRESSED_XZ, "xzcat",                               compress_stream_xz, decompress_stream_xz, argv[0]);  #else @@ -232,15 +285,21 @@ int main(int argc, char *argv[]) {                                   text, sizeof(text), false);          test_compress_decompress(OBJECT_COMPRESSED_LZ4, compress_blob_lz4, decompress_blob_lz4,                                   data, sizeof(data), true); +          test_decompress_startswith(OBJECT_COMPRESSED_LZ4,                                     compress_blob_lz4, decompress_startswith_lz4,                                     text, sizeof(text), false);          test_decompress_startswith(OBJECT_COMPRESSED_LZ4,                                     compress_blob_lz4, decompress_startswith_lz4,                                     data, sizeof(data), true); +        test_decompress_startswith(OBJECT_COMPRESSED_LZ4, +                                   compress_blob_lz4, decompress_startswith_lz4, +                                   huge, sizeof(huge), true);          test_compress_stream(OBJECT_COMPRESSED_LZ4, "lz4cat",                               compress_stream_lz4, decompress_stream_lz4, argv[0]); + +        test_lz4_decompress_partial();  #else          log_info("/* LZ4 test skipped */");  #endif diff --git a/src/libsystemd-network/sd-lldp.c b/src/libsystemd-network/sd-lldp.c index d3ea74404b..1c696f9ef0 100644 --- a/src/libsystemd-network/sd-lldp.c +++ b/src/libsystemd-network/sd-lldp.c @@ -145,12 +145,9 @@ static int lldp_receive_frame(sd_lldp *lldp, tlv_packet *tlv) {  /* 10.3.2 LLDPDU validation: rxProcessFrame() */  int lldp_handle_packet(tlv_packet *tlv, uint16_t length) { +        bool system_description = false, system_name = false, chassis_id = false; +        bool malformed = false, port_id = false, ttl = false, end = false;          uint16_t type, len, i, l, t; -        bool chassis_id = false; -        bool malformed = false; -        bool port_id = false; -        bool ttl = false; -        bool end = false;          lldp_port *port;          uint8_t *p, *q;          sd_lldp *lldp; @@ -163,8 +160,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {          lldp = (sd_lldp *) port->userdata;          if (lldp->port->status == LLDP_PORT_STATUS_DISABLED) { -                log_lldp("Port is disabled : %s . Dropping ...", -                         lldp->port->ifname); +                log_lldp("Port: %s is disabled. Dropping.", lldp->port->ifname);                  goto out;          } @@ -182,8 +178,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                  if (type == LLDP_TYPE_END) {                          if (len != 0) { -                                log_lldp("TLV type end is not length 0. Length:%d received . Dropping ...", -                                         len); +                                log_lldp("TLV type end must be length 0 (not %d). Dropping.", len);                                  malformed = true;                                  goto out; @@ -193,8 +188,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                          break;                  } else if (type >=_LLDP_TYPE_MAX) { -                        log_lldp("TLV type not recognized %d . Dropping ...", -                                 type); +                        log_lldp("TLV type: %d not recognized. Dropping.", type);                          malformed = true;                          goto out; @@ -209,7 +203,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                  if (i <= 3) {                          if (i != type) { -                                log_lldp("TLV missing or out of order. Dropping ..."); +                                log_lldp("TLV missing or out of order. Dropping.");                                  malformed = true;                                  goto out; @@ -220,25 +214,22 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                  case LLDP_TYPE_CHASSIS_ID:                          if (len < 2) { -                                log_lldp("Received malformed Chassis ID TLV len = %d. Dropping", -                                         len); +                                log_lldp("Received malformed Chassis ID TLV length: %d. Dropping.", len);                                  malformed = true;                                  goto out;                          }                          if (chassis_id) { -                                log_lldp("Duplicate Chassis ID TLV found. Dropping ..."); +                                log_lldp("Duplicate Chassis ID TLV found. Dropping.");                                  malformed = true;                                  goto out;                          }                          /* Look what subtype it has */ -                        if (*q == LLDP_CHASSIS_SUBTYPE_RESERVED || -                            *q > LLDP_CHASSIS_SUBTYPE_LOCALLY_ASSIGNED) { -                                log_lldp("Unknown subtype: %d found in Chassis ID TLV . Dropping ...", -                                         *q); +                        if (*q == LLDP_CHASSIS_SUBTYPE_RESERVED || *q > LLDP_CHASSIS_SUBTYPE_LOCALLY_ASSIGNED) { +                                log_lldp("Unknown subtype: %d found in Chassis ID TLV. Dropping.", *q);                                  malformed = true;                                  goto out; @@ -251,25 +242,22 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                  case LLDP_TYPE_PORT_ID:                          if (len < 2) { -                                log_lldp("Received malformed Port ID TLV len = %d. Dropping", -                                         len); +                                log_lldp("Received malformed Port ID TLV length: %d. Dropping.", len);                                  malformed = true;                                  goto out;                          }                          if (port_id) { -                                log_lldp("Duplicate Port ID TLV found. Dropping ..."); +                                log_lldp("Duplicate Port ID TLV found. Dropping.");                                  malformed = true;                                  goto out;                          }                          /* Look what subtype it has */ -                        if (*q == LLDP_PORT_SUBTYPE_RESERVED || -                            *q > LLDP_PORT_SUBTYPE_LOCALLY_ASSIGNED) { -                                log_lldp("Unknown subtype: %d found in Port ID TLV . Dropping ...", -                                         *q); +                        if (*q == LLDP_PORT_SUBTYPE_RESERVED || *q > LLDP_PORT_SUBTYPE_LOCALLY_ASSIGNED) { +                                log_lldp("Unknown subtype: %d found in Port ID TLV. Dropping.", *q);                                  malformed = true;                                  goto out; @@ -282,16 +270,14 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                  case LLDP_TYPE_TTL:                          if(len != 2) { -                                log_lldp( -                                         "Received invalid lenth: %d TTL TLV. Dropping ...", -                                         len); +                                log_lldp("Received invalid TTL TLV lenth: %d. Dropping.", len);                                  malformed = true;                                  goto out;                          }                          if (ttl) { -                                log_lldp("Duplicate TTL TLV found. Dropping ..."); +                                log_lldp("Duplicate TTL TLV found. Dropping.");                                  malformed = true;                                  goto out; @@ -300,11 +286,45 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {                          ttl = true;                          break; +                case LLDP_TYPE_SYSTEM_NAME: + +                        /* According to RFC 1035 the length of a FQDN is limited to 255 characters */ +                        if (len > 255) { +                                log_lldp("Received invalid system name length: %d. Dropping.", len); +                                malformed = true; +                                goto out; +                        } + +                        if (system_name) { +                                log_lldp("Duplicate system name found. Dropping."); +                                malformed = true; +                                goto out; +                        } + +                        system_name = true; + +                        break; +                case LLDP_TYPE_SYSTEM_DESCRIPTION: + +                        /* 0 <= n <= 255 octets */ +                        if (len > 255) { +                                log_lldp("Received invalid system description length: %d. Dropping.", len); +                                malformed = true; +                                goto out; +                        } + +                        if (system_description) { +                                log_lldp("Duplicate system description found. Dropping."); +                                malformed = true; +                                goto out; +                        } + +                        system_description = true; +                        break;                  default:                          if (len == 0) { -                                log_lldp("TLV type = %d's, length 0 received . Dropping ...", -                                         type); +                                log_lldp("TLV type: %d length 0 received. Dropping.", type);                                  malformed = true;                                  goto out; @@ -314,7 +334,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {          }          if(!chassis_id || !port_id || !ttl || !end) { -                log_lldp( "One or more mandotory TLV missing . Dropping ..."); +                log_lldp("One or more mandatory TLV missing. Dropping.");                  malformed = true;                  goto out; @@ -323,7 +343,7 @@ int lldp_handle_packet(tlv_packet *tlv, uint16_t length) {          r = tlv_packet_parse_pdu(tlv, length);          if (r < 0) { -                log_lldp( "Failed to parse the TLV. Dropping ..."); +                log_lldp("Failed to parse the TLV. Dropping.");                  malformed = true;                  goto out; diff --git a/src/libsystemd/sd-event/sd-event.c b/src/libsystemd/sd-event/sd-event.c index 3191b458d1..7fccbd1a71 100644 --- a/src/libsystemd/sd-event/sd-event.c +++ b/src/libsystemd/sd-event/sd-event.c @@ -37,6 +37,7 @@  #include "process-util.h"  #include "set.h"  #include "signal-util.h" +#include "string-table.h"  #include "string-util.h"  #include "time-util.h"  #include "util.h" @@ -60,6 +61,23 @@ typedef enum EventSourceType {          _SOURCE_EVENT_SOURCE_TYPE_INVALID = -1  } EventSourceType; +static const char* const event_source_type_table[_SOURCE_EVENT_SOURCE_TYPE_MAX] = { +        [SOURCE_IO] = "io", +        [SOURCE_TIME_REALTIME] = "realtime", +        [SOURCE_TIME_BOOTTIME] = "bootime", +        [SOURCE_TIME_MONOTONIC] = "monotonic", +        [SOURCE_TIME_REALTIME_ALARM] = "realtime-alarm", +        [SOURCE_TIME_BOOTTIME_ALARM] = "boottime-alarm", +        [SOURCE_SIGNAL] = "signal", +        [SOURCE_CHILD] = "child", +        [SOURCE_DEFER] = "defer", +        [SOURCE_POST] = "post", +        [SOURCE_EXIT] = "exit", +        [SOURCE_WATCHDOG] = "watchdog", +}; + +DEFINE_PRIVATE_STRING_TABLE_LOOKUP_TO_STRING(event_source_type, int); +  /* All objects we use in epoll events start with this value, so that   * we know how to dispatch it */  typedef enum WakeupType { @@ -482,7 +500,8 @@ static void source_io_unregister(sd_event_source *s) {          r = epoll_ctl(s->event->epoll_fd, EPOLL_CTL_DEL, s->io.fd, NULL);          if (r < 0) -                log_debug_errno(errno, "Failed to remove source %s from epoll: %m", strna(s->description)); +                log_debug_errno(errno, "Failed to remove source %s (type %s) from epoll: %m", +                                strna(s->description), event_source_type_to_string(s->type));          s->io.registered = false;  } @@ -2281,12 +2300,9 @@ static int source_dispatch(sd_event_source *s) {          s->dispatching = false; -        if (r < 0) { -                if (s->description) -                        log_debug_errno(r, "Event source '%s' returned error, disabling: %m", s->description); -                else -                        log_debug_errno(r, "Event source %p returned error, disabling: %m", s); -        } +        if (r < 0) +                log_debug_errno(r, "Event source %s (type %s) returned error, disabling: %m", +                                strna(s->description), event_source_type_to_string(s->type));          if (s->n_ref == 0)                  source_free(s); @@ -2319,12 +2335,9 @@ static int event_prepare(sd_event *e) {                  r = s->prepare(s, s->userdata);                  s->dispatching = false; -                if (r < 0) { -                        if (s->description) -                                log_debug_errno(r, "Prepare callback of event source '%s' returned error, disabling: %m", s->description); -                        else -                                log_debug_errno(r, "Prepare callback of event source %p returned error, disabling: %m", s); -                } +                if (r < 0) +                        log_debug_errno(r, "Prepare callback of event source %s (type %s) returned error, disabling: %m", +                                        strna(s->description), event_source_type_to_string(s->type));                  if (s->n_ref == 0)                          source_free(s); diff --git a/src/resolve/resolved-dns-transaction.c b/src/resolve/resolved-dns-transaction.c index 32d52d8192..347b1683f9 100644 --- a/src/resolve/resolved-dns-transaction.c +++ b/src/resolve/resolved-dns-transaction.c @@ -1714,10 +1714,10 @@ static int dns_transaction_requires_rrsig(DnsTransaction *t, DnsResourceRecord *                           * RRs we are looking at. If it discovered signed DS                           * RRs, then we need to be signed, too. */ -                         if (!dt->answer_authenticated) -                                 return false; +                        if (!dt->answer_authenticated) +                                return false; -                         return dns_answer_match_key(dt->answer, dt->key, NULL); +                        return dns_answer_match_key(dt->answer, dt->key, NULL);                  }                  /* We found nothing that proves this is safe to leave diff --git a/src/shared/logs-show.c b/src/shared/logs-show.c index 193dad1943..a1f65d1a88 100644 --- a/src/shared/logs-show.c +++ b/src/shared/logs-show.c @@ -435,8 +435,9 @@ static int output_verbose(                  r = parse_field(data, length, "_SOURCE_REALTIME_TIMESTAMP=", &value, &size);                  if (r < 0) -                        log_debug_errno(r, "_SOURCE_REALTIME_TIMESTAMP invalid: %m"); +                        return r;                  else { +                        assert(r > 0);                          r = safe_atou64(value, &realtime);                          if (r < 0)                                  log_debug_errno(r, "Failed to parse realtime timestamp: %m"); | 
