summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorYu Watanabe <watanabe.yu+github@gmail.com>2024-08-05 05:29:46 +0200
committerYu Watanabe <watanabe.yu+github@gmail.com>2024-08-05 22:22:10 +0200
commit67ea8a4c0edef33b1775536bc81d5de2c8ac4d88 (patch)
tree9a56173b7573a3071012fae375113edcf447ab0b /src
parentudevadm-info: trival cleanups (diff)
downloadsystemd-67ea8a4c0edef33b1775536bc81d5de2c8ac4d88.tar.xz
systemd-67ea8a4c0edef33b1775536bc81d5de2c8ac4d88.zip
udevadm-info: support json output for --attribute-walk
Closes #33852.
Diffstat (limited to 'src')
-rw-r--r--src/udev/udevadm-info.c110
1 files changed, 107 insertions, 3 deletions
diff --git a/src/udev/udevadm-info.c b/src/udev/udevadm-info.c
index 1a0e2656ce..824b6fc726 100644
--- a/src/udev/udevadm-info.c
+++ b/src/udev/udevadm-info.c
@@ -186,6 +186,107 @@ static int print_device_chain(sd_device *device) {
return 0;
}
+static int print_all_attributes_in_json(sd_device *device, bool is_parent) {
+ _cleanup_(sd_json_variant_unrefp) sd_json_variant *v = NULL, *w = NULL;
+ _cleanup_free_ SysAttr *sysattrs = NULL;
+ const char *value;
+ size_t n_items = 0;
+ int r;
+
+ assert(device);
+
+ value = NULL;
+ (void) sd_device_get_devpath(device, &value);
+ r = sd_json_variant_set_field_string(&v, "DEVPATH", value);
+ if (r < 0)
+ return r;
+
+ value = NULL;
+ (void) sd_device_get_sysname(device, &value);
+ r = sd_json_variant_set_field_string(&v, is_parent ? "KERNELS" : "KERNEL", value);
+ if (r < 0)
+ return r;
+
+ value = NULL;
+ (void) sd_device_get_subsystem(device, &value);
+ r = sd_json_variant_set_field_string(&v, is_parent ? "SUBSYSTEMS" : "SUBSYSTEM", value);
+ if (r < 0)
+ return r;
+
+ value = NULL;
+ (void) sd_device_get_driver(device, &value);
+ r = sd_json_variant_set_field_string(&v, is_parent ? "DRIVERS" : "DRIVER", value);
+ if (r < 0)
+ return r;
+
+ FOREACH_DEVICE_SYSATTR(device, name) {
+ size_t len;
+
+ if (skip_attribute(name))
+ continue;
+
+ r = sd_device_get_sysattr_value(device, name, &value);
+ if (r >= 0) {
+ /* skip any values that look like a path */
+ if (value[0] == '/')
+ continue;
+
+ /* skip nonprintable attributes */
+ len = strlen(value);
+ while (len > 0 && isprint((unsigned char) value[len-1]))
+ len--;
+ if (len > 0)
+ continue;
+
+ } else if (ERRNO_IS_PRIVILEGE(r))
+ value = "(not readable)";
+ else
+ continue;
+
+ if (!GREEDY_REALLOC(sysattrs, n_items + 1))
+ return log_oom();
+
+ sysattrs[n_items] = (SysAttr) {
+ .name = name,
+ .value = value,
+ };
+ n_items++;
+ }
+
+ typesafe_qsort(sysattrs, n_items, sysattr_compare);
+
+ FOREACH_ARRAY(i, sysattrs, n_items) {
+ r = sd_json_variant_set_field_string(&w, i->name, i->value);
+ if (r < 0)
+ return r;
+ }
+
+ r = sd_json_variant_set_field(&v, is_parent ? "ATTRS" : "ATTR", w);
+ if (r < 0)
+ return r;
+
+ return sd_json_variant_dump(v, arg_json_format_flags, stdout, NULL);
+}
+
+static int print_device_chain_in_json(sd_device *device) {
+ sd_device *child, *parent;
+ int r;
+
+ assert(device);
+
+ r = print_all_attributes_in_json(device, /* is_parent = */ false);
+ if (r < 0)
+ return r;
+
+ for (child = device; sd_device_get_parent(child, &parent) >= 0; child = parent) {
+ r = print_all_attributes_in_json(parent, /* is_parent = */ true);
+ if (r < 0)
+ return r;
+ }
+
+ return 0;
+}
+
static int print_record(sd_device *device, const char *prefix) {
const char *str, *subsys;
dev_t devnum;
@@ -1108,9 +1209,12 @@ int info_main(int argc, char *argv[], void *userdata) {
if (action == ACTION_QUERY)
r = query_device(query, device);
- else if (action == ACTION_ATTRIBUTE_WALK)
- r = print_device_chain(device);
- else if (action == ACTION_TREE)
+ else if (action == ACTION_ATTRIBUTE_WALK) {
+ if (arg_json_format_flags & SD_JSON_FORMAT_OFF)
+ r = print_device_chain(device);
+ else
+ r = print_device_chain_in_json(device);
+ } else if (action == ACTION_TREE)
r = print_tree(device);
else
assert_not_reached();