driver core: Add a wrapper around __device_release_driver()
authorRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 10 Oct 2016 12:37:56 +0000 (14:37 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 28 Oct 2016 06:42:26 +0000 (02:42 -0400)
Add an internal wrapper around __device_release_driver() that will
acquire device locks and do the necessary checks before calling it.

The next patch will make use of it.

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/base/dd.c

index d22a7260f42b26f498037f35c943b547300181dc..df4ab5509c045a2863b87bced9054abac7077c30 100644 (file)
@@ -811,6 +811,22 @@ static void __device_release_driver(struct device *dev)
        }
 }
 
+static void device_release_driver_internal(struct device *dev,
+                                          struct device_driver *drv,
+                                          struct device *parent)
+{
+       if (parent)
+               device_lock(parent);
+
+       device_lock(dev);
+       if (!drv || drv == dev->driver)
+               __device_release_driver(dev);
+
+       device_unlock(dev);
+       if (parent)
+               device_unlock(parent);
+}
+
 /**
  * device_release_driver - manually detach device from driver.
  * @dev: device.
@@ -825,9 +841,7 @@ void device_release_driver(struct device *dev)
         * within their ->remove callback for the same device, they
         * will deadlock right here.
         */
-       device_lock(dev);
-       __device_release_driver(dev);
-       device_unlock(dev);
+       device_release_driver_internal(dev, NULL, NULL);
 }
 EXPORT_SYMBOL_GPL(device_release_driver);
 
@@ -852,15 +866,7 @@ void driver_detach(struct device_driver *drv)
                dev = dev_prv->device;
                get_device(dev);
                spin_unlock(&drv->p->klist_devices.k_lock);
-
-               if (dev->parent)        /* Needed for USB */
-                       device_lock(dev->parent);
-               device_lock(dev);
-               if (dev->driver == drv)
-                       __device_release_driver(dev);
-               device_unlock(dev);
-               if (dev->parent)
-                       device_unlock(dev->parent);
+               device_release_driver_internal(dev, drv, dev->parent);
                put_device(dev);
        }
 }