greybus: module: implement controlled module removal
authorJohan Hovold <johan@hovoldconsulting.com>
Sat, 23 Apr 2016 16:47:25 +0000 (18:47 +0200)
committerGreg Kroah-Hartman <gregkh@google.com>
Mon, 25 Apr 2016 18:08:30 +0000 (11:08 -0700)
Implement controlled module removal through a new module attribute
"eject".

When a non-zero argument is written to the attribute, all interfaces of
the module are disabled (e.g. bundles are deregistered) and deactivated
(e.g. powered off) before instructing the SVC to physically eject the
module.

Note that the module device is not deregistered until the SVC has
reported the physical removal of all of its interfaces.

A new interface mutex is added to enforce interface state-change
serialisation.

Signed-off-by: Johan Hovold <johan@hovoldconsulting.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@google.com>
drivers/staging/greybus/Documentation/sysfs-bus-greybus
drivers/staging/greybus/interface.c
drivers/staging/greybus/interface.h
drivers/staging/greybus/module.c
drivers/staging/greybus/svc.c

index 41ffc40b8efb3a336dff21d00888460aa3c5cca2..a18ee7eed75af5d9728774b7f2f8f7b0d6e54248 100644 (file)
@@ -14,6 +14,14 @@ Description:
                A Module M on the bus N, where M is the 1-byte interface
                ID of the module's primary interface.
 
+What:          /sys/bus/greybus/device/N-M/eject
+Date:          March 2016
+KernelVersion: 4.XX
+Contact:       Greg Kroah-Hartman <greg@kroah.com>
+Description:
+               Writing a non-zero argument to this attibute disables the
+               module's interfaces before physically ejecting it.
+
 What:          /sys/bus/greybus/device/N-M/module_id
 Date:          March 2016
 KernelVersion: 4.XX
