]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
driver core: Add flag to autoremove device link on supplier unbind
authorVivek Gautam <vivek.gautam@codeaurora.org>
Wed, 27 Jun 2018 12:50:56 +0000 (18:20 +0530)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 9 Jul 2018 10:14:31 +0000 (12:14 +0200)
Add a flag to autoremove the device links on supplier driver
unbind. This obviates the need to explicitly delete the link
in the remove path.
We remove these links only when the supplier's link to its
consumers has gone to DL_STATE_SUPPLIER_UNBIND state.

Signed-off-by: Vivek Gautam <vivek.gautam@codeaurora.org>
Suggested-by: Lukas Wunner <lukas@wunner.de>
Reviewed-by: Ulf Hansson <ulf.hansson@linaro.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Documentation/driver-api/device_link.rst
drivers/base/core.c
include/linux/device.h

index a005b904a264b5f79f6e0b989f084afbc366f704..d6763272e747c80361fafe85a69c4720c01eb227 100644 (file)
@@ -86,6 +86,10 @@ automatically purged when the consumer fails to probe or later unbinds.
 This obviates the need to explicitly delete the link in the ``->remove``
 callback or in the error path of the ``->probe`` callback.
 
+Similarly, when the device link is added from supplier's ``->probe`` callback,
+``DL_FLAG_AUTOREMOVE_SUPPLIER`` causes the device link to be automatically
+purged when the supplier fails to probe or later unbinds.
+
 Limitations
 ===========
 
index 14c1e3151e08d65cb09412e04dc66d0531253c0c..e721218bf352f89afc84eaf09f6f10b5d679c72f 100644 (file)
@@ -518,6 +518,16 @@ void device_links_driver_cleanup(struct device *dev)
 
                WARN_ON(link->flags & DL_FLAG_AUTOREMOVE_CONSUMER);
                WARN_ON(link->status != DL_STATE_SUPPLIER_UNBIND);
+
+               /*
+                * autoremove the links between this @dev and its consumer
+                * devices that are not active, i.e. where the link state
+                * has moved to DL_STATE_SUPPLIER_UNBIND.
+                */
+               if (link->status == DL_STATE_SUPPLIER_UNBIND &&
+                   link->flags & DL_FLAG_AUTOREMOVE_SUPPLIER)
+                       kref_put(&link->kref, __device_link_del);
+
                WRITE_ONCE(link->status, DL_STATE_DORMANT);
        }
 
index 3929805cdd59405cb0c1476dfad94bdc06e1ee85..e80920452b49a7910c879af3259b1c67601a7fa3 100644 (file)
@@ -787,11 +787,13 @@ enum device_link_state {
  * AUTOREMOVE_CONSUMER: Remove the link automatically on consumer driver unbind.
  * PM_RUNTIME: If set, the runtime PM framework will use this link.
  * RPM_ACTIVE: Run pm_runtime_get_sync() on the supplier during link creation.
+ * AUTOREMOVE_SUPPLIER: Remove the link automatically on supplier driver unbind.
  */
 #define DL_FLAG_STATELESS              BIT(0)
 #define DL_FLAG_AUTOREMOVE_CONSUMER    BIT(1)
 #define DL_FLAG_PM_RUNTIME             BIT(2)
 #define DL_FLAG_RPM_ACTIVE             BIT(3)
+#define DL_FLAG_AUTOREMOVE_SUPPLIER    BIT(4)
 
 /**
  * struct device_link - Device link representation.