driver core: bus_type: add bus_groups
authorGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 8 Aug 2013 22:22:57 +0000 (15:22 -0700)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 12 Aug 2013 22:33:31 +0000 (15:33 -0700)
attribute groups are much more flexible than just a list of attributes,
due to their support for visibility of the attributes, and binary
attributes. Add bus_groups to struct bus_type which should be used
instead of bus_attrs.

bus_attrs will be removed from the structure soon.

Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/base/bus.c
include/linux/device.h

index de5ce22df306c8d4b95073781e9d3eabf1457d42..5ee5d3c1d74bb0e34e0341267b803dad94b0a6da 100644 (file)
@@ -881,6 +881,37 @@ static void bus_remove_attrs(struct bus_type *bus)
        }
 }
 
+static int bus_add_groups(struct bus_type *bus,
+                         const struct attribute_group **groups)
+{
+       int error = 0;
+       int i;
+
+       if (groups) {
+               for (i = 0; groups[i]; i++) {
+                       error = sysfs_create_group(&bus->p->subsys.kobj,
+                                                  groups[i]);
+                       if (error) {
+                               while (--i >= 0)
+                                       sysfs_remove_group(&bus->p->subsys.kobj,
+                                                          groups[i]);
+                               break;
+                       }
+               }
+       }
+       return error;
+}
+
+static void bus_remove_groups(struct bus_type *bus,
+                             const struct attribute_group **groups)
+{
+       int i;
+
+       if (groups)
+               for (i = 0; groups[i]; i++)
+                       sysfs_remove_group(&bus->p->subsys.kobj, groups[i]);
+}
+
 static void klist_devices_get(struct klist_node *n)
 {
        struct device_private *dev_prv = to_device_private_bus(n);
@@ -973,10 +1004,15 @@ int bus_register(struct bus_type *bus)
        retval = bus_add_attrs(bus);
        if (retval)
                goto bus_attrs_fail;
+       retval = bus_add_groups(bus, bus->bus_groups);
+       if (retval)
+               goto bus_groups_fail;
 
        pr_debug("bus: '%s': registered\n", bus->name);
        return 0;
 
+bus_groups_fail:
+       bus_remove_attrs(bus);
 bus_attrs_fail:
        remove_probe_files(bus);
 bus_probe_files_fail:
@@ -1007,6 +1043,7 @@ void bus_unregister(struct bus_type *bus)
        if (bus->dev_root)
                device_unregister(bus->dev_root);
        bus_remove_attrs(bus);
+       bus_remove_groups(bus, bus->bus_groups);
        remove_probe_files(bus);
        kset_unregister(bus->p->drivers_kset);
        kset_unregister(bus->p->devices_kset);
index 94d987f128bbe4ea2bc7c6321664bf2f08245653..26e23b22ed0a20a45ecb4e4eef0711a5d5e6b7bf 100644 (file)
@@ -66,6 +66,7 @@ extern void bus_remove_file(struct bus_type *, struct bus_attribute *);
  * @bus_attrs: Default attributes of the bus.
  * @dev_attrs: Default attributes of the devices on the bus.
  * @drv_attrs: Default attributes of the device drivers on the bus.
+ * @bus_groups:        Default attributes of the bus.
  * @dev_groups:        Default attributes of the devices on the bus.
  * @drv_groups: Default attributes of the device drivers on the bus.
  * @match:     Called, perhaps multiple times, whenever a new device or driver
@@ -105,9 +106,10 @@ struct bus_type {
        const char              *name;
        const char              *dev_name;
        struct device           *dev_root;
-       struct bus_attribute    *bus_attrs;
+       struct bus_attribute    *bus_attrs;     /* use bus_groups instead */
        struct device_attribute *dev_attrs;     /* use dev_groups instead */
        struct driver_attribute *drv_attrs;     /* use drv_groups instead */
+       const struct attribute_group **bus_groups;
        const struct attribute_group **dev_groups;
        const struct attribute_group **drv_groups;