Do not disable driver and bus shutdown hook when class shutdown hook is set.
authorMichal Suchanek <msuchanek@suse.de>
Fri, 11 Aug 2017 13:44:43 +0000 (15:44 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Mon, 28 Aug 2017 16:02:46 +0000 (18:02 +0200)
As seen from the implementation of the single class shutdown hook this
is not very sound design.

Rename the class shutdown hook to shutdown_pre to make it clear it runs
before the driver shutdown hook.

Signed-off-by: Michal Suchanek <msuchanek@suse.de>
Reviewed-by: Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/base/core.c
drivers/char/tpm/tpm-chip.c
include/linux/device.h

index cb4b5b1f1b095e951187289ab565fe632a4013c6..12ebd055724cd6cdc01edff013583f38979fe92a 100644 (file)
@@ -2796,11 +2796,12 @@ void device_shutdown(void)
                pm_runtime_get_noresume(dev);
                pm_runtime_barrier(dev);
 
-               if (dev->class && dev->class->shutdown) {
+               if (dev->class && dev->class->shutdown_pre) {
                        if (initcall_debug)
-                               dev_info(dev, "shutdown\n");
-                       dev->class->shutdown(dev);
-               } else if (dev->bus && dev->bus->shutdown) {
+                               dev_info(dev, "shutdown_pre\n");
+                       dev->class->shutdown_pre(dev);
+               }
+               if (dev->bus && dev->bus->shutdown) {
                        if (initcall_debug)
                                dev_info(dev, "shutdown\n");
                        dev->bus->shutdown(dev);
index 67ec9d3d04f529ba07fa1feb926bd1809a78a405..0eca20c5a80cf21f95864f93102f10116151aacf 100644 (file)
@@ -164,14 +164,7 @@ static int tpm_class_shutdown(struct device *dev)
                chip->ops = NULL;
                up_write(&chip->ops_sem);
        }
-       /* Allow bus- and device-specific code to run. Note: since chip->ops
-        * is NULL, more-specific shutdown code will not be able to issue TPM
-        * commands.
-        */
-       if (dev->bus && dev->bus->shutdown)
-               dev->bus->shutdown(dev);
-       else if (dev->driver && dev->driver->shutdown)
-               dev->driver->shutdown(dev);
+
        return 0;
 }
 
@@ -214,7 +207,7 @@ struct tpm_chip *tpm_chip_alloc(struct device *pdev,
        device_initialize(&chip->devs);
 
        chip->dev.class = tpm_class;
-       chip->dev.class->shutdown = tpm_class_shutdown;
+       chip->dev.class->shutdown_pre = tpm_class_shutdown;
        chip->dev.release = tpm_dev_release;
        chip->dev.parent = pdev;
        chip->dev.groups = chip->groups;
index d5bcc5f109eb2a67201fc0a2444de27394255d98..c6f27207dbe87392502dba2082e73617bd6d02ac 100644 (file)
@@ -375,7 +375,7 @@ int subsys_virtual_register(struct bus_type *subsys,
  * @suspend:   Used to put the device to sleep mode, usually to a low power
  *             state.
  * @resume:    Used to bring the device from the sleep mode.
- * @shutdown:  Called at shut-down time to quiesce the device.
+ * @shutdown_pre: Called at shut-down time before driver shutdown.
  * @ns_type:   Callbacks so sysfs can detemine namespaces.
  * @namespace: Namespace of the device belongs to this class.
  * @pm:                The default device power management operations of this class.
@@ -404,7 +404,7 @@ struct class {
 
        int (*suspend)(struct device *dev, pm_message_t state);
        int (*resume)(struct device *dev);
-       int (*shutdown)(struct device *dev);
+       int (*shutdown_pre)(struct device *dev);
 
        const struct kobj_ns_type_operations *ns_type;
        const void *(*namespace)(struct device *dev);