index d51c5635f22bedeace70488ad9e18ba459adca50..2553312dc33292a4728fd88cf46e2b216fc80c4c 100644 (file)
@@ -372,6 +372,7 @@ struct gb_interface *gb_interface_create(struct gb_module *module,
        intf->interface_id = interface_id;
        INIT_LIST_HEAD(&intf->bundles);
        INIT_LIST_HEAD(&intf->manifest_descs);
+       mutex_init(&intf->mutex);
 
        /* Invalid device id to start with */
        intf->device_id = GB_INTERFACE_DEVICE_ID_BAD;
@@ -388,10 +389,18 @@ struct gb_interface *gb_interface_create(struct gb_module *module,
        return intf;
 }
 
+/*
+ * Activate an interface.
+ *
+ * Locking: Caller holds the interface mutex.
+ */
 int gb_interface_activate(struct gb_interface *intf)
 {
        int ret;
 
+       if (intf->ejected)
+               return -ENODEV;
+
        ret = gb_interface_read_dme(intf);
        if (ret)
                return ret;
@@ -403,6 +412,11 @@ int gb_interface_activate(struct gb_interface *intf)
        return 0;
 }
 
+/*
+ * Deactivate an interface.
+ *
+ * Locking: Caller holds the interface mutex.
+ */
 void gb_interface_deactivate(struct gb_interface *intf)
 {
        gb_interface_route_destroy(intf);
@@ -412,6 +426,8 @@ void gb_interface_deactivate(struct gb_interface *intf)
  * Enable an interface by enabling its control connection, fetching the
  * manifest and other information over it, and finally registering its child
  * devices.
+ *
+ * Locking: Caller holds the interface mutex.
  */
 int gb_interface_enable(struct gb_interface *intf)
 {
@@ -516,7 +532,11 @@ err_put_control:
        return ret;
 }
 
-/* Disable an interface and destroy its bundles. */
+/*
+ * Disable an interface and destroy its bundles.
+ *
+ * Locking: Caller holds the interface mutex.
+ */
 void gb_interface_disable(struct gb_interface *intf)
 {
        struct gb_bundle *bundle;
index 9185c7f1bd32b44579e54b732ee9f1cc9c414c1d..61399e7ea1028e5d491caa94c9c39dcc8989e477 100644 (file)
@@ -39,7 +39,10 @@ struct gb_interface {
 
        unsigned long quirks;
 
+       struct mutex mutex;
+
        bool disconnected;
+       bool ejected;
        bool enabled;
 };
 #define to_gb_interface(d) container_of(d, struct gb_interface, dev)
index 5c498e0a4eaf7e6c738b994084b4a6cde099d59e..5077253037c8fd0322ce8bea2dcb3dd33dca7c0d 100644 (file)
 #include "greybus.h"
 
 
+static ssize_t eject_store(struct device *dev,
+                               struct device_attribute *attr,
+                               const char *buf, size_t len)
+{
+       struct gb_module *module = to_gb_module(dev);
+       struct gb_interface *intf;
+       size_t i;
+       long val;
+       int ret;
+
+       ret = kstrtol(buf, 0, &val);
+       if (ret)
+               return ret;
+
+       if (!val)
+               return len;
+
+       for (i = 0; i < module->num_interfaces; ++i) {
+               intf = module->interfaces[i];
+
+               mutex_lock(&intf->mutex);
+               /* Set flag to prevent concurrent activation. */
+               intf->ejected = true;
+               gb_interface_disable(intf);
+               gb_interface_deactivate(intf);
+               mutex_unlock(&intf->mutex);
+       }
+
+       /* Tell the SVC to eject the primary interface. */
+       ret = gb_svc_intf_eject(module->hd->svc, module->module_id);
+       if (ret)
+               return ret;
+
+       return len;
+}
+static DEVICE_ATTR_WO(eject);
+
 static ssize_t module_id_show(struct device *dev,
                                struct device_attribute *attr, char *buf)
 {
@@ -29,6 +66,7 @@ static ssize_t num_interfaces_show(struct device *dev,
 static DEVICE_ATTR_RO(num_interfaces);
 
 static struct attribute *module_attrs[] = {
+       &dev_attr_eject.attr,
        &dev_attr_module_id.attr,
        &dev_attr_num_interfaces.attr,
        NULL,
@@ -101,12 +139,14 @@ static void gb_module_register_interface(struct gb_interface *intf)
        u8 intf_id = intf->interface_id;
        int ret;
 
+       mutex_lock(&intf->mutex);
+
        ret = gb_interface_activate(intf);
        if (ret) {
                dev_err(&module->dev, "failed to activate interface %u: %d\n",
                                intf_id, ret);
                gb_interface_add(intf);
-               return;
+               goto err_unlock;
        }
 
        ret = gb_interface_add(intf);
@@ -120,10 +160,14 @@ static void gb_module_register_interface(struct gb_interface *intf)
                goto err_interface_deactivate;
        }
 
+       mutex_unlock(&intf->mutex);
+
        return;
 
 err_interface_deactivate:
        gb_interface_deactivate(intf);
+err_unlock:
+       mutex_unlock(&intf->mutex);
 }
 
 static void gb_module_deregister_interface(struct gb_interface *intf)
@@ -132,8 +176,10 @@ static void gb_module_deregister_interface(struct gb_interface *intf)
        if (intf->module->disconnected)
                intf->disconnected = true;
 
+       mutex_lock(&intf->mutex);
        gb_interface_disable(intf);
        gb_interface_deactivate(intf);
+       mutex_unlock(&intf->mutex);
 
        gb_interface_del(intf);
 }
index 94016954a6aba7a339579f56d81be224e4c2175e..96b3027db528362bde4ead095793098ff9d1c8d3 100644 (file)
@@ -682,6 +682,8 @@ static void gb_svc_intf_reenable(struct gb_svc *svc, struct gb_interface *intf)
 {
        int ret;
 
+       mutex_lock(&intf->mutex);
+
        /* Mark as disconnected to prevent I/O during disable. */
        intf->disconnected = true;
        gb_interface_disable(intf);
@@ -694,6 +696,8 @@ static void gb_svc_intf_reenable(struct gb_svc *svc, struct gb_interface *intf)
 
                gb_interface_deactivate(intf);
        }
+
+       mutex_unlock(&intf->mutex);
 }
 
 static void gb_svc_process_intf_hotplug(struct gb_operation *operation)