diff options
59 files changed, 450 insertions, 497 deletions
| diff --git a/Makefile.am b/Makefile.am index 5d30d7d51c..783e19b2f7 100644 --- a/Makefile.am +++ b/Makefile.am @@ -714,9 +714,9 @@ man/systemd.index.xml: $(top_srcdir)/tools/make-man-index.py $(NON_INDEX_XML_FIL  	$(AM_V_at)$(MKDIR_P) $(dir $@)  	$(AM_V_GEN)$(PYTHON) $< $@ $(filter-out $<,$^) -man/systemd.directives.xml: $(top_srcdir)/tools/make-directive-index.py $(SOURCE_XML_FILES) +man/systemd.directives.xml: $(top_srcdir)/tools/make-directive-index.py man/custom-entities.ent $(SOURCE_XML_FILES)  	$(AM_V_at)$(MKDIR_P) $(dir $@) -	$(AM_V_GEN)$(PYTHON) $< $@ $(filter-out $<,$^) +	$(AM_V_GEN)$(PYTHON) $< $@ $(SOURCE_XML_FILES)  CLEANFILES += \  	man/systemd.index.xml \ @@ -762,6 +762,8 @@ libsystemd_shared_la_SOURCES = \  	src/shared/udev-util.h \  	src/shared/device-nodes.c \  	src/shared/device-nodes.h \ +	src/shared/dns-domain.c \ +	src/shared/dns-domain.h \  	src/shared/util.c \  	src/shared/util.h \  	src/shared/virt.c \ @@ -978,6 +980,7 @@ libsystemd_shared_la_CFLAGS = \  libsystemd_shared_la_LIBADD = \  	$(SELINUX_LIBS) \  	$(CAP_LIBS) \ +	$(LIBIDN_LIBS) \  	-lm  # ----------------------------------------------------------------------------- @@ -1446,7 +1449,8 @@ tests += \  	test-copy \  	test-cap-list \  	test-sigbus \ -	test-verbs +	test-verbs \ +	test-dns-domain  EXTRA_DIST += \  	test/a.service \ @@ -1577,6 +1581,15 @@ test_hostname_SOURCES = \  test_hostname_LDADD = \  	libsystemd-core.la +test_dns_domain_SOURCES = \ +	src/test/test-dns-domain.c + +test_dns_domain_LDADD = \ +	libsystemd-network.la \ +	libsystemd-internal.la \ +	libsystemd-shared.la \ +	$(LIBIDN_LIBS) +  if ENABLE_EFI  manual_tests += \  	test-boot-timestamp @@ -3382,7 +3395,8 @@ test_dhcp_client_SOURCES = \  test_dhcp_client_LDADD = \  	libsystemd-network.la \  	libsystemd-internal.la \ -	libsystemd-shared.la +	libsystemd-shared.la \ +	$(LIBIDN_LIBS)  test_dhcp_server_SOURCES = \  	src/libsystemd-network/test-dhcp-server.c @@ -3603,6 +3617,9 @@ test_unifont_LDADD = \  src/libsystemd-terminal/unifont-glyph-array.bin: tools/compile-unifont.py $(UNIFONT)  	$(AM_V_GEN)$(PYTHON) $< <$(UNIFONT) >$@ +EXTRA_DIST += \ +	tools/compile-unifont.py +  # ------------------------------------------------------------------------------  include_HEADERS += \  	src/libudev/libudev.h @@ -5405,8 +5422,6 @@ systemd_resolved_SOURCES = \  	src/resolve/resolved-link.h \  	src/resolve/resolved-link.c \  	src/resolve/resolved-def.h \ -	src/resolve/resolved-dns-domain.h \ -	src/resolve/resolved-dns-domain.c \  	src/resolve/resolved-dns-rr.h \  	src/resolve/resolved-dns-rr.c \  	src/resolve/resolved-dns-question.h \ @@ -5476,20 +5491,6 @@ GENERAL_ALIASES += \  nodist_pkgsysconf_DATA += \  	src/resolve/resolved.conf -tests += \ -	test-dns-domain - -test_dns_domain_SOURCES = \ -	src/resolve/resolved-dns-domain.h \ -	src/resolve/resolved-dns-domain.c \ -	src/resolve/test-dns-domain.c - -test_dns_domain_LDADD = \ -	libsystemd-network.la \ -	libsystemd-internal.la \ -	libsystemd-shared.la \ -	$(LIBIDN_LIBS) -  libnss_resolve_la_SOURCES = \  	src/nss-resolve/nss-resolve.sym \  	src/nss-resolve/nss-resolve.c @@ -5520,8 +5521,6 @@ systemd_resolve_host_SOURCES = \  	src/resolve/resolved-dns-answer.h \  	src/resolve/resolved-dns-question.c \  	src/resolve/resolved-dns-question.h \ -	src/resolve/resolved-dns-domain.c \ -	src/resolve/resolved-dns-domain.h \  	src/resolve/dns-type.c \  	src/resolve/dns-type.h @@ -5551,7 +5550,8 @@ systemd_networkd_SOURCES = \  	src/network/networkd.c  systemd_networkd_LDADD = \ -	libsystemd-networkd-core.la +	libsystemd-networkd-core.la \ +	$(LIBIDN_LIBS)  if HAVE_LIBIPTC  systemd_networkd_LDADD += \ @@ -5650,7 +5650,8 @@ test_network_SOURCES = \  	src/network/test-network.c  test_network_LDADD = \ -	libsystemd-networkd-core.la +	libsystemd-networkd-core.la \ +	$(LIBIDN_LIBS)  if HAVE_LIBIPTC  test_network_LDADD += \ diff --git a/configure.ac b/configure.ac index 5719075934..878b23b061 100644 --- a/configure.ac +++ b/configure.ac @@ -159,10 +159,13 @@ CC_CHECK_FLAGS_APPEND([with_cflags], [CFLAGS], [\          -Wdeclaration-after-statement \          -Wfloat-equal \          -Wsuggest-attribute=noreturn \ -        -Wmissing-prototypes \ +        -Werror=missing-prototypes \ +        -Werror=implicit-function-declaration \ +        -Werror=missing-declarations \ +        -Werror=return-type \ +        -Werror=shadow \          -Wstrict-prototypes \          -Wredundant-decls \ -        -Wmissing-declarations \          -Wmissing-noreturn \          -Wshadow \          -Wendif-labels \ diff --git a/hwdb/60-keyboard.hwdb b/hwdb/60-keyboard.hwdb index 9c7e553a41..007c6a809e 100644 --- a/hwdb/60-keyboard.hwdb +++ b/hwdb/60-keyboard.hwdb @@ -980,6 +980,8 @@ evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*550P*:pvr*   KEYBOARD_KEY_a9=!                                      # Fn Lock - Function lock off  # Series 7 / 9 +evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*350V*:pvr* +evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*670Z*:pvr*  evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*700Z*:pvr*  evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*700G*:pvr*  evdev:atkbd:dmi:bvn*:bvr*:bd*:svn[sS][aA][mM][sS][uU][nN][gG]*:pn*900X[34]*:pvr* diff --git a/hwdb/70-mouse.hwdb b/hwdb/70-mouse.hwdb index 2784b941a1..6c77b74e53 100644 --- a/hwdb/70-mouse.hwdb +++ b/hwdb/70-mouse.hwdb @@ -169,6 +169,8 @@ mouse:usb:v046dpc00e:name:Logitech USB-PS/2 Optical Mouse:  mouse:usb:v046dpc01b:name:Logitech USB-PS/2 Optical Mouse:  # Logitech USB-PS/2 M-BT58  mouse:usb:v046dpc03e:name:Logitech USB-PS/2 Optical Mouse: +# Logitech TrackMan Marble Wheel USB +mouse:usb:v046dpc401:name:Logitech USB-PS/2 Trackball:   MOUSE_DPI=400@125  # Lenovo USB mouse model MO28UOL @@ -180,6 +182,7 @@ mouse:usb:v046dpc045:name:Logitech USB-PS/2 Optical Mouse:   MOUSE_DPI=600@125  # Logitech Wireless Mouse M325 +mouse:usb:v046dp400a:name:Logitech M325:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:400a:   MOUSE_DPI=600@166   MOUSE_WHEEL_CLICK_ANGLE=20 @@ -213,8 +216,10 @@ mouse:usb:v046dp1028:name:Logitech M570:   MOUSE_DPI=540@167  # Logitech Wireless Mouse M185 +mouse:usb:v046dp4008:name:Logitech M185:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:4008:  # Logitech M705 (marathon mouse) +mouse:usb:v046dp101b:name:Logitech M705:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:101b:   MOUSE_DPI=800@166 @@ -225,6 +230,8 @@ mouse:usb:v046dpc24e:name:Logitech G500s Laser Gaming Mouse:   MOUSE_DPI=400@500 *800@500 2000@500  # Logitech B605 Wireless Mouse (also M505) +mouse:usb:v046dp101d:name:Logitech B605: +mouse:usb:v046dp101d:name:Logitech M505:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:101d:   MOUSE_DPI=900@166 @@ -253,10 +260,12 @@ mouse:usb:v046dpc069:name:Logitech USB Laser Mouse:   MOUSE_DPI=1200@125  # Logitech T620 (or, the soap) +mouse:usb:v046dp4027:name:Logitech T620:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:4027:   MOUSE_DPI=1200@250  # Logitech ZoneTouch Mouse T400 +mouse:usb:v046dp4026:name:Logitech T400:  mouse:usb:v046dpc52b:name:Logitech Unifying Device. Wireless PID:4026:   MOUSE_DPI=1300@166 diff --git a/man/systemd-udevd.service.xml b/man/systemd-udevd.service.xml index c0d323033c..7a4b62aaf1 100644 --- a/man/systemd-udevd.service.xml +++ b/man/systemd-udevd.service.xml @@ -106,7 +106,7 @@          <term><option>--event-timeout=</option></term>          <listitem>            <para>Set the number of seconds to wait for events to finish. After -          this time the event will be terminated. The default is 30 seconds.</para> +          this time the event will be terminated. The default is 180 seconds.</para>          </listitem>        </varlistentry> diff --git a/man/udev.xml b/man/udev.xml index d5d8a17cdb..70f4f59c08 100644 --- a/man/udev.xml +++ b/man/udev.xml @@ -472,7 +472,8 @@                <varlistentry>                  <term><literal>program</literal></term>                  <listitem> -                  <para>Execute an external program specified as the assigned value and +                  <para>Execute an external program specified as the assigned +                  value and if it returns successfully                    import its output, which must be in environment key                    format. Path specification, command/argument separation,                    and quoting work like in <varname>RUN</varname>.</para> diff --git a/src/core/manager.c b/src/core/manager.c index 564fb5d579..eb80dd1b46 100644 --- a/src/core/manager.c +++ b/src/core/manager.c @@ -1546,7 +1546,7 @@ static int manager_dispatch_notify_fd(sd_event_source *source, int fd, uint32_t                          return -errno;                  } -                for (cmsg = CMSG_FIRSTHDR(&msghdr); cmsg; cmsg = CMSG_NXTHDR(&msghdr, cmsg)) { +                CMSG_FOREACH(cmsg, &msghdr) {                          if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS) {                                  fd_array = (int*) CMSG_DATA(cmsg); diff --git a/src/core/namespace.c b/src/core/namespace.c index 01a817bf23..045321e1d4 100644 --- a/src/core/namespace.c +++ b/src/core/namespace.c @@ -696,12 +696,11 @@ int setup_netns(int netns_storage_socket[2]) {          } else {                  /* Yay, found something, so let's join the namespace */ -                for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) { +                CMSG_FOREACH(cmsg, &mh)                          if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS) {                                  assert(cmsg->cmsg_len == CMSG_LEN(sizeof(int)));                                  netns = *(int*) CMSG_DATA(cmsg);                          } -                }                  if (setns(netns, CLONE_NEWNET) < 0) {                          r = -errno; diff --git a/src/core/selinux-access.c b/src/core/selinux-access.c index 5e9a4a5e02..decd42f95a 100644 --- a/src/core/selinux-access.c +++ b/src/core/selinux-access.c @@ -261,7 +261,7 @@ int mac_selinux_generic_access_check(          audit_info.path = path;          audit_info.cmdline = cl; -        r = selinux_check_access((security_context_t) scon, fcon, tclass, permission, &audit_info); +        r = selinux_check_access(scon, fcon, tclass, permission, &audit_info);          if (r < 0)                  r = sd_bus_error_setf(error, SD_BUS_ERROR_ACCESS_DENIED, "SELinux policy denies access."); diff --git a/src/import/importd.c b/src/import/importd.c index e2df44ad26..05a619ac0c 100644 --- a/src/import/importd.c +++ b/src/import/importd.c @@ -599,7 +599,7 @@ static int manager_on_notify(sd_event_source *s, int fd, uint32_t revents, void          cmsg_close_all(&msghdr); -        for (cmsg = CMSG_FIRSTHDR(&msghdr); cmsg; cmsg = CMSG_NXTHDR(&msghdr, cmsg)) { +        CMSG_FOREACH(cmsg, &msghdr) {                  if (cmsg->cmsg_level == SOL_SOCKET &&                             cmsg->cmsg_type == SCM_CREDENTIALS &&                             cmsg->cmsg_len == CMSG_LEN(sizeof(struct ucred))) { diff --git a/src/journal/journald-server.c b/src/journal/journald-server.c index 3353024f4e..32da8d61fc 100644 --- a/src/journal/journald-server.c +++ b/src/journal/journald-server.c @@ -1177,7 +1177,7 @@ int server_process_datagram(sd_event_source *es, int fd, uint32_t revents, void                          return -errno;                  } -                for (cmsg = CMSG_FIRSTHDR(&msghdr); cmsg; cmsg = CMSG_NXTHDR(&msghdr, cmsg)) { +                CMSG_FOREACH(cmsg, &msghdr) {                          if (cmsg->cmsg_level == SOL_SOCKET &&                              cmsg->cmsg_type == SCM_CREDENTIALS && diff --git a/src/journal/journald-stream.c b/src/journal/journald-stream.c index b572147a56..db2f581972 100644 --- a/src/journal/journald-stream.c +++ b/src/journal/journald-stream.c @@ -59,10 +59,7 @@ struct StdoutStream {          int fd;          struct ucred ucred; -#ifdef HAVE_SELINUX -        security_context_t security_context; -#endif - +        char *label;          char *identifier;          char *unit_id;          int priority; @@ -99,12 +96,7 @@ void stdout_stream_free(StdoutStream *s) {          }          safe_close(s->fd); - -#ifdef HAVE_SELINUX -        if (s->security_context) -                freecon(s->security_context); -#endif - +        free(s->label);          free(s->identifier);          free(s->unit_id);          free(s->state_file); @@ -225,8 +217,7 @@ static int stdout_stream_log(StdoutStream *s, const char *p) {          char syslog_facility[sizeof("SYSLOG_FACILITY=")-1 + DECIMAL_STR_MAX(int) + 1];          _cleanup_free_ char *message = NULL, *syslog_identifier = NULL;          unsigned n = 0; -        char *label = NULL; -        size_t label_len = 0; +        size_t label_len;          assert(s);          assert(p); @@ -271,14 +262,8 @@ static int stdout_stream_log(StdoutStream *s, const char *p) {          if (message)                  IOVEC_SET_STRING(iovec[n++], message); -#ifdef HAVE_SELINUX -        if (s->security_context) { -                label = (char*) s->security_context; -                label_len = strlen((char*) s->security_context); -        } -#endif - -        server_dispatch_message(s->server, iovec, n, ELEMENTSOF(iovec), &s->ucred, NULL, label, label_len, s->unit_id, priority, 0); +        label_len = s->label ? strlen(s->label) : 0; +        server_dispatch_message(s->server, iovec, n, ELEMENTSOF(iovec), &s->ucred, NULL, s->label, label_len, s->unit_id, priority, 0);          return 0;  } @@ -489,12 +474,11 @@ static int stdout_stream_install(Server *s, int fd, StdoutStream **ret) {          if (r < 0)                  return log_error_errno(r, "Failed to determine peer credentials: %m"); -#ifdef HAVE_SELINUX          if (mac_selinux_use()) { -                if (getpeercon(fd, &stream->security_context) < 0 && errno != ENOPROTOOPT) -                        log_error_errno(errno, "Failed to determine peer security context: %m"); +                r = getpeersec(fd, &stream->label); +                if (r < 0 && r != -EOPNOTSUPP) +                        (void) log_warning_errno(r, "Failed to determine peer security context: %m");          } -#endif          (void) shutdown(fd, SHUT_WR); diff --git a/src/libsystemd-network/sd-dhcp-client.c b/src/libsystemd-network/sd-dhcp-client.c index 0193e42d65..6853038b67 100644 --- a/src/libsystemd-network/sd-dhcp-client.c +++ b/src/libsystemd-network/sd-dhcp-client.c @@ -1588,7 +1588,7 @@ static int client_receive_message_raw(sd_event_source *s, int fd,          } else if ((size_t)len < sizeof(DHCPPacket))                  return 0; -        for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) { +        CMSG_FOREACH(cmsg, &msg) {                  if (cmsg->cmsg_level == SOL_PACKET &&                      cmsg->cmsg_type == PACKET_AUXDATA &&                      cmsg->cmsg_len == CMSG_LEN(sizeof(struct tpacket_auxdata))) { diff --git a/src/libsystemd-network/sd-dhcp-lease.c b/src/libsystemd-network/sd-dhcp-lease.c index 8a4220621b..d8bc76edda 100644 --- a/src/libsystemd-network/sd-dhcp-lease.c +++ b/src/libsystemd-network/sd-dhcp-lease.c @@ -32,6 +32,7 @@  #include "dhcp-lease-internal.h"  #include "sd-dhcp-lease.h"  #include "network-internal.h" +#include "dns-domain.h"  int sd_dhcp_lease_get_address(sd_dhcp_lease *lease, struct in_addr *addr) {          assert_return(lease, -EINVAL); @@ -504,9 +505,18 @@ int dhcp_lease_parse_options(uint8_t code, uint8_t len, const uint8_t *option,                  if (e)                          *e = 0; -                if (!hostname_is_valid(domainname) || is_localhost(domainname)) +                if (is_localhost(domainname))                          break; +                r = dns_name_is_valid(domainname); +                if (r <= 0) { +                        if (r < 0) +                                log_error_errno(r, "Failed to validate domain name: %s: %m", domainname); +                        if (r == 0) +                                log_warning("Domain name is not valid, ignoring: %s", domainname); +                        break; +                } +                  free(lease->domainname);                  lease->domainname = domainname;                  domainname = NULL; diff --git a/src/libsystemd-network/sd-dhcp-server.c b/src/libsystemd-network/sd-dhcp-server.c index a0a2320efa..cc5e032344 100644 --- a/src/libsystemd-network/sd-dhcp-server.c +++ b/src/libsystemd-network/sd-dhcp-server.c @@ -902,7 +902,7 @@ static int server_receive_message(sd_event_source *s, int fd,          else if ((size_t)len < sizeof(DHCPMessage))                  return 0; -        for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) { +        CMSG_FOREACH(cmsg, &msg) {                  if (cmsg->cmsg_level == IPPROTO_IP &&                      cmsg->cmsg_type == IP_PKTINFO &&                      cmsg->cmsg_len == CMSG_LEN(sizeof(struct in_pktinfo))) { diff --git a/src/libsystemd/sd-bus/bus-bloom.c b/src/libsystemd/sd-bus/bus-bloom.c index 3556774074..91fab90cb0 100644 --- a/src/libsystemd/sd-bus/bus-bloom.c +++ b/src/libsystemd/sd-bus/bus-bloom.c @@ -116,11 +116,19 @@ void bloom_add_prefixes(uint64_t filter[], size_t size, unsigned k, const char *          p = stpcpy(stpcpy(c, a), ":");          strcpy(p, b); +        bloom_add_data(filter, size, k, c, n); +          for (;;) {                  char *e;                  e = strrchr(p, sep); -                if (!e || e == p) +                if (!e) +                        break; + +                *(e + 1) = 0; +                bloom_add_data(filter, size, k, c, e - c + 1); + +                if (e == p)                          break;                  *e = 0; diff --git a/src/libsystemd/sd-bus/bus-container.c b/src/libsystemd/sd-bus/bus-container.c index f157c25bba..fa7a207448 100644 --- a/src/libsystemd/sd-bus/bus-container.c +++ b/src/libsystemd/sd-bus/bus-container.c @@ -222,7 +222,7 @@ int bus_container_connect_kernel(sd_bus *b) {          if (recvmsg(pair[0], &mh, MSG_NOSIGNAL|MSG_CMSG_CLOEXEC) < 0)                  return -errno; -        for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) +        CMSG_FOREACH(cmsg, &mh)                  if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS) {                          int *fds;                          unsigned n_fds; diff --git a/src/libsystemd/sd-bus/bus-control.c b/src/libsystemd/sd-bus/bus-control.c index 43ddfc651d..7a59702cb2 100644 --- a/src/libsystemd/sd-bus/bus-control.c +++ b/src/libsystemd/sd-bus/bus-control.c @@ -979,8 +979,10 @@ static int bus_get_owner_creds_dbus1(sd_bus *bus, uint64_t mask, sd_bus_creds **          _cleanup_bus_creds_unref_ sd_bus_creds *c = NULL;          pid_t pid = 0;          int r; +        bool do_label = bus->label && (mask & SD_BUS_CREDS_SELINUX_CONTEXT); -        if (!bus->ucred_valid && !isempty(bus->label)) +        /* Avoid allocating anything if we have no chance of returning useful data */ +        if (!bus->ucred_valid && !do_label)                  return -ENODATA;          c = bus_creds_new(); @@ -1004,7 +1006,7 @@ static int bus_get_owner_creds_dbus1(sd_bus *bus, uint64_t mask, sd_bus_creds **                  }          } -        if (!isempty(bus->label) && (mask & SD_BUS_CREDS_SELINUX_CONTEXT)) { +        if (do_label) {                  c->label = strdup(bus->label);                  if (!c->label)                          return -ENOMEM; @@ -1289,10 +1291,8 @@ int bus_add_match_internal_kernel(                          break;                  case BUS_MATCH_PATH_NAMESPACE: -                        if (!streq(c->value_str, "/")) { -                                bloom_add_pair(bloom, bus->bloom_size, bus->bloom_n_hash, "path-slash-prefix", c->value_str); -                                using_bloom = true; -                        } +                        bloom_add_pair(bloom, bus->bloom_size, bus->bloom_n_hash, "path-slash-prefix", c->value_str); +                        using_bloom = true;                          break;                  case BUS_MATCH_ARG...BUS_MATCH_ARG_LAST: { @@ -1308,11 +1308,18 @@ int bus_add_match_internal_kernel(                  }                  case BUS_MATCH_ARG_PATH...BUS_MATCH_ARG_PATH_LAST: { -                        char buf[sizeof("arg")-1 + 2 + sizeof("-slash-prefix")]; - -                        xsprintf(buf, "arg%i-slash-prefix", c->type - BUS_MATCH_ARG_PATH); -                        bloom_add_pair(bloom, bus->bloom_size, bus->bloom_n_hash, buf, c->value_str); -                        using_bloom = true; +                        /* +                         * XXX: DBus spec defines arg[0..63]path= matching to be +                         * a two-way glob. That is, if either string is a prefix +                         * of the other, it matches. +                         * This is really hard to realize in bloom-filters, as +                         * we would have to create a bloom-match for each prefix +                         * of @c->value_str. This is excessive, hence we just +                         * ignore all those matches and accept everything from +                         * the kernel. People should really avoid those matches. +                         * If they're used in real-life some day, we will have +                         * to properly support multiple-matches here. +                         */                          break;                  } diff --git a/src/libsystemd/sd-bus/bus-creds.c b/src/libsystemd/sd-bus/bus-creds.c index 4d67619cf8..1c365b7fcd 100644 --- a/src/libsystemd/sd-bus/bus-creds.c +++ b/src/libsystemd/sd-bus/bus-creds.c @@ -773,11 +773,13 @@ int bus_creds_add_more(sd_bus_creds *c, uint64_t mask, pid_t pid, pid_t tid) {                  return 0;          /* Try to retrieve PID from creds if it wasn't passed to us */ -        if (pid <= 0 && (c->mask & SD_BUS_CREDS_PID)) +        if (pid > 0) { +                c->pid = pid; +                c->mask |= SD_BUS_CREDS_PID; +        } else if (c->mask & SD_BUS_CREDS_PID)                  pid = c->pid; - -        /* Without pid we cannot do much... */ -        if (pid <= 0) +        else +                /* Without pid we cannot do much... */                  return 0;          /* Try to retrieve TID from creds if it wasn't passed to us */ @@ -789,9 +791,6 @@ int bus_creds_add_more(sd_bus_creds *c, uint64_t mask, pid_t pid, pid_t tid) {          if (missing == 0)                  return 0; -        c->pid = pid; -        c->mask |= SD_BUS_CREDS_PID; -          if (tid > 0) {                  c->tid = tid;                  c->mask |= SD_BUS_CREDS_TID; diff --git a/src/libsystemd/sd-bus/bus-internal.c b/src/libsystemd/sd-bus/bus-internal.c index 91b288cd25..37793e48ed 100644 --- a/src/libsystemd/sd-bus/bus-internal.c +++ b/src/libsystemd/sd-bus/bus-internal.c @@ -211,6 +211,17 @@ bool member_name_is_valid(const char *p) {          return true;  } +/* + * Complex pattern match + * This checks whether @a is a 'complex-prefix' of @b, or @b is a + * 'complex-prefix' of @a, based on strings that consist of labels with @c as + * spearator. This function returns true if: + *   - both strings are equal + *   - either is a prefix of the other and ends with @c + * The second rule makes sure that either string needs to be fully included in + * the other, and the string which is considered the prefix needs to end with a + * separator. + */  static bool complex_pattern_check(char c, const char *a, const char *b) {          bool separator = false; @@ -222,9 +233,7 @@ static bool complex_pattern_check(char c, const char *a, const char *b) {          for (;;) {                  if (*a != *b) -                        return (separator && (*a == 0 || *b == 0)) || -                                (*a == 0 && *b == c && b[1] == 0) || -                                (*b == 0 && *a == c && a[1] == 0); +                        return (separator && (*a == 0 || *b == 0));                  if (*a == 0)                          return true; @@ -243,7 +252,18 @@ bool path_complex_pattern(const char *pattern, const char *value) {          return complex_pattern_check('/', pattern, value);  } +/* + * Simple pattern match + * This checks whether @a is a 'simple-prefix' of @b, based on strings that + * consist of labels with @c as separator. This function returns true, if: + *   - if @a and @b are equal + *   - if @a is a prefix of @b, and the first following character in @b (or the + *     last character in @a) is @c + * The second rule basically makes sure that if @a is a prefix of @b, then @b + * must follow with a new label separated by @c. It cannot extend the label. + */  static bool simple_pattern_check(char c, const char *a, const char *b) { +        bool separator = false;          if (!a && !b)                  return true; @@ -253,11 +273,13 @@ static bool simple_pattern_check(char c, const char *a, const char *b) {          for (;;) {                  if (*a != *b) -                        return *a == 0 && *b == c; +                        return *a == 0 && (*b == c || separator);                  if (*a == 0)                          return true; +                separator = *a == c; +                  a++, b++;          }  } diff --git a/src/libsystemd/sd-bus/bus-internal.h b/src/libsystemd/sd-bus/bus-internal.h index 1351938c80..2ee0eabc02 100644 --- a/src/libsystemd/sd-bus/bus-internal.h +++ b/src/libsystemd/sd-bus/bus-internal.h @@ -261,7 +261,7 @@ struct sd_bus {          usec_t auth_timeout;          struct ucred ucred; -        char label[NAME_MAX]; +        char *label;          uint64_t creds_mask; diff --git a/src/libsystemd/sd-bus/bus-kernel.c b/src/libsystemd/sd-bus/bus-kernel.c index 572a9c6e64..3aaaabf4ed 100644 --- a/src/libsystemd/sd-bus/bus-kernel.c +++ b/src/libsystemd/sd-bus/bus-kernel.c @@ -498,7 +498,6 @@ static int bus_kernel_make_message(sd_bus *bus, struct kdbus_msg *k) {                          footer, footer_size,                          n_bytes,                          fds, n_fds, -                        NULL,                          seclabel, 0, &m);          if (r < 0)                  return r; diff --git a/src/libsystemd/sd-bus/bus-message.c b/src/libsystemd/sd-bus/bus-message.c index 6ee209dd1b..c38b2a5fa5 100644 --- a/src/libsystemd/sd-bus/bus-message.c +++ b/src/libsystemd/sd-bus/bus-message.c @@ -435,7 +435,6 @@ int bus_message_from_header(                  size_t message_size,                  int *fds,                  unsigned n_fds, -                const struct ucred *ucred,                  const char *label,                  size_t extra,                  sd_bus_message **ret) { @@ -528,23 +527,6 @@ int bus_message_from_header(          m->fds = fds;          m->n_fds = n_fds; -        if (ucred) { -                m->creds.pid = ucred->pid; -                m->creds.euid = ucred->uid; -                m->creds.egid = ucred->gid; - -                /* Due to namespace translations some data might be -                 * missing from this ucred record. */ -                if (m->creds.pid > 0) -                        m->creds.mask |= SD_BUS_CREDS_PID; - -                if (m->creds.euid != UID_INVALID) -                        m->creds.mask |= SD_BUS_CREDS_EUID; - -                if (m->creds.egid != GID_INVALID) -                        m->creds.mask |= SD_BUS_CREDS_EGID; -        } -          if (label) {                  m->creds.label = (char*) m + ALIGN(sizeof(sd_bus_message)) + ALIGN(extra);                  memcpy(m->creds.label, label, label_sz + 1); @@ -565,7 +547,6 @@ int bus_message_from_malloc(                  size_t length,                  int *fds,                  unsigned n_fds, -                const struct ucred *ucred,                  const char *label,                  sd_bus_message **ret) { @@ -579,7 +560,7 @@ int bus_message_from_malloc(                          buffer, length,                          length,                          fds, n_fds, -                        ucred, label, +                        label,                          0, &m);          if (r < 0)                  return r; diff --git a/src/libsystemd/sd-bus/bus-message.h b/src/libsystemd/sd-bus/bus-message.h index d784e603dd..088d5b1109 100644 --- a/src/libsystemd/sd-bus/bus-message.h +++ b/src/libsystemd/sd-bus/bus-message.h @@ -205,7 +205,6 @@ int bus_message_from_header(                  size_t message_size,                  int *fds,                  unsigned n_fds, -                const struct ucred *ucred,                  const char *label,                  size_t extra,                  sd_bus_message **ret); @@ -216,7 +215,6 @@ int bus_message_from_malloc(                  size_t length,                  int *fds,                  unsigned n_fds, -                const struct ucred *ucred,                  const char *label,                  sd_bus_message **ret); diff --git a/src/libsystemd/sd-bus/bus-socket.c b/src/libsystemd/sd-bus/bus-socket.c index 93ebe80b07..322d57ddbb 100644 --- a/src/libsystemd/sd-bus/bus-socket.c +++ b/src/libsystemd/sd-bus/bus-socket.c @@ -500,11 +500,8 @@ static int bus_socket_read_auth(sd_bus *b) {          void *p;          union {                  struct cmsghdr cmsghdr; -                uint8_t buf[CMSG_SPACE(sizeof(int) * BUS_FDS_MAX) + -                            CMSG_SPACE(sizeof(struct ucred)) + -                            CMSG_SPACE(NAME_MAX)]; /*selinux label */ +                uint8_t buf[CMSG_SPACE(sizeof(int) * BUS_FDS_MAX)];          } control; -        struct cmsghdr *cmsg;          bool handle_cmsg = false;          assert(b); @@ -555,7 +552,9 @@ static int bus_socket_read_auth(sd_bus *b) {          b->rbuffer_size += k;          if (handle_cmsg) { -                for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) { +                struct cmsghdr *cmsg; + +                CMSG_FOREACH(cmsg, &mh)                          if (cmsg->cmsg_level == SOL_SOCKET &&                              cmsg->cmsg_type == SCM_RIGHTS) {                                  int j; @@ -566,30 +565,9 @@ static int bus_socket_read_auth(sd_bus *b) {                                  j = (cmsg->cmsg_len - CMSG_LEN(0)) / sizeof(int);                                  close_many((int*) CMSG_DATA(cmsg), j);                                  return -EIO; - -                        } else if (cmsg->cmsg_level == SOL_SOCKET && -                                   cmsg->cmsg_type == SCM_CREDENTIALS && -                                   cmsg->cmsg_len == CMSG_LEN(sizeof(struct ucred))) { - -                                /* Ignore bogus data, which we might -                                 * get on socketpair() sockets */ -                                if (((struct ucred*) CMSG_DATA(cmsg))->pid != 0) { -                                        memcpy(&b->ucred, CMSG_DATA(cmsg), sizeof(struct ucred)); -                                        b->ucred_valid = true; -                                } - -                        } else if (cmsg->cmsg_level == SOL_SOCKET && -                                   cmsg->cmsg_type == SCM_SECURITY) { - -                                size_t l; - -                                l = cmsg->cmsg_len - CMSG_LEN(0); -                                if (l > 0) { -                                        memcpy(&b->label, CMSG_DATA(cmsg), l); -                                        b->label[l] = 0; -                                } -                        } -                } +                        } else +                                log_debug("Got unexpected auxiliary data with level=%d and type=%d", +                                          cmsg->cmsg_level, cmsg->cmsg_type);          }          r = bus_socket_auth_verify(b); @@ -600,18 +578,8 @@ static int bus_socket_read_auth(sd_bus *b) {  }  void bus_socket_setup(sd_bus *b) { -        int enable; -          assert(b); -        /* Enable SO_PASSCRED + SO_PASSEC. We try this on any -         * socket, just in case. */ -        enable = !b->bus_client; -        (void) setsockopt(b->input_fd, SOL_SOCKET, SO_PASSCRED, &enable, sizeof(enable)); - -        enable = !b->bus_client && (b->attach_flags & KDBUS_ATTACH_SECLABEL); -        (void) setsockopt(b->input_fd, SOL_SOCKET, SO_PASSSEC, &enable, sizeof(enable)); -          /* Increase the buffers to 8 MB */          fd_inc_rcvbuf(b->input_fd, SNDBUF_SIZE);          fd_inc_sndbuf(b->output_fd, SNDBUF_SIZE); @@ -622,10 +590,17 @@ void bus_socket_setup(sd_bus *b) {  }  static void bus_get_peercred(sd_bus *b) { +        int r; +          assert(b);          /* Get the peer for socketpair() sockets */          b->ucred_valid = getpeercred(b->input_fd, &b->ucred) >= 0; + +        /* Get the SELinux context of the peer */ +        r = getpeersec(b->input_fd, &b->label); +        if (r < 0 && r != -EOPNOTSUPP) +                log_debug_errno(r, "Failed to determine peer security context: %m");  }  static int bus_socket_start_auth_client(sd_bus *b) { @@ -915,7 +890,6 @@ static int bus_socket_make_message(sd_bus *bus, size_t size) {                                      bus->rbuffer, size,                                      bus->fds, bus->n_fds,                                      NULL, -                                    NULL,                                      &t);          if (r < 0) {                  free(b); @@ -942,11 +916,8 @@ int bus_socket_read_message(sd_bus *bus) {          void *b;          union {                  struct cmsghdr cmsghdr; -                uint8_t buf[CMSG_SPACE(sizeof(int) * BUS_FDS_MAX) + -                            CMSG_SPACE(sizeof(struct ucred)) + -                            CMSG_SPACE(NAME_MAX)]; /*selinux label */ +                uint8_t buf[CMSG_SPACE(sizeof(int) * BUS_FDS_MAX)];          } control; -        struct cmsghdr *cmsg;          bool handle_cmsg = false;          assert(bus); @@ -992,7 +963,9 @@ int bus_socket_read_message(sd_bus *bus) {          bus->rbuffer_size += k;          if (handle_cmsg) { -                for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) { +                struct cmsghdr *cmsg; + +                CMSG_FOREACH(cmsg, &mh)                          if (cmsg->cmsg_level == SOL_SOCKET &&                              cmsg->cmsg_type == SCM_RIGHTS) {                                  int n, *f; @@ -1017,28 +990,9 @@ int bus_socket_read_message(sd_bus *bus) {                                  memcpy(f + bus->n_fds, CMSG_DATA(cmsg), n * sizeof(int));                                  bus->fds = f;                                  bus->n_fds += n; -                        } else if (cmsg->cmsg_level == SOL_SOCKET && -                                   cmsg->cmsg_type == SCM_CREDENTIALS && -                                   cmsg->cmsg_len == CMSG_LEN(sizeof(struct ucred))) { - -                                /* Ignore bogus data, which we might -                                 * get on socketpair() sockets */ -                                if (((struct ucred*) CMSG_DATA(cmsg))->pid != 0) { -                                        memcpy(&bus->ucred, CMSG_DATA(cmsg), sizeof(struct ucred)); -                                        bus->ucred_valid = true; -                                } - -                        } else if (cmsg->cmsg_level == SOL_SOCKET && -                                   cmsg->cmsg_type == SCM_SECURITY) { - -                                size_t l; -                                l = cmsg->cmsg_len - CMSG_LEN(0); -                                if (l > 0) { -                                        memcpy(&bus->label, CMSG_DATA(cmsg), l); -                                        bus->label[l] = 0; -                                } -                        } -                } +                        } else +                                log_debug("Got unexpected auxiliary data with level=%d and type=%d", +                                          cmsg->cmsg_level, cmsg->cmsg_type);          }          r = bus_socket_read_message_need(bus, &need); diff --git a/src/libsystemd/sd-bus/sd-bus.c b/src/libsystemd/sd-bus/sd-bus.c index edc27aef87..2805b29839 100644 --- a/src/libsystemd/sd-bus/sd-bus.c +++ b/src/libsystemd/sd-bus/sd-bus.c @@ -49,6 +49,21 @@  #include "bus-track.h"  #include "bus-slot.h" +#define log_debug_bus_message(m)                                         \ +        do {                                                             \ +                sd_bus_message *_mm = (m);                               \ +                log_debug("Got message type=%s sender=%s destination=%s object=%s interface=%s member=%s cookie=%" PRIu64 " reply_cookie=%" PRIu64 " error=%s", \ +                          bus_message_type_to_string(_mm->header->type), \ +                          strna(sd_bus_message_get_sender(_mm)),         \ +                          strna(sd_bus_message_get_destination(_mm)),    \ +                          strna(sd_bus_message_get_path(_mm)),           \ +                          strna(sd_bus_message_get_interface(_mm)),      \ +                          strna(sd_bus_message_get_member(_mm)),         \ +                          BUS_MESSAGE_COOKIE(_mm),                       \ +                          _mm->reply_cookie,                             \ +                          strna(_mm->error.message));                    \ +        } while (false) +  static int bus_poll(sd_bus *bus, bool need_more, uint64_t timeout_usec);  static int attach_io_events(sd_bus *b);  static void detach_io_events(sd_bus *b); @@ -116,6 +131,7 @@ static void bus_free(sd_bus *b) {          if (b->kdbus_buffer)                  munmap(b->kdbus_buffer, KDBUS_POOL_SIZE); +        free(b->label);          free(b->rbuffer);          free(b->unique_name);          free(b->auth_buffer); @@ -1992,6 +2008,7 @@ _public_ int sd_bus_call(                                  memmove(bus->rqueue + i, bus->rqueue + i + 1, sizeof(sd_bus_message*) * (bus->rqueue_size - i - 1));                                  bus->rqueue_size--; +                                log_debug_bus_message(incoming);                                  if (incoming->header->type == SD_BUS_MESSAGE_METHOD_RETURN) { @@ -2480,16 +2497,7 @@ static int process_message(sd_bus *bus, sd_bus_message *m) {          bus->current_message = m;          bus->iteration_counter++; -        log_debug("Got message type=%s sender=%s destination=%s object=%s interface=%s member=%s cookie=%" PRIu64 " reply_cookie=%" PRIu64 " error=%s", -                  bus_message_type_to_string(m->header->type), -                  strna(sd_bus_message_get_sender(m)), -                  strna(sd_bus_message_get_destination(m)), -                  strna(sd_bus_message_get_path(m)), -                  strna(sd_bus_message_get_interface(m)), -                  strna(sd_bus_message_get_member(m)), -                  BUS_MESSAGE_COOKIE(m), -                  m->reply_cookie, -                  strna(m->error.message)); +        log_debug_bus_message(m);          r = process_hello(bus, m);          if (r != 0) diff --git a/src/libsystemd/sd-bus/test-bus-gvariant.c b/src/libsystemd/sd-bus/test-bus-gvariant.c index 992edacb28..22ea00c2fb 100644 --- a/src/libsystemd/sd-bus/test-bus-gvariant.c +++ b/src/libsystemd/sd-bus/test-bus-gvariant.c @@ -198,7 +198,7 @@ static void test_marshal(void) {          }  #endif -        assert_se(bus_message_from_malloc(bus, blob, sz, NULL, 0, NULL, NULL, &n) >= 0); +        assert_se(bus_message_from_malloc(bus, blob, sz, NULL, 0, NULL, &n) >= 0);          blob = NULL;          assert_se(bus_message_dump(n, NULL, BUS_MESSAGE_DUMP_WITH_HEADER) >= 0); diff --git a/src/libsystemd/sd-bus/test-bus-kernel-bloom.c b/src/libsystemd/sd-bus/test-bus-kernel-bloom.c index b11c43bd7b..90eb1f2a33 100644 --- a/src/libsystemd/sd-bus/test-bus-kernel-bloom.c +++ b/src/libsystemd/sd-bus/test-bus-kernel-bloom.c @@ -26,6 +26,14 @@  #include "bus-kernel.h"  #include "bus-util.h" +static int test_match(sd_bus_message *m, void *userdata, sd_bus_error *ret_error) { +        int *found = userdata; + +        *found = 1; + +        return 0; +} +  static void test_one(                  const char *path,                  const char *interface, @@ -39,7 +47,7 @@ static void test_one(          _cleanup_free_ char *name = NULL, *bus_name = NULL, *address = NULL;          _cleanup_bus_message_unref_ sd_bus_message *m = NULL;          sd_bus *a, *b; -        int r; +        int r, found = 0;          assert_se(asprintf(&name, "deine-mutter-%u", (unsigned) getpid()) >= 0); @@ -71,7 +79,7 @@ static void test_one(          assert_se(r >= 0);          log_debug("match"); -        r = sd_bus_add_match(b, NULL, match, NULL, NULL); +        r = sd_bus_add_match(b, NULL, match, test_match, &found);          assert_se(r >= 0);          log_debug("signal"); @@ -83,7 +91,7 @@ static void test_one(          assert_se(r >= 0);          r = sd_bus_process(b, &m); -        assert_se(r >= 0 && (good == !!m)); +        assert_se(r >= 0 && good == !!found);          sd_bus_unref(a);          sd_bus_unref(b); @@ -115,6 +123,17 @@ int main(int argc, char *argv[]) {          test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path_namespace='/foo'", true);          test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path_namespace='/'", true);          test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path_namespace='/quux'", false); +        test_one("/", "waldo.com", "Piep", false, "foobar", "path_namespace='/'", true); + +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path='/foo/bar/waldo/'", false); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path='/foo/'", false); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path_namespace='/foo/bar/waldo/'", false); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "foobar", "path_namespace='/foo/'", true); + +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "/foo/bar/waldo", "arg0path='/foo/'", true); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "/foo", "arg0path='/foo'", true); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "/foo", "arg0path='/foo/bar/waldo'", false); +        test_one("/foo/bar/waldo", "waldo.com", "Piep", false, "/foo/", "arg0path='/foo/bar/waldo'", true);          return 0;  } diff --git a/src/libsystemd/sd-bus/test-bus-marshal.c b/src/libsystemd/sd-bus/test-bus-marshal.c index f8ecadf499..a866a56179 100644 --- a/src/libsystemd/sd-bus/test-bus-marshal.c +++ b/src/libsystemd/sd-bus/test-bus-marshal.c @@ -212,7 +212,7 @@ int main(int argc, char *argv[]) {          m = sd_bus_message_unref(m); -        r = bus_message_from_malloc(bus, buffer, sz, NULL, 0, NULL, NULL, &m); +        r = bus_message_from_malloc(bus, buffer, sz, NULL, 0, NULL, &m);          assert_se(r >= 0);          bus_message_dump(m, stdout, BUS_MESSAGE_DUMP_WITH_HEADER); diff --git a/src/libsystemd/sd-bus/test-bus-signature.c b/src/libsystemd/sd-bus/test-bus-signature.c index 4165c9273a..17c6188ca0 100644 --- a/src/libsystemd/sd-bus/test-bus-signature.c +++ b/src/libsystemd/sd-bus/test-bus-signature.c @@ -95,23 +95,28 @@ int main(int argc, char *argv[]) {          assert_se(!namespace_complex_pattern("foo.", ""));          assert_se(path_complex_pattern("", "")); -        assert_se(path_complex_pattern("", "/")); -        assert_se(path_complex_pattern("/", "")); +        assert_se(!path_complex_pattern("", "/")); +        assert_se(!path_complex_pattern("/", ""));          assert_se(path_complex_pattern("/", "/"));          assert_se(path_complex_pattern("/foobar/", "/")); -        assert_se(path_complex_pattern("/foobar/", "/foobar")); +        assert_se(!path_complex_pattern("/foobar/", "/foobar"));          assert_se(path_complex_pattern("/foobar", "/foobar")); -        assert_se(path_complex_pattern("/foobar", "/foobar/")); +        assert_se(!path_complex_pattern("/foobar", "/foobar/"));          assert_se(!path_complex_pattern("/foobar", "/foobar/waldo"));          assert_se(path_complex_pattern("/foobar/", "/foobar/waldo")); +        assert_se(path_complex_pattern("/foobar/waldo", "/foobar/")); + +        assert_se(path_simple_pattern("/foo/", "/foo/bar/waldo"));          assert_se(namespace_simple_pattern("", "")); +        assert_se(namespace_simple_pattern("", ".foobar"));          assert_se(namespace_simple_pattern("foobar", "foobar"));          assert_se(namespace_simple_pattern("foobar.waldo", "foobar.waldo"));          assert_se(namespace_simple_pattern("foobar", "foobar.waldo"));          assert_se(!namespace_simple_pattern("foobar.waldo", "foobar"));          assert_se(!namespace_simple_pattern("", "foo"));          assert_se(!namespace_simple_pattern("foo", "")); +        assert_se(namespace_simple_pattern("foo.", "foo.bar.waldo"));          assert_se(streq(object_path_startswith("/foo/bar", "/foo"), "bar"));          assert_se(streq(object_path_startswith("/foo", "/foo"), "")); diff --git a/src/libsystemd/sd-network/sd-network.c b/src/libsystemd/sd-network/sd-network.c index db1f6997cb..207eda163b 100644 --- a/src/libsystemd/sd-network/sd-network.c +++ b/src/libsystemd/sd-network/sd-network.c @@ -297,8 +297,31 @@ static inline sd_network_monitor* FD_TO_MONITOR(int fd) {          return (sd_network_monitor*) (unsigned long) (fd + 1);  } +static int monitor_add_inotify_watch(int fd) { +        int k; + +        k = inotify_add_watch(fd, "/run/systemd/netif/links/", IN_MOVED_TO|IN_DELETE); +        if (k >= 0) +                return 0; +        else if (errno != ENOENT) +                return -errno; + +        k = inotify_add_watch(fd, "/run/systemd/netif/", IN_CREATE|IN_ISDIR); +        if (k >= 0) +                return 0; +        else if (errno != ENOENT) +                return -errno; + +        k = inotify_add_watch(fd, "/run/systemd/", IN_CREATE|IN_ISDIR); +        if (k < 0) +                return -errno; + +        return 0; +} +  _public_ int sd_network_monitor_new(sd_network_monitor **m, const char *category) { -        int fd, k; +        _cleanup_close_ int fd = -1; +        int k;          bool good = false;          assert_return(m, -EINVAL); @@ -308,11 +331,9 @@ _public_ int sd_network_monitor_new(sd_network_monitor **m, const char *category                  return -errno;          if (!category || streq(category, "links")) { -                k = inotify_add_watch(fd, "/run/systemd/netif/links/", IN_MOVED_TO|IN_DELETE); -                if (k < 0) { -                        safe_close(fd); -                        return -errno; -                } +                k = monitor_add_inotify_watch(fd); +                if (k < 0) +                        return k;                  good = true;          } @@ -323,25 +344,53 @@ _public_ int sd_network_monitor_new(sd_network_monitor **m, const char *category          }          *m = FD_TO_MONITOR(fd); +        fd = -1; +          return 0;  }  _public_ sd_network_monitor* sd_network_monitor_unref(sd_network_monitor *m) {          int fd; -        assert_return(m, NULL); - -        fd = MONITOR_TO_FD(m); -        close_nointr(fd); +        if (m) { +                fd = MONITOR_TO_FD(m); +                close_nointr(fd); +        }          return NULL;  }  _public_ int sd_network_monitor_flush(sd_network_monitor *m) { +        union inotify_event_buffer buffer; +        struct inotify_event *e; +        ssize_t l; +        int fd, k;          assert_return(m, -EINVAL); -        return flush_fd(MONITOR_TO_FD(m)); +        fd = MONITOR_TO_FD(m); + +        l = read(fd, &buffer, sizeof(buffer)); +        if (l < 0) { +                if (errno == EAGAIN || errno == EINTR) +                        return 0; + +                return -errno; +        } + +        FOREACH_INOTIFY_EVENT(e, buffer, l) { +                if (e->mask & IN_ISDIR) { +                        k = monitor_add_inotify_watch(fd); +                        if (k < 0) +                                return k; + +                        k = inotify_rm_watch(fd, e->wd); +                        if (k < 0) +                                return -errno; +                } +        } + +        return 0;  }  _public_ int sd_network_monitor_get_fd(sd_network_monitor *m) { diff --git a/src/libsystemd/sd-rtnl/rtnl-message.c b/src/libsystemd/sd-rtnl/rtnl-message.c index bab2a4ff08..9dcf7df559 100644 --- a/src/libsystemd/sd-rtnl/rtnl-message.c +++ b/src/libsystemd/sd-rtnl/rtnl-message.c @@ -1442,7 +1442,7 @@ static int socket_recv_message(int fd, struct iovec *iov, uint32_t *_group, bool                  return (errno == EAGAIN || errno == EINTR) ? 0 : -errno;          } -        for (cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg)) { +        CMSG_FOREACH(cmsg, &msg) {                  if (cmsg->cmsg_level == SOL_SOCKET &&                      cmsg->cmsg_type == SCM_CREDENTIALS &&                      cmsg->cmsg_len == CMSG_LEN(sizeof(struct ucred))) { diff --git a/src/network/networkd-link.c b/src/network/networkd-link.c index 127bc1249a..7841f69dab 100644 --- a/src/network/networkd-link.c +++ b/src/network/networkd-link.c @@ -749,7 +749,6 @@ static int set_hostname_handler(sd_bus_message *m, void *userdata, sd_bus_error  }  int link_set_hostname(Link *link, const char *hostname) { -        _cleanup_bus_message_unref_ sd_bus_message *m = NULL;          int r = 0;          assert(link); @@ -764,22 +763,19 @@ int link_set_hostname(Link *link, const char *hostname) {                  return 0;          } -        r = sd_bus_message_new_method_call( +        r = sd_bus_call_method_async(                          link->manager->bus, -                        &m, +                        NULL,                          "org.freedesktop.hostname1",                          "/org/freedesktop/hostname1",                          "org.freedesktop.hostname1", -                        "SetHostname"); -        if (r < 0) -                return r; - -        r = sd_bus_message_append(m, "sb", hostname, false); -        if (r < 0) -                return r; +                        "SetHostname", +                        set_hostname_handler, +                        link, +                        "sb", +                        hostname, +                        false); -        r = sd_bus_call_async(link->manager->bus, NULL, m, set_hostname_handler, -                              link, 0);          if (r < 0)                  return log_link_error_errno(link, r, "Could not set transient hostname: %m"); diff --git a/src/network/networkd-netdev-bond.c b/src/network/networkd-netdev-bond.c index 70df08a5e1..9919955f51 100644 --- a/src/network/networkd-netdev-bond.c +++ b/src/network/networkd-netdev-bond.c @@ -191,215 +191,132 @@ static int netdev_bond_fill_message_create(NetDev *netdev, Link *link, sd_rtnl_m          if (b->mode != _NETDEV_BOND_MODE_INVALID) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_MODE,                                                bond_mode_to_kernel(b->mode)); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_MODE attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_MODE attribute: %m");          }          if (b->xmit_hash_policy != _NETDEV_BOND_XMIT_HASH_POLICY_INVALID) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_XMIT_HASH_POLICY,                                                bond_xmit_hash_policy_to_kernel(b->xmit_hash_policy)); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_XMIT_HASH_POLICY attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_XMIT_HASH_POLICY attribute: %m");          }          if (b->lacp_rate != _NETDEV_BOND_LACP_RATE_INVALID &&              b->mode == NETDEV_BOND_MODE_802_3AD) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_AD_LACP_RATE, b->lacp_rate );                  if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_AD_LACP_RATE attribute: %s", -                                         strerror(-r)); -                        return r; +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_AD_LACP_RATE attribute: %m");                  }          }          if (b->miimon != 0) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_MIIMON, b->miimon / USEC_PER_MSEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_BOND_MIIMON attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_BOND_MIIMON attribute: %m");          }          if (b->downdelay != 0) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_DOWNDELAY, b->downdelay / USEC_PER_MSEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_DOWNDELAY attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_DOWNDELAY attribute: %m");          }          if (b->updelay != 0) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_UPDELAY, b->updelay / USEC_PER_MSEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_UPDELAY attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_UPDELAY attribute: %m");          }          if (b->arp_interval != 0) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_ARP_INTERVAL, b->arp_interval / USEC_PER_MSEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_ARP_INTERVAL attribute: %s", -                                         strerror(-r)); -                        return r; -                } -        } - -        if ((b->lp_interval >= LEARNING_PACKETS_INTERVAL_MIN_SEC) && -            (b->lp_interval <= LEARNING_PACKETS_INTERVAL_MAX_SEC)) { -                r = sd_rtnl_message_append_u32(m, IFLA_BOND_LP_INTERVAL, b->lp_interval / USEC_PER_SEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_LP_INTERVAL attribute: %s", -                                         strerror(-r)); -                        return r; +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ARP_INTERVAL attribute: %m"); + +                if ((b->lp_interval >= LEARNING_PACKETS_INTERVAL_MIN_SEC) && +                    (b->lp_interval <= LEARNING_PACKETS_INTERVAL_MAX_SEC)) { +                        r = sd_rtnl_message_append_u32(m, IFLA_BOND_LP_INTERVAL, b->lp_interval / USEC_PER_SEC); +                        if (r < 0) +                                return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_LP_INTERVAL attribute: %m");                  }          }          if (b->ad_select != _NETDEV_BOND_AD_SELECT_INVALID &&              b->mode == BOND_MODE_8023AD) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_AD_SELECT, b->ad_select); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_AD_SELECT attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_AD_SELECT attribute: %m");          }          if (b->fail_over_mac != _NETDEV_BOND_FAIL_OVER_MAC_INVALID &&              b->mode == NETDEV_BOND_MODE_ACTIVE_BACKUP) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_FAIL_OVER_MAC, b->fail_over_mac); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_FAIL_OVER_MAC attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_FAIL_OVER_MAC attribute: %m");          }          if (b->arp_validate != _NETDEV_BOND_ARP_VALIDATE_INVALID) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_ARP_VALIDATE, b->arp_validate); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_ARP_VALIDATE attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ARP_VALIDATE attribute: %m");          }          if (b->arp_all_targets != _NETDEV_BOND_ARP_ALL_TARGETS_INVALID) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_ARP_ALL_TARGETS, b->arp_all_targets); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_ARP_VALIDATE attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ARP_VALIDATE attribute: %m");          }          if (b->primary_reselect != _NETDEV_BOND_PRIMARY_RESELECT_INVALID) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_ARP_ALL_TARGETS, b->primary_reselect); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_ARP_ALL_TARGETS attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ARP_ALL_TARGETS attribute: %m");          }          if (b->resend_igmp <= RESEND_IGMP_MAX) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_RESEND_IGMP, b->resend_igmp); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_RESEND_IGMP attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_RESEND_IGMP attribute: %m");          }          if (b->packets_per_slave <= PACKETS_PER_SLAVE_MAX) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_PACKETS_PER_SLAVE, b->packets_per_slave); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_PACKETS_PER_SLAVE attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_PACKETS_PER_SLAVE attribute: %m");          }          if (b->num_grat_arp <= GRATUITOUS_ARP_MAX) {                  r = sd_rtnl_message_append_u8(m, IFLA_BOND_NUM_PEER_NOTIF, b->num_grat_arp); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_NUM_PEER_NOTIF attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_NUM_PEER_NOTIF attribute: %m");          }          if (b->min_links != 0) {                  r = sd_rtnl_message_append_u32(m, IFLA_BOND_MIN_LINKS, b->min_links); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_BOND_MIN_LINKS attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_MIN_LINKS attribute: %m");          }          r = sd_rtnl_message_append_u8(m, IFLA_BOND_ALL_SLAVES_ACTIVE, b->all_slaves_active); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_BOND_ALL_SLAVES_ACTIVE attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ALL_SLAVES_ACTIVE attribute: %m");          if (b->arp_interval > 0)  {                  if (b->n_arp_ip_targets > 0) {                          r = sd_rtnl_message_open_container(m, IFLA_BOND_ARP_IP_TARGET); -                        if (r < 0) { -                                log_netdev_error(netdev, -                                                 "Could not open contaniner IFLA_BOND_ARP_IP_TARGET : %s", -                                                 strerror(-r)); -                                return r; -                        } +                        if (r < 0) +                                return log_netdev_error_errno(netdev, r, "Could not open contaniner IFLA_BOND_ARP_IP_TARGET : %m");                          LIST_FOREACH(arp_ip_target, target, b->arp_ip_targets) {                                  r = sd_rtnl_message_append_u32(m, i++, target->ip.in.s_addr); -                                if (r < 0) { -                                        log_netdev_error(netdev, -                                                         "Could not append IFLA_BOND_ARP_ALL_TARGETS attribute: %s", -                                                         strerror(-r)); -                                        return r; -                                } +                                if (r < 0) +                                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_BOND_ARP_ALL_TARGETS attribute: %m");                          }                          r = sd_rtnl_message_close_container(m); -                        if (r < 0) { -                                log_netdev_error(netdev, -                                                 "Could not close contaniner IFLA_BOND_ARP_IP_TARGET : %s", -                                                 strerror(-r)); -                                return r; -                        } +                        if (r < 0) +                                return log_netdev_error_errno(netdev, r, "Could not close contaniner IFLA_BOND_ARP_IP_TARGET : %m");                  }          } diff --git a/src/network/networkd-netdev-veth.c b/src/network/networkd-netdev-veth.c index 9e9e1225e7..7bb02d1ae1 100644 --- a/src/network/networkd-netdev-veth.c +++ b/src/network/networkd-netdev-veth.c @@ -35,12 +35,8 @@ static int netdev_veth_fill_message_create(NetDev *netdev, Link *link, sd_rtnl_m          assert(m);          r = sd_rtnl_message_open_container(m, VETH_INFO_PEER); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append VETH_INFO_PEER attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append VETH_INFO_PEER attribute: %m");          if (v->ifname_peer) {                  r = sd_rtnl_message_append_string(m, IFLA_IFNAME, v->ifname_peer); @@ -50,21 +46,13 @@ static int netdev_veth_fill_message_create(NetDev *netdev, Link *link, sd_rtnl_m          if (v->mac_peer) {                  r = sd_rtnl_message_append_ether_addr(m, IFLA_ADDRESS, v->mac_peer); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_ADDRESS attribute: %s", -                                         strerror(-r)); -                    return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_ADDRESS attribute: %m");          }          r = sd_rtnl_message_close_container(m); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_INFO_DATA attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_INFO_DATA attribute: %m");          return r;  } diff --git a/src/network/networkd-netdev-vxlan.c b/src/network/networkd-netdev-vxlan.c index e2c2b108b9..01a1e5089c 100644 --- a/src/network/networkd-netdev-vxlan.c +++ b/src/network/networkd-netdev-vxlan.c @@ -39,123 +39,67 @@ static int netdev_vxlan_fill_message_create(NetDev *netdev, Link *link, sd_rtnl_          if (v->id <= VXLAN_VID_MAX) {                  r = sd_rtnl_message_append_u32(m, IFLA_VXLAN_ID, v->id); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_VXLAN_ID attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_ID attribute: %m");          }          r = sd_rtnl_message_append_in_addr(m, IFLA_VXLAN_GROUP, &v->group.in); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_GROUP attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_GROUP attribute: %m");          r = sd_rtnl_message_append_u32(m, IFLA_VXLAN_LINK, link->ifindex); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_LINK attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_LINK attribute: %m");          if(v->ttl) {                  r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_TTL, v->ttl); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_VXLAN_TTL attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_TTL attribute: %m");          }          if(v->tos) {                  r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_TOS, v->tos); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_VXLAN_TOS attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_TOS attribute: %m");          }          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_LEARNING, v->learning); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_LEARNING attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_LEARNING attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_RSC, v->route_short_circuit); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_RSC attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_RSC attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_PROXY, v->arp_proxy); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_PROXY attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_PROXY attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_L2MISS, v->l2miss); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_L2MISS attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_L2MISS attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_L3MISS, v->l3miss); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_L3MISS attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_L3MISS attribute: %m");          if(v->fdb_ageing) {                  r = sd_rtnl_message_append_u32(m, IFLA_VXLAN_AGEING, v->fdb_ageing / USEC_PER_SEC); -                if (r < 0) { -                        log_netdev_error(netdev, -                                         "Could not append IFLA_VXLAN_AGEING attribute: %s", -                                         strerror(-r)); -                        return r; -                } +                if (r < 0) +                        return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_AGEING attribute: %m");          }          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_UDP_CSUM, v->udpcsum); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_UDP_CSUM attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_UDP_CSUM attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_UDP_ZERO_CSUM6_TX, v->udp6zerocsumtx); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_TX attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_TX attribute: %m");          r = sd_rtnl_message_append_u8(m, IFLA_VXLAN_UDP_ZERO_CSUM6_RX, v->udp6zerocsumrx); -        if (r < 0) { -                log_netdev_error(netdev, -                                 "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_RX attribute: %s", -                                 strerror(-r)); -                return r; -        } +        if (r < 0) +                return log_netdev_error_errno(netdev, r, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_RX attribute: %m");          return r;  } diff --git a/src/network/networkd-network.c b/src/network/networkd-network.c index 5947084106..6f32e5f4a4 100644 --- a/src/network/networkd-network.c +++ b/src/network/networkd-network.c @@ -30,6 +30,7 @@  #include "networkd-netdev.h"  #include "networkd-link.h"  #include "network-internal.h" +#include "dns-domain.h"  static int network_load_one(Manager *manager, const char *filename) {          _cleanup_network_free_ Network *network = NULL; @@ -466,11 +467,16 @@ int config_parse_domains(const char *unit,          STRV_FOREACH(domain, *domains) {                  if (is_localhost(*domain))                          log_syntax(unit, LOG_ERR, filename, line, EINVAL, "'localhost' domain names may not be configured, ignoring assignment: %s", *domain); -                else if (!hostname_is_valid(*domain)) { -                        if (!streq(*domain, "*")) -                                log_syntax(unit, LOG_ERR, filename, line, EINVAL, "domain name is not valid, ignoring assignment: %s", *domain); -                } else -                        continue; +                else { +                        r = dns_name_is_valid(*domain); +                        if (r <= 0 && !streq(*domain, "*")) { +                                if (r < 0) +                                        log_error_errno(r, "Failed to validate domain name: %s: %m", *domain); +                                if (r == 0) +                                        log_warning("Domain name is not valid, ignoring assignment: %s", *domain); +                        } else +                                continue; +                }                  strv_remove(*domains, *domain); diff --git a/src/resolve/resolved-bus.c b/src/resolve/resolved-bus.c index 171141e3a4..6db12511f9 100644 --- a/src/resolve/resolved-bus.c +++ b/src/resolve/resolved-bus.c @@ -22,7 +22,7 @@  #include "bus-common-errors.h"  #include "bus-util.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-bus.h"  #include "resolved-def.h" diff --git a/src/resolve/resolved-dns-answer.c b/src/resolve/resolved-dns-answer.c index e08eb667cc..f77b98e505 100644 --- a/src/resolve/resolved-dns-answer.c +++ b/src/resolve/resolved-dns-answer.c @@ -20,7 +20,7 @@  ***/  #include "resolved-dns-answer.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  DnsAnswer *dns_answer_new(unsigned n) {          DnsAnswer *a; diff --git a/src/resolve/resolved-dns-packet.c b/src/resolve/resolved-dns-packet.c index 21756f566f..bb74b1828e 100644 --- a/src/resolve/resolved-dns-packet.c +++ b/src/resolve/resolved-dns-packet.c @@ -23,7 +23,7 @@  #include "util.h"  #include "strv.h"  #include "unaligned.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-dns-packet.h"  int dns_packet_new(DnsPacket **ret, DnsProtocol protocol, size_t mtu) { diff --git a/src/resolve/resolved-dns-question.c b/src/resolve/resolved-dns-question.c index 45bcbbf23a..4d71f5e3d4 100644 --- a/src/resolve/resolved-dns-question.c +++ b/src/resolve/resolved-dns-question.c @@ -20,7 +20,7 @@  ***/  #include "resolved-dns-question.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  DnsQuestion *dns_question_new(unsigned n) {          DnsQuestion *q; diff --git a/src/resolve/resolved-dns-rr.c b/src/resolve/resolved-dns-rr.c index 78d9e4a412..c1818eef9c 100644 --- a/src/resolve/resolved-dns-rr.c +++ b/src/resolve/resolved-dns-rr.c @@ -23,7 +23,7 @@  #include "strv.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-dns-rr.h"  #include "resolved-dns-packet.h"  #include "dns-type.h" diff --git a/src/resolve/resolved-dns-scope.c b/src/resolve/resolved-dns-scope.c index 7369cbf50f..c25ac2216d 100644 --- a/src/resolve/resolved-dns-scope.c +++ b/src/resolve/resolved-dns-scope.c @@ -27,7 +27,7 @@  #include "af-list.h"  #include "random-util.h"  #include "hostname-util.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-dns-scope.h"  #define MULTICAST_RATELIMIT_INTERVAL_USEC (1*USEC_PER_SEC) diff --git a/src/resolve/resolved-dns-stream.c b/src/resolve/resolved-dns-stream.c index 4c0b557bad..7f47e7223a 100644 --- a/src/resolve/resolved-dns-stream.c +++ b/src/resolve/resolved-dns-stream.c @@ -113,7 +113,8 @@ static int dns_stream_identify(DnsStream *s) {          mh.msg_control = &control;          mh.msg_controllen = sl; -        for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) { + +        CMSG_FOREACH(cmsg, &mh) {                  if (cmsg->cmsg_level == IPPROTO_IPV6) {                          assert(s->peer.sa.sa_family == AF_INET6); diff --git a/src/resolve/resolved-dns-zone.c b/src/resolve/resolved-dns-zone.c index a4c9b7d7af..32d771a954 100644 --- a/src/resolve/resolved-dns-zone.c +++ b/src/resolve/resolved-dns-zone.c @@ -22,7 +22,7 @@  #include "list.h"  #include "resolved-dns-zone.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-dns-packet.h"  /* Never allow more than 1K entries */ diff --git a/src/resolve/resolved-manager.c b/src/resolve/resolved-manager.c index 52695376f0..f8d4db7aad 100644 --- a/src/resolve/resolved-manager.c +++ b/src/resolve/resolved-manager.c @@ -34,7 +34,7 @@  #include "random-util.h"  #include "hostname-util.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  #include "resolved-conf.h"  #include "resolved-bus.h"  #include "resolved-manager.h" @@ -920,7 +920,7 @@ int manager_recv(Manager *m, int fd, DnsProtocol protocol, DnsPacket **ret) {          } else                  return -EAFNOSUPPORT; -        for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) { +        CMSG_FOREACH(cmsg, &mh) {                  if (cmsg->cmsg_level == IPPROTO_IPV6) {                          assert(p->family == AF_INET6); diff --git a/src/run/run.c b/src/run/run.c index 5b9f31c4aa..f18f77b55a 100644 --- a/src/run/run.c +++ b/src/run/run.c @@ -68,7 +68,7 @@ static void help(void) {          printf("%s [OPTIONS...] {COMMAND} [ARGS...]\n\n"                 "Run the specified command in a transient scope or service or timer\n"                 "unit. If timer option is specified and unit is exist which is\n" -               "specified with --unit option then command can be ommited.\n\n" +               "specified with --unit option then command can be omitted.\n\n"                 "  -h --help                       Show this help\n"                 "     --version                    Show package version\n"                 "     --user                       Run as user unit\n" diff --git a/src/shared/cgroup-util.c b/src/shared/cgroup-util.c index 9988e5c574..d83cdf7e5f 100644 --- a/src/shared/cgroup-util.c +++ b/src/shared/cgroup-util.c @@ -441,9 +441,7 @@ static const char *normalize_controller(const char *controller) {          assert(controller); -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER)) -                return "systemd"; -        else if (startswith(controller, "name=")) +        if (startswith(controller, "name="))                  return controller + 5;          else                  return controller; @@ -483,7 +481,7 @@ int cg_get_path(const char *controller, const char *path, const char *suffix, ch          assert(fs); -        if (controller && !cg_controller_is_valid(controller, true)) +        if (controller && !cg_controller_is_valid(controller))                  return -EINVAL;          if (_unlikely_(!good)) { @@ -526,7 +524,7 @@ int cg_get_path_and_check(const char *controller, const char *path, const char *          assert(fs); -        if (!cg_controller_is_valid(controller, true)) +        if (!cg_controller_is_valid(controller))                  return -EINVAL;          /* Normalize the controller syntax */ @@ -742,7 +740,7 @@ int cg_pid_get_path(const char *controller, pid_t pid, char **path) {          assert(pid >= 0);          if (controller) { -                if (!cg_controller_is_valid(controller, true)) +                if (!cg_controller_is_valid(controller))                          return -EINVAL;                  controller = normalize_controller(controller); @@ -971,7 +969,7 @@ int cg_split_spec(const char *spec, char **controller, char **path) {          e = strchr(spec, ':');          if (!e) { -                if (!cg_controller_is_valid(spec, true)) +                if (!cg_controller_is_valid(spec))                          return -EINVAL;                  if (controller) { @@ -994,7 +992,7 @@ int cg_split_spec(const char *spec, char **controller, char **path) {          t = strdup(normalize_controller(v));          if (!t)                  return -ENOMEM; -        if (!cg_controller_is_valid(t, true)) { +        if (!cg_controller_is_valid(t)) {                  free(t);                  return -EINVAL;          } @@ -1610,17 +1608,15 @@ char *cg_unescape(const char *p) {          DIGITS LETTERS                          \          "_" -bool cg_controller_is_valid(const char *p, bool allow_named) { +bool cg_controller_is_valid(const char *p) {          const char *t, *s;          if (!p)                  return false; -        if (allow_named) { -                s = startswith(p, "name="); -                if (s) -                        p = s; -        } +        s = startswith(p, "name="); +        if (s) +                p = s;          if (*p == 0 || *p == '_')                  return false; diff --git a/src/shared/cgroup-util.h b/src/shared/cgroup-util.h index cbf7201370..fd72e9e5c5 100644 --- a/src/shared/cgroup-util.h +++ b/src/shared/cgroup-util.h @@ -122,7 +122,7 @@ int cg_path_decode_unit(const char *cgroup, char **unit);  char *cg_escape(const char *p);  char *cg_unescape(const char *p) _pure_; -bool cg_controller_is_valid(const char *p, bool allow_named); +bool cg_controller_is_valid(const char *p);  int cg_slice_to_path(const char *unit, char **ret); diff --git a/src/shared/def.h b/src/shared/def.h index a3d9fcf388..011c7c667e 100644 --- a/src/shared/def.h +++ b/src/shared/def.h @@ -35,7 +35,7 @@   * the watchdog pings will keep the loop busy. */  #define DEFAULT_EXIT_USEC (30*USEC_PER_SEC) -#define SYSTEMD_CGROUP_CONTROLLER "name=systemd" +#define SYSTEMD_CGROUP_CONTROLLER "systemd"  #define SIGNALS_CRASH_HANDLER SIGSEGV,SIGILL,SIGFPE,SIGBUS,SIGQUIT,SIGABRT  #define SIGNALS_IGNORE SIGPIPE diff --git a/src/resolve/resolved-dns-domain.c b/src/shared/dns-domain.c index e1eb3ddfe5..20a44ce4e1 100644 --- a/src/resolve/resolved-dns-domain.c +++ b/src/shared/dns-domain.c @@ -24,7 +24,7 @@  #include <stringprep.h>  #endif -#include "resolved-dns-domain.h" +#include "dns-domain.h"  int dns_label_unescape(const char **name, char *dest, size_t sz) {          const char *n; diff --git a/src/resolve/resolved-dns-domain.h b/src/shared/dns-domain.h index 516d244f7a..00caf5d700 100644 --- a/src/resolve/resolved-dns-domain.h +++ b/src/shared/dns-domain.h @@ -35,6 +35,15 @@ int dns_label_apply_idna(const char *encoded, size_t encoded_size, char *decoded  int dns_label_undo_idna(const char *encoded, size_t encoded_size, char *decoded, size_t decoded_max);  int dns_name_normalize(const char *s, char **_ret); +static inline int dns_name_is_valid(const char *s) { +        int r; +        r = dns_name_normalize(s, NULL); +        if (r == -EINVAL) +                return 0; +        if (r < 0) +                return r; +        return 1; +}  unsigned long dns_name_hash_func(const void *s, const uint8_t hash_key[HASH_KEY_SIZE]);  int dns_name_compare_func(const void *a, const void *b); diff --git a/src/shared/macro.h b/src/shared/macro.h index 7ae1ed80b6..cc1c9e73c0 100644 --- a/src/shared/macro.h +++ b/src/shared/macro.h @@ -467,4 +467,7 @@ do {                                                                    \          }                                                       \          struct __useless_struct_to_allow_trailing_semicolon__ +#define CMSG_FOREACH(cmsg, mh)                                          \ +        for ((cmsg) = CMSG_FIRSTHDR(mh); (cmsg); (cmsg) = CMSG_NXTHDR((mh), (cmsg))) +  #include "log.h" diff --git a/src/shared/util.c b/src/shared/util.c index a20e7bb2ef..6f6906f877 100644 --- a/src/shared/util.c +++ b/src/shared/util.c @@ -5520,7 +5520,7 @@ int openpt_in_namespace(pid_t pid, int flags) {          if (recvmsg(pair[0], &mh, MSG_NOSIGNAL|MSG_CMSG_CLOEXEC) < 0)                  return -errno; -        for (cmsg = CMSG_FIRSTHDR(&mh); cmsg; cmsg = CMSG_NXTHDR(&mh, cmsg)) +        CMSG_FOREACH(cmsg, &mh)                  if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS) {                          int *fds;                          unsigned n_fds; @@ -5908,7 +5908,7 @@ void cmsg_close_all(struct msghdr *mh) {          assert(mh); -        for (cmsg = CMSG_FIRSTHDR(mh); cmsg; cmsg = CMSG_NXTHDR(mh, cmsg)) +        CMSG_FOREACH(cmsg, mh)                  if (cmsg->cmsg_level == SOL_SOCKET && cmsg->cmsg_type == SCM_RIGHTS)                          close_many((int*) CMSG_DATA(cmsg), (cmsg->cmsg_len - CMSG_LEN(0)) / sizeof(int));  } diff --git a/src/test/test-cgroup-util.c b/src/test/test-cgroup-util.c index 4a89f64518..ecc9d70bf4 100644 --- a/src/test/test-cgroup-util.c +++ b/src/test/test-cgroup-util.c @@ -244,16 +244,16 @@ static void test_escape(void) {  }  static void test_controller_is_valid(void) { -        assert_se(cg_controller_is_valid("foobar", false)); -        assert_se(cg_controller_is_valid("foo_bar", false)); -        assert_se(cg_controller_is_valid("name=foo", true)); -        assert_se(!cg_controller_is_valid("", false)); -        assert_se(!cg_controller_is_valid("name=", true)); -        assert_se(!cg_controller_is_valid("=", false)); -        assert_se(!cg_controller_is_valid("cpu,cpuacct", false)); -        assert_se(!cg_controller_is_valid("_", false)); -        assert_se(!cg_controller_is_valid("_foobar", false)); -        assert_se(!cg_controller_is_valid("tatü", false)); +        assert_se(cg_controller_is_valid("foobar")); +        assert_se(cg_controller_is_valid("foo_bar")); +        assert_se(cg_controller_is_valid("name=foo")); +        assert_se(!cg_controller_is_valid("")); +        assert_se(!cg_controller_is_valid("name=")); +        assert_se(!cg_controller_is_valid("=")); +        assert_se(!cg_controller_is_valid("cpu,cpuacct")); +        assert_se(!cg_controller_is_valid("_")); +        assert_se(!cg_controller_is_valid("_foobar")); +        assert_se(!cg_controller_is_valid("tatü"));  }  static void test_slice_to_path_one(const char *unit, const char *path, int error) { diff --git a/src/test/test-copy.c b/src/test/test-copy.c index 403d85bff0..e55ffaa16a 100644 --- a/src/test/test-copy.c +++ b/src/test/test-copy.c @@ -133,10 +133,45 @@ static void test_copy_tree(void) {          (void) rm_rf(original_dir, REMOVE_ROOT|REMOVE_PHYSICAL);  } +static void test_copy_bytes(void) { +        _cleanup_close_pair_ int pipefd[2] = {-1, -1}; +        _cleanup_close_ int infd = -1; +        int r, r2; +        char buf[1024], buf2[1024]; + +        infd = open("/etc/os-release", O_RDONLY|O_CLOEXEC); +        assert_se(infd >= 0); + +        assert_se(pipe2(pipefd, O_CLOEXEC) == 0); + +        r = copy_bytes(infd, pipefd[1], (off_t) -1, false); +        assert_se(r == 0); + +        r = read(pipefd[0], buf, sizeof(buf)); +        assert_se(r >= 0); + +        assert_se(lseek(infd, 0, SEEK_SET) == 0); +        r2 = read(infd, buf2, sizeof(buf2)); +        assert_se(r == r2); + +        assert_se(strneq(buf, buf2, r)); + +        /* test copy_bytes with invalid descriptors */ +        r = copy_bytes(pipefd[0], pipefd[0], 1, false); +        assert_se(r == -EBADF); + +        r = copy_bytes(pipefd[1], pipefd[1], 1, false); +        assert_se(r == -EBADF); + +        r = copy_bytes(pipefd[1], infd, 1, false); +        assert_se(r == -EBADF); +} +  int main(int argc, char *argv[]) {          test_copy_file();          test_copy_file_fd();          test_copy_tree(); +        test_copy_bytes();          return 0;  } diff --git a/src/resolve/test-dns-domain.c b/src/test/test-dns-domain.c index c3208abc78..527cdd3b54 100644 --- a/src/resolve/test-dns-domain.c +++ b/src/test/test-dns-domain.c @@ -20,7 +20,7 @@   ***/  #include "macro.h" -#include "resolved-dns-domain.h" +#include "dns-domain.h"  static void test_dns_label_unescape_one(const char *what, const char *expect, size_t buffer_sz, int ret) {          char buffer[buffer_sz]; diff --git a/src/timesync/timesyncd-manager.c b/src/timesync/timesyncd-manager.c index 88e9cf98ed..40e0fd31fe 100644 --- a/src/timesync/timesyncd-manager.c +++ b/src/timesync/timesyncd-manager.c @@ -528,7 +528,7 @@ static int manager_receive_response(sd_event_source *source, int fd, uint32_t re          }          recv_time = NULL; -        for (cmsg = CMSG_FIRSTHDR(&msghdr); cmsg; cmsg = CMSG_NXTHDR(&msghdr, cmsg)) { +        CMSG_FOREACH(cmsg, &msghdr) {                  if (cmsg->cmsg_level != SOL_SOCKET)                          continue; diff --git a/src/udev/udevd.c b/src/udev/udevd.c index 26aae89990..2affb5944a 100644 --- a/src/udev/udevd.c +++ b/src/udev/udevd.c @@ -875,7 +875,7 @@ static int on_worker(sd_event_source *s, int fd, uint32_t revents, void *userdat                          continue;                  } -                for (cmsg = CMSG_FIRSTHDR(&msghdr); cmsg; cmsg = CMSG_NXTHDR(&msghdr, cmsg)) { +                CMSG_FOREACH(cmsg, &msghdr) {                          if (cmsg->cmsg_level == SOL_SOCKET &&                              cmsg->cmsg_type == SCM_CREDENTIALS &&                              cmsg->cmsg_len == CMSG_LEN(sizeof(struct ucred))) | 
