diff options
Diffstat (limited to 'src/libsystemd/sd-bus')
| -rw-r--r-- | src/libsystemd/sd-bus/GVARIANT-SERIALIZATION | 4 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-common-errors.h | 3 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-control.c | 45 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-kernel.c | 14 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-message.c | 17 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-objects.c | 100 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-slot.c | 2 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/bus-socket.c | 20 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/busctl.c | 3 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/kdbus.h | 1 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/sd-bus.c | 11 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-chat.c | 4 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-gvariant.c | 2 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-marshal.c | 21 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-match.c | 2 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-objects.c | 21 | ||||
| -rw-r--r-- | src/libsystemd/sd-bus/test-bus-proxy.c | 109 | 
17 files changed, 322 insertions, 57 deletions
| diff --git a/src/libsystemd/sd-bus/GVARIANT-SERIALIZATION b/src/libsystemd/sd-bus/GVARIANT-SERIALIZATION index 859e2715f9..6aeb11364a 100644 --- a/src/libsystemd/sd-bus/GVARIANT-SERIALIZATION +++ b/src/libsystemd/sd-bus/GVARIANT-SERIALIZATION @@ -25,8 +25,8 @@ The header consists of the following:      = 12 bytes -This header is then followed by the the fields array, whose first -value is a 32bit array size. +This header is then followed by the fields array, whose first value is +a 32bit array size.  When using GVariant we keep the basic structure in place, only  slightly alter the header, and define protocol version '2'. The new diff --git a/src/libsystemd/sd-bus/bus-common-errors.h b/src/libsystemd/sd-bus/bus-common-errors.h index b17b62ac93..f2092795f4 100644 --- a/src/libsystemd/sd-bus/bus-common-errors.h +++ b/src/libsystemd/sd-bus/bus-common-errors.h @@ -46,6 +46,8 @@  #define BUS_ERROR_NO_MACHINE_FOR_PID "org.freedesktop.machine1.NoMachineForPID"  #define BUS_ERROR_MACHINE_EXISTS "org.freedesktop.machine1.MachineExists"  #define BUS_ERROR_NO_PRIVATE_NETWORKING "org.freedesktop.machine1.NoPrivateNetworking" +#define BUS_ERROR_NO_SUCH_USER_MAPPING "org.freedesktop.machine1.NoSuchUserMapping" +#define BUS_ERROR_NO_SUCH_GROUP_MAPPING "org.freedesktop.machine1.NoSuchGroupMapping"  #define BUS_ERROR_NO_SUCH_SESSION "org.freedesktop.login1.NoSuchSession"  #define BUS_ERROR_NO_SESSION_FOR_PID "org.freedesktop.login1.NoSessionForPID" @@ -58,6 +60,7 @@  #define BUS_ERROR_DEVICE_NOT_TAKEN "org.freedesktop.login1.DeviceNotTaken"  #define BUS_ERROR_OPERATION_IN_PROGRESS "org.freedesktop.login1.OperationInProgress"  #define BUS_ERROR_SLEEP_VERB_NOT_SUPPORTED "org.freedesktop.login1.SleepVerbNotSupported" +#define BUS_ERROR_SESSION_BUSY "org.freedesktop.login1.SessionBusy"  #define BUS_ERROR_AUTOMATIC_TIME_SYNC_ENABLED "org.freedesktop.timedate1.AutomaticTimeSyncEnabled" diff --git a/src/libsystemd/sd-bus/bus-control.c b/src/libsystemd/sd-bus/bus-control.c index 7a59702cb2..c53666ddd0 100644 --- a/src/libsystemd/sd-bus/bus-control.c +++ b/src/libsystemd/sd-bus/bus-control.c @@ -1131,7 +1131,7 @@ static int add_name_change_match(sd_bus *bus,                  /* If the old name is unset or empty, then                   * this can match against added names */ -                if (!old_owner || old_owner[0] == 0) { +                if (isempty(old_owner)) {                          item->type = KDBUS_ITEM_NAME_ADD;                          r = ioctl(bus->input_fd, KDBUS_CMD_MATCH_ADD, m); @@ -1141,7 +1141,7 @@ static int add_name_change_match(sd_bus *bus,                  /* If the new name is unset or empty, then                   * this can match against removed names */ -                if (!new_owner || new_owner[0] == 0) { +                if (isempty(new_owner)) {                          item->type = KDBUS_ITEM_NAME_REMOVE;                          r = ioctl(bus->input_fd, KDBUS_CMD_MATCH_ADD, m); @@ -1185,8 +1185,10 @@ static int add_name_change_match(sd_bus *bus,                  /* If the old name is unset or empty, then this can                   * match against added ids */ -                if (!old_owner || old_owner[0] == 0) { +                if (isempty(old_owner)) {                          item->type = KDBUS_ITEM_ID_ADD; +                        if (!isempty(new_owner)) +                                item->id_change.id = new_owner_id;                          r = ioctl(bus->input_fd, KDBUS_CMD_MATCH_ADD, m);                          if (r < 0) @@ -1195,8 +1197,10 @@ static int add_name_change_match(sd_bus *bus,                  /* If thew new name is unset or empty, then this can                   * match against removed ids */ -                if (!new_owner || new_owner[0] == 0) { +                if (isempty(new_owner)) {                          item->type = KDBUS_ITEM_ID_REMOVE; +                        if (!isempty(old_owner)) +                                item->id_change.id = old_owner_id;                          r = ioctl(bus->input_fd, KDBUS_CMD_MATCH_ADD, m);                          if (r < 0) @@ -1219,7 +1223,7 @@ int bus_add_match_internal_kernel(          size_t sz;          const char *sender = NULL;          size_t sender_length = 0; -        uint64_t src_id = KDBUS_MATCH_ID_ANY; +        uint64_t src_id = KDBUS_MATCH_ID_ANY, dst_id = KDBUS_MATCH_ID_ANY;          bool using_bloom = false;          unsigned i;          bool matches_name_change = true; @@ -1332,13 +1336,25 @@ int bus_add_match_internal_kernel(                          break;                  } -                case BUS_MATCH_DESTINATION: -                        /* The bloom filter does not include -                           the destination, since it is only -                           available for broadcast messages -                           which do not carry a destination -                           since they are undirected. */ +                case BUS_MATCH_DESTINATION: { +                        /* +                         * Kernel only supports matching on destination IDs, but +                         * not on destination names. So just skip the +                         * destination name restriction and verify it in +                         * user-space on retrieval. +                         */ +                        r = bus_kernel_parse_unique_name(c->value_str, &dst_id); +                        if (r < 0) +                                return r; +                        else if (r > 0) +                                sz += ALIGN8(offsetof(struct kdbus_item, id) + sizeof(uint64_t)); + +                        /* if not a broadcast, it cannot be a name-change */ +                        if (r <= 0 || dst_id != KDBUS_DST_ID_BROADCAST) +                                matches_name_change = false; +                          break; +                }                  case BUS_MATCH_ROOT:                  case BUS_MATCH_VALUE: @@ -1365,6 +1381,13 @@ int bus_add_match_internal_kernel(                  item = KDBUS_ITEM_NEXT(item);          } +        if (dst_id != KDBUS_MATCH_ID_ANY) { +                item->size = offsetof(struct kdbus_item, id) + sizeof(uint64_t); +                item->type = KDBUS_ITEM_DST_ID; +                item->id = dst_id; +                item = KDBUS_ITEM_NEXT(item); +        } +          if (using_bloom) {                  item->size = offsetof(struct kdbus_item, data64) + bus->bloom_size;                  item->type = KDBUS_ITEM_BLOOM_MASK; diff --git a/src/libsystemd/sd-bus/bus-kernel.c b/src/libsystemd/sd-bus/bus-kernel.c index f08db2da89..6ac5ebc3da 100644 --- a/src/libsystemd/sd-bus/bus-kernel.c +++ b/src/libsystemd/sd-bus/bus-kernel.c @@ -1332,8 +1332,7 @@ static int bus_kernel_translate_message(sd_bus *bus, struct kdbus_msg *k) {          KDBUS_ITEM_FOREACH(d, k, items) {                  if (d->type == KDBUS_ITEM_TIMESTAMP)                          ts = &d->timestamp; - -                if (d->type >= _KDBUS_ITEM_KERNEL_BASE && d->type < _KDBUS_ITEM_KERNEL_BASE + ELEMENTSOF(translate)) { +                else if (d->type >= _KDBUS_ITEM_KERNEL_BASE && d->type < _KDBUS_ITEM_KERNEL_BASE + ELEMENTSOF(translate)) {                          if (found)                                  return -EBADMSG;                          found = d; @@ -1385,15 +1384,16 @@ int bus_kernel_read_message(sd_bus *bus, bool hint_priority, int64_t priority) {                          r = 0;                  } -        } else if (k->payload_type == KDBUS_PAYLOAD_KERNEL) +                if (r <= 0) +                        close_kdbus_msg(bus, k); +        } else if (k->payload_type == KDBUS_PAYLOAD_KERNEL) {                  r = bus_kernel_translate_message(bus, k); -        else { +                close_kdbus_msg(bus, k); +        } else {                  log_debug("Ignoring message with unknown payload type %llu.", (unsigned long long) k->payload_type);                  r = 0; -        } - -        if (r <= 0)                  close_kdbus_msg(bus, k); +        }          return r < 0 ? r : 1;  } diff --git a/src/libsystemd/sd-bus/bus-message.c b/src/libsystemd/sd-bus/bus-message.c index 983e2f62cd..18685be8ff 100644 --- a/src/libsystemd/sd-bus/bus-message.c +++ b/src/libsystemd/sd-bus/bus-message.c @@ -2161,6 +2161,7 @@ static int bus_message_close_variant(sd_bus_message *m, struct bus_container *c)  }  static int bus_message_close_struct(sd_bus_message *m, struct bus_container *c, bool add_offset) { +        bool fixed_size = true;          size_t n_variable = 0;          unsigned i = 0;          const char *p; @@ -2196,6 +2197,8 @@ static int bus_message_close_struct(sd_bus_message *m, struct bus_container *c,                  /* We need to add an offset for each item that has a                   * variable size and that is not the last one in the                   * list */ +                if (r == 0) +                        fixed_size = false;                  if (r == 0 && p[n] != 0)                          n_variable++; @@ -2207,7 +2210,19 @@ static int bus_message_close_struct(sd_bus_message *m, struct bus_container *c,          assert(c->need_offsets || n_variable == 0);          if (n_variable <= 0) { -                a = message_extend_body(m, 1, 0, add_offset, false); +                int alignment = 1; + +                /* Structures with fixed-size members only have to be +                 * fixed-size themselves. But gvariant requires all fixed-size +                 * elements to be sized a multiple of their alignment. Hence, +                 * we must *always* add final padding after the last member so +                 * the overall size of the structure is properly aligned. */ +                if (fixed_size) +                        alignment = bus_gvariant_get_alignment(strempty(c->signature)); + +                assert(alignment > 0); + +                a = message_extend_body(m, alignment, 0, add_offset, false);                  if (!a)                          return -ENOMEM;          } else { diff --git a/src/libsystemd/sd-bus/bus-objects.c b/src/libsystemd/sd-bus/bus-objects.c index e4bbd880e5..c25293e5e9 100644 --- a/src/libsystemd/sd-bus/bus-objects.c +++ b/src/libsystemd/sd-bus/bus-objects.c @@ -68,6 +68,12 @@ static int node_vtable_get_userdata(          return 1;  } +static void *vtable_method_convert_userdata(const sd_bus_vtable *p, void *u) { +        assert(p); + +        return (uint8_t*) u + p->x.method.offset; +} +  static void *vtable_property_convert_userdata(const sd_bus_vtable *p, void *u) {          assert(p); @@ -167,6 +173,7 @@ static int add_subtree_to_set(                  sd_bus *bus,                  const char *prefix,                  struct node *n, +                bool skip_subhierarchies,                  Set *s,                  sd_bus_error *error) { @@ -198,11 +205,13 @@ static int add_subtree_to_set(                  if (r < 0 && r != -EEXIST)                          return r; -                r = add_subtree_to_set(bus, prefix, i, s, error); -                if (r < 0) -                        return r; -                if (bus->nodes_modified) -                        return 0; +                if (!skip_subhierarchies || !i->object_managers) { +                        r = add_subtree_to_set(bus, prefix, i, skip_subhierarchies, s, error); +                        if (r < 0) +                                return r; +                        if (bus->nodes_modified) +                                return 0; +                }          }          return 0; @@ -212,6 +221,7 @@ static int get_child_nodes(                  sd_bus *bus,                  const char *prefix,                  struct node *n, +                bool skip_subhierarchies,                  Set **_s,                  sd_bus_error *error) { @@ -227,7 +237,7 @@ static int get_child_nodes(          if (!s)                  return -ENOMEM; -        r = add_subtree_to_set(bus, prefix, n, s, error); +        r = add_subtree_to_set(bus, prefix, n, skip_subhierarchies, s, error);          if (r < 0) {                  set_free_free(s);                  return r; @@ -360,6 +370,8 @@ static int method_callbacks_run(          if (bus->nodes_modified)                  return 0; +        u = vtable_method_convert_userdata(c->vtable, u); +          *found_object = true;          if (c->last_iteration == bus->iteration_counter) @@ -892,7 +904,7 @@ static int process_introspect(          assert(n);          assert(found_object); -        r = get_child_nodes(bus, m->path, n, &s, &error); +        r = get_child_nodes(bus, m->path, n, false, &s, &error);          if (r < 0)                  return bus_maybe_reply_error(m, r, &error);          if (bus->nodes_modified) @@ -1158,12 +1170,16 @@ static int process_get_managed_objects(          if (require_fallback || !n->object_managers)                  return 0; -        r = get_child_nodes(bus, m->path, n, &s, &error); +        r = get_child_nodes(bus, m->path, n, true, &s, &error);          if (r < 0)                  return r;          if (bus->nodes_modified)                  return 0; +        r = set_put_strdup(s, m->path); +        if (r < 0) +                return r; +          r = sd_bus_message_new_method_return(m, &reply);          if (r < 0)                  return r; @@ -1412,7 +1428,7 @@ static struct node *bus_node_allocate(sd_bus *bus, const char *path) {                  e = strrchr(path, '/');                  assert(e); -                p = strndupa(path, MAX(1, path - e)); +                p = strndupa(path, MAX(1, e - path));                  parent = bus_node_allocate(bus, p);                  if (!parent) @@ -1463,6 +1479,32 @@ void bus_node_gc(sd_bus *b, struct node *n) {          free(n);  } +static int bus_find_parent_object_manager(sd_bus *bus, struct node **out, const char *path) { +        struct node *n; + +        assert(bus); +        assert(path); + +        n = hashmap_get(bus->nodes, path); +        if (!n) { +                char *prefix; + +                prefix = alloca(strlen(path) + 1); +                OBJECT_PATH_FOREACH_PREFIX(prefix, path) { +                        n = hashmap_get(bus->nodes, prefix); +                        if (n) +                                break; +                } +        } + +        while (n && !n->object_managers) +                n = n->parent; + +        if (out) +                *out = n; +        return !!n; +} +  static int bus_add_object(                  sd_bus *bus,                  sd_bus_slot **slot, @@ -2261,10 +2303,11 @@ static int object_added_append_all(sd_bus *bus, sd_bus_message *m, const char *p          return 0;  } -int sd_bus_emit_object_added(sd_bus *bus, const char *path) { +_public_ int sd_bus_emit_object_added(sd_bus *bus, const char *path) {          BUS_DONT_DESTROY(bus);          _cleanup_bus_message_unref_ sd_bus_message *m = NULL; +        struct node *object_manager;          int r;          /* @@ -2285,11 +2328,17 @@ int sd_bus_emit_object_added(sd_bus *bus, const char *path) {          if (!BUS_IS_OPEN(bus->state))                  return -ENOTCONN; +        r = bus_find_parent_object_manager(bus, &object_manager, path); +        if (r < 0) +                return r; +        if (r == 0) +                return -ESRCH; +          do {                  bus->nodes_modified = false;                  m = sd_bus_message_unref(m); -                r = sd_bus_message_new_signal(bus, &m, path, "org.freedesktop.DBus.ObjectManager", "InterfacesAdded"); +                r = sd_bus_message_new_signal(bus, &m, object_manager->path, "org.freedesktop.DBus.ObjectManager", "InterfacesAdded");                  if (r < 0)                          return r; @@ -2424,10 +2473,11 @@ static int object_removed_append_all(sd_bus *bus, sd_bus_message *m, const char          return 0;  } -int sd_bus_emit_object_removed(sd_bus *bus, const char *path) { +_public_ int sd_bus_emit_object_removed(sd_bus *bus, const char *path) {          BUS_DONT_DESTROY(bus);          _cleanup_bus_message_unref_ sd_bus_message *m = NULL; +        struct node *object_manager;          int r;          /* @@ -2448,11 +2498,17 @@ int sd_bus_emit_object_removed(sd_bus *bus, const char *path) {          if (!BUS_IS_OPEN(bus->state))                  return -ENOTCONN; +        r = bus_find_parent_object_manager(bus, &object_manager, path); +        if (r < 0) +                return r; +        if (r == 0) +                return -ESRCH; +          do {                  bus->nodes_modified = false;                  m = sd_bus_message_unref(m); -                r = sd_bus_message_new_signal(bus, &m, path, "org.freedesktop.DBus.ObjectManager", "InterfacesRemoved"); +                r = sd_bus_message_new_signal(bus, &m, object_manager->path, "org.freedesktop.DBus.ObjectManager", "InterfacesRemoved");                  if (r < 0)                          return r; @@ -2584,6 +2640,7 @@ _public_ int sd_bus_emit_interfaces_added_strv(sd_bus *bus, const char *path, ch          BUS_DONT_DESTROY(bus);          _cleanup_bus_message_unref_ sd_bus_message *m = NULL; +        struct node *object_manager;          char **i;          int r; @@ -2597,11 +2654,17 @@ _public_ int sd_bus_emit_interfaces_added_strv(sd_bus *bus, const char *path, ch          if (strv_isempty(interfaces))                  return 0; +        r = bus_find_parent_object_manager(bus, &object_manager, path); +        if (r < 0) +                return r; +        if (r == 0) +                return -ESRCH; +          do {                  bus->nodes_modified = false;                  m = sd_bus_message_unref(m); -                r = sd_bus_message_new_signal(bus, &m, path, "org.freedesktop.DBus.ObjectManager", "InterfacesAdded"); +                r = sd_bus_message_new_signal(bus, &m, object_manager->path, "org.freedesktop.DBus.ObjectManager", "InterfacesAdded");                  if (r < 0)                          return r; @@ -2661,6 +2724,7 @@ _public_ int sd_bus_emit_interfaces_added(sd_bus *bus, const char *path, const c  _public_ int sd_bus_emit_interfaces_removed_strv(sd_bus *bus, const char *path, char **interfaces) {          _cleanup_bus_message_unref_ sd_bus_message *m = NULL; +        struct node *object_manager;          int r;          assert_return(bus, -EINVAL); @@ -2673,7 +2737,13 @@ _public_ int sd_bus_emit_interfaces_removed_strv(sd_bus *bus, const char *path,          if (strv_isempty(interfaces))                  return 0; -        r = sd_bus_message_new_signal(bus, &m, path, "org.freedesktop.DBus.ObjectManager", "InterfacesRemoved"); +        r = bus_find_parent_object_manager(bus, &object_manager, path); +        if (r < 0) +                return r; +        if (r == 0) +                return -ESRCH; + +        r = sd_bus_message_new_signal(bus, &m, object_manager->path, "org.freedesktop.DBus.ObjectManager", "InterfacesRemoved");          if (r < 0)                  return r; diff --git a/src/libsystemd/sd-bus/bus-slot.c b/src/libsystemd/sd-bus/bus-slot.c index c452477566..b149ea16da 100644 --- a/src/libsystemd/sd-bus/bus-slot.c +++ b/src/libsystemd/sd-bus/bus-slot.c @@ -273,7 +273,7 @@ _public_ int sd_bus_slot_set_description(sd_bus_slot *slot, const char *descript          return free_and_strdup(&slot->description, description);  } -_public_ int sd_bus_slot_get_description(sd_bus_slot *slot, char **description) { +_public_ int sd_bus_slot_get_description(sd_bus_slot *slot, const char **description) {          assert_return(slot, -EINVAL);          assert_return(description, -EINVAL);          assert_return(slot->description, -ENXIO); diff --git a/src/libsystemd/sd-bus/bus-socket.c b/src/libsystemd/sd-bus/bus-socket.c index 322d57ddbb..735a775cb4 100644 --- a/src/libsystemd/sd-bus/bus-socket.c +++ b/src/libsystemd/sd-bus/bus-socket.c @@ -264,6 +264,8 @@ static bool line_begins(const char *s, size_t m, const char *word) {  static int verify_anonymous_token(sd_bus *b, const char *p, size_t l) {          _cleanup_free_ char *token = NULL; +        size_t len; +        int r;          if (!b->anonymous_auth)                  return 0; @@ -276,11 +278,12 @@ static int verify_anonymous_token(sd_bus *b, const char *p, size_t l) {          if (l % 2 != 0)                  return 0; -        token = unhexmem(p, l); -        if (!token) -                return -ENOMEM; -        if (memchr(token, 0, l/2)) +        r = unhexmem(p, l, (void **) &token, &len); +        if (r < 0) +                return 0; + +        if (memchr(token, 0, len))                  return 0;          return !!utf8_is_valid(token); @@ -288,6 +291,7 @@ static int verify_anonymous_token(sd_bus *b, const char *p, size_t l) {  static int verify_external_token(sd_bus *b, const char *p, size_t l) {          _cleanup_free_ char *token = NULL; +        size_t len;          uid_t u;          int r; @@ -307,11 +311,11 @@ static int verify_external_token(sd_bus *b, const char *p, size_t l) {          if (l % 2 != 0)                  return 0; -        token = unhexmem(p, l); -        if (!token) -                return -ENOMEM; +        r = unhexmem(p, l, (void**) &token, &len); +        if (r < 0) +                return 0; -        if (memchr(token, 0, l/2)) +        if (memchr(token, 0, len))                  return 0;          r = parse_uid(token, &u); diff --git a/src/libsystemd/sd-bus/busctl.c b/src/libsystemd/sd-bus/busctl.c index 39caa4e7d6..6aaaf0e5ec 100644 --- a/src/libsystemd/sd-bus/busctl.c +++ b/src/libsystemd/sd-bus/busctl.c @@ -1137,6 +1137,7 @@ static int monitor(sd_bus *bus, char *argv[], int (*dump)(sd_bus_message *m, FIL                  if (m) {                          dump(m, stdout); +                        fflush(stdout);                          if (sd_bus_message_is_signal(m, "org.freedesktop.DBus.Local", "Disconnected") > 0) {                                  log_info("Connection terminated, exiting."); @@ -1973,7 +1974,7 @@ static int busctl_main(sd_bus *bus, int argc, char *argv[]) {  }  int main(int argc, char *argv[]) { -        _cleanup_bus_close_unref_ sd_bus *bus = NULL; +        _cleanup_bus_flush_close_unref_ sd_bus *bus = NULL;          int r;          log_parse_environment(); diff --git a/src/libsystemd/sd-bus/kdbus.h b/src/libsystemd/sd-bus/kdbus.h index 00a6e142c9..ecffc6b13c 100644 --- a/src/libsystemd/sd-bus/kdbus.h +++ b/src/libsystemd/sd-bus/kdbus.h @@ -374,6 +374,7 @@ enum kdbus_item_type {  	KDBUS_ITEM_ATTACH_FLAGS_RECV,  	KDBUS_ITEM_ID,  	KDBUS_ITEM_NAME, +	KDBUS_ITEM_DST_ID,  	/* keep these item types in sync with KDBUS_ATTACH_* flags */  	_KDBUS_ITEM_ATTACH_BASE	= 0x1000, diff --git a/src/libsystemd/sd-bus/sd-bus.c b/src/libsystemd/sd-bus/sd-bus.c index 5dd6468707..0ca225c617 100644 --- a/src/libsystemd/sd-bus/sd-bus.c +++ b/src/libsystemd/sd-bus/sd-bus.c @@ -1428,6 +1428,17 @@ _public_ void sd_bus_close(sd_bus *bus) {           * ioctl on the fd when they are freed. */  } +_public_ sd_bus* sd_bus_flush_close_unref(sd_bus *bus) { + +        if (!bus) +                return NULL; + +        sd_bus_flush(bus); +        sd_bus_close(bus); + +        return sd_bus_unref(bus); +} +  static void bus_enter_closing(sd_bus *bus) {          assert(bus); diff --git a/src/libsystemd/sd-bus/test-bus-chat.c b/src/libsystemd/sd-bus/test-bus-chat.c index 046e999008..754335b5e7 100644 --- a/src/libsystemd/sd-bus/test-bus-chat.c +++ b/src/libsystemd/sd-bus/test-bus-chat.c @@ -262,7 +262,7 @@ fail:  static void* client1(void*p) {          _cleanup_bus_message_unref_ sd_bus_message *reply = NULL; -        _cleanup_bus_close_unref_ sd_bus *bus = NULL; +        _cleanup_bus_flush_close_unref_ sd_bus *bus = NULL;          _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;          const char *hello;          int r; @@ -361,7 +361,7 @@ static int quit_callback(sd_bus_message *m, void *userdata, sd_bus_error *ret_er  static void* client2(void*p) {          _cleanup_bus_message_unref_ sd_bus_message *m = NULL, *reply = NULL; -        _cleanup_bus_close_unref_ sd_bus *bus = NULL; +        _cleanup_bus_flush_close_unref_ sd_bus *bus = NULL;          _cleanup_bus_error_free_ sd_bus_error error = SD_BUS_ERROR_NULL;          bool quit = false;          const char *mid; diff --git a/src/libsystemd/sd-bus/test-bus-gvariant.c b/src/libsystemd/sd-bus/test-bus-gvariant.c index 22ea00c2fb..9b7dd2e499 100644 --- a/src/libsystemd/sd-bus/test-bus-gvariant.c +++ b/src/libsystemd/sd-bus/test-bus-gvariant.c @@ -132,7 +132,7 @@ static void test_bus_gvariant_get_alignment(void) {  static void test_marshal(void) {          _cleanup_bus_message_unref_ sd_bus_message *m = NULL, *n = NULL; -        _cleanup_bus_close_unref_ sd_bus *bus = NULL; +        _cleanup_bus_flush_close_unref_ sd_bus *bus = NULL;          _cleanup_free_ void *blob;          size_t sz;          int r; diff --git a/src/libsystemd/sd-bus/test-bus-marshal.c b/src/libsystemd/sd-bus/test-bus-marshal.c index a866a56179..59deaea89f 100644 --- a/src/libsystemd/sd-bus/test-bus-marshal.c +++ b/src/libsystemd/sd-bus/test-bus-marshal.c @@ -131,6 +131,9 @@ int main(int argc, char *argv[]) {          r = sd_bus_message_append(m, "a{yv}", 2, 3, "s", "foo", 5, "s", "waldo");          assert_se(r >= 0); +        r = sd_bus_message_append(m, "y(ty)y(yt)y", 8, 777ULL, 7, 9, 77, 7777ULL, 10); +        assert_se(r >= 0); +          r = sd_bus_message_append(m, "ba(ss)", 255, 3, "aaa", "1", "bbb", "2", "ccc", "3");          assert_se(r >= 0); @@ -252,6 +255,22 @@ int main(int argc, char *argv[]) {          assert_se(v == 5);          assert_se(streq(y, "waldo")); +        r = sd_bus_message_read(m, "y(ty)", &v, &u64, &u); +        assert_se(r > 0); +        assert_se(v == 8); +        assert_se(u64 == 777); +        assert_se(u == 7); + +        r = sd_bus_message_read(m, "y(yt)", &v, &u, &u64); +        assert_se(r > 0); +        assert_se(v == 9); +        assert_se(u == 77); +        assert_se(u64 == 7777); + +        r = sd_bus_message_read(m, "y", &v); +        assert_se(r > 0); +        assert_se(v == 10); +          r = sd_bus_message_read(m, "ba(ss)", &boolean, 3, &x, &y, &a, &b, &c, &d);          assert_se(r > 0);          assert_se(boolean); @@ -331,7 +350,7 @@ int main(int argc, char *argv[]) {          assert_se(sd_bus_message_verify_type(m, 'a', "{yv}") > 0); -        r = sd_bus_message_skip(m, "a{yv}"); +        r = sd_bus_message_skip(m, "a{yv}y(ty)y(yt)y");          assert_se(r >= 0);          assert_se(sd_bus_message_verify_type(m, 'b', NULL) > 0); diff --git a/src/libsystemd/sd-bus/test-bus-match.c b/src/libsystemd/sd-bus/test-bus-match.c index a1687b1c7b..83cb5c62c2 100644 --- a/src/libsystemd/sd-bus/test-bus-match.c +++ b/src/libsystemd/sd-bus/test-bus-match.c @@ -92,7 +92,7 @@ int main(int argc, char *argv[]) {          };          _cleanup_bus_message_unref_ sd_bus_message *m = NULL; -        _cleanup_bus_close_unref_ sd_bus *bus = NULL; +        _cleanup_bus_flush_close_unref_ sd_bus *bus = NULL;          enum bus_match_node_type i;          sd_bus_slot slots[19];          int r; diff --git a/src/libsystemd/sd-bus/test-bus-objects.c b/src/libsystemd/sd-bus/test-bus-objects.c index 52952603e4..359984c7f3 100644 --- a/src/libsystemd/sd-bus/test-bus-objects.c +++ b/src/libsystemd/sd-bus/test-bus-objects.c @@ -115,14 +115,13 @@ static int set_handler(sd_bus *bus, const char *path, const char *interface, con  static int value_handler(sd_bus *bus, const char *path, const char *interface, const char *property, sd_bus_message *reply, void *userdata, sd_bus_error *error) {          _cleanup_free_ char *s = NULL; -        const char *x;          int r;          assert_se(asprintf(&s, "object %p, path %s", userdata, path) >= 0);          r = sd_bus_message_append(reply, "s", s);          assert_se(r >= 0); -        assert_se(x = startswith(path, "/value/")); +        assert_se(startswith(path, "/value/") != NULL || strcmp(path, "/value") == 0);          assert_se(PTR_TO_UINT(userdata) == 30); @@ -154,7 +153,7 @@ static int notify_test2(sd_bus_message *m, void *userdata, sd_bus_error *error)  static int emit_interfaces_added(sd_bus_message *m, void *userdata, sd_bus_error *error) {          int r; -        assert_se(sd_bus_emit_interfaces_added(sd_bus_message_get_bus(m), m->path, "org.freedesktop.systemd.test", NULL) >= 0); +        assert_se(sd_bus_emit_interfaces_added(sd_bus_message_get_bus(m), "/value/a/x", "org.freedesktop.systemd.ValueTest", NULL) >= 0);          r = sd_bus_reply_method_return(m, NULL);          assert_se(r >= 0); @@ -165,7 +164,7 @@ static int emit_interfaces_added(sd_bus_message *m, void *userdata, sd_bus_error  static int emit_interfaces_removed(sd_bus_message *m, void *userdata, sd_bus_error *error) {          int r; -        assert_se(sd_bus_emit_interfaces_removed(sd_bus_message_get_bus(m), m->path, "org.freedesktop.systemd.test", NULL) >= 0); +        assert_se(sd_bus_emit_interfaces_removed(sd_bus_message_get_bus(m), "/value/a/x", "org.freedesktop.systemd.ValueTest", NULL) >= 0);          r = sd_bus_reply_method_return(m, NULL);          assert_se(r >= 0); @@ -176,7 +175,7 @@ static int emit_interfaces_removed(sd_bus_message *m, void *userdata, sd_bus_err  static int emit_object_added(sd_bus_message *m, void *userdata, sd_bus_error *error) {          int r; -        assert_se(sd_bus_emit_object_added(sd_bus_message_get_bus(m), m->path) >= 0); +        assert_se(sd_bus_emit_object_added(sd_bus_message_get_bus(m), "/value/a/x") >= 0);          r = sd_bus_reply_method_return(m, NULL);          assert_se(r >= 0); @@ -187,7 +186,7 @@ static int emit_object_added(sd_bus_message *m, void *userdata, sd_bus_error *er  static int emit_object_removed(sd_bus_message *m, void *userdata, sd_bus_error *error) {          int r; -        assert_se(sd_bus_emit_object_removed(sd_bus_message_get_bus(m), m->path) >= 0); +        assert_se(sd_bus_emit_object_removed(sd_bus_message_get_bus(m), "/value/a/x") >= 0);          r = sd_bus_reply_method_return(m, NULL);          assert_se(r >= 0); @@ -229,6 +228,14 @@ static int enumerator_callback(sd_bus *bus, const char *path, void *userdata, ch          return 1;  } +static int enumerator2_callback(sd_bus *bus, const char *path, void *userdata, char ***nodes, sd_bus_error *error) { + +        if (object_path_startswith("/value/a", path)) +                assert_se(*nodes = strv_new("/value/a/x", "/value/a/y", "/value/a/z", NULL)); + +        return 1; +} +  static void *server(void *p) {          struct context *c = p;          sd_bus *bus = NULL; @@ -247,7 +254,9 @@ static void *server(void *p) {          assert_se(sd_bus_add_object_vtable(bus, NULL, "/foo", "org.freedesktop.systemd.test2", vtable, c) >= 0);          assert_se(sd_bus_add_fallback_vtable(bus, NULL, "/value", "org.freedesktop.systemd.ValueTest", vtable2, NULL, UINT_TO_PTR(20)) >= 0);          assert_se(sd_bus_add_node_enumerator(bus, NULL, "/value", enumerator_callback, NULL) >= 0); +        assert_se(sd_bus_add_node_enumerator(bus, NULL, "/value/a", enumerator2_callback, NULL) >= 0);          assert_se(sd_bus_add_object_manager(bus, NULL, "/value") >= 0); +        assert_se(sd_bus_add_object_manager(bus, NULL, "/value/a") >= 0);          assert_se(sd_bus_start(bus) >= 0); diff --git a/src/libsystemd/sd-bus/test-bus-proxy.c b/src/libsystemd/sd-bus/test-bus-proxy.c new file mode 100644 index 0000000000..369c2f331c --- /dev/null +++ b/src/libsystemd/sd-bus/test-bus-proxy.c @@ -0,0 +1,109 @@ +/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/ + +/*** +  This file is part of systemd. + +  Copyright 2015 David Herrmann <dh.herrmann@gmail.com> + +  systemd is free software; you can redistribute it and/or modify it +  under the terms of the GNU Lesser General Public License as published by +  the Free Software Foundation; either version 2.1 of the License, or +  (at your option) any later version. + +  systemd is distributed in the hope that it will be useful, but +  WITHOUT ANY WARRANTY; without even the implied warranty of +  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +  Lesser General Public License for more details. + +  You should have received a copy of the GNU Lesser General Public License +  along with systemd; If not, see <http://www.gnu.org/licenses/>. +***/ + +#include <errno.h> +#include <fcntl.h> +#include <stdlib.h> + +#include "util.h" +#include "log.h" + +#include "sd-bus.h" +#include "bus-kernel.h" +#include "bus-util.h" +#include "bus-dump.h" + +typedef struct { +        const char *sender; +        int matched_acquired; +} TestProxyMatch; + +static int test_proxy_acquired(sd_bus_message *m, void *userdata, sd_bus_error *error) { +        TestProxyMatch *match = userdata; +        const char *name; +        int r; + +        r = sd_bus_message_read(m, "s", &name); +        assert_se(r >= 0); + +        if (!streq_ptr(match->sender, name)) +                return 0; + +        ++match->matched_acquired; +        return 1; +} + +static void test_proxy_matched(void) { +        _cleanup_bus_flush_close_unref_ sd_bus *a = NULL; +        TestProxyMatch match = {}; +        int r; + +        /* open bus 'a' */ + +        r = sd_bus_new(&a); +        assert_se(r >= 0); + +        r = sd_bus_set_address(a, "unix:path=/var/run/dbus/system_bus_socket"); +        assert_se(r >= 0); + +        r = sd_bus_set_bus_client(a, true); +        assert_se(r >= 0); + +        r = sd_bus_start(a); +        assert_se(r >= 0); + +        r = sd_bus_add_match(a, NULL, +                             "type='signal'," +                             "member='NameAcquired'", +                             test_proxy_acquired, &match); +        assert_se(r >= 0); + +        r = sd_bus_get_unique_name(a, &match.sender); +        assert_se(r >= 0); + +        /* barrier to guarantee proxy/dbus-daemon handled the previous data  */ +        r = sd_bus_call_method(a, +                               "org.freedesktop.DBus", +                               "/org/freedesktop/DBus", +                               "org.freedesktop.DBus", +                               "GetId", +                               NULL, NULL, NULL); +        assert_se(r >= 0); + +        /* now we can be sure the Name* signals were sent */ +        do { +                r = sd_bus_process(a, NULL); +        } while (r > 0); +        assert_se(r == 0); + +        assert_se(match.matched_acquired == 1); +} + +int main(int argc, char **argv) { +        if (access("/var/run/dbus/system_bus_socket", F_OK) < 0) +                return EXIT_TEST_SKIP; + +        log_parse_environment(); + +        test_proxy_matched(); + +        return EXIT_SUCCESS; +} | 
