* [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
@ 2015-09-17 12:56 ` Tomeu Vizoso
2015-09-17 15:52 ` Alan Stern
2015-09-17 12:56 ` [PATCH v5 02/23] ARM: amba: Move reading of periphid to pre_probe() Tomeu Vizoso
` (20 subsequent siblings)
21 siblings, 1 reply; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:56 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Some buses (eg. AMBA) need access to some HW resources (it may need a
clock to be enabled so a device ID can be read) before a device can be
matched to a driver.
The pre_probe callback allows the device-driver core to request the bus
to perform this initialization and can defer the probe if any of the
resources needed are missing.
This gives us more flexibility when setting the order in which devices
are probed because the resources needed to get the matching information
don't need to be available by the time that the bus devices are
registered.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5:
- Reduce some code duplication by adding device_pre_probe()
- Print a warning if pre_probe() returns an error
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/base/dd.c | 39 +++++++++++++++++++++++++++++++++++++++
include/linux/device.h | 4 ++++
2 files changed, 43 insertions(+)
diff --git a/drivers/base/dd.c b/drivers/base/dd.c
index be0eb4639128..5caa8478404d 100644
--- a/drivers/base/dd.c
+++ b/drivers/base/dd.c
@@ -539,6 +539,37 @@ static void __device_attach_async_helper(void *_dev, async_cookie_t cookie)
put_device(dev);
}
+/**
+ * device_pre_probe - perform initializations before a device can be matched
+ * @dev: device.
+ *
+ * Call the pre_probe() callback in the bus, if any, to give it a chance to
+ * perform any initializations needed before the device can be matched to a
+ * driver (for example, read a device id from a register).
+ *
+ * Returns 1 if the operation was successful or there's no pre_probe callback
+ * 0 if an error happened and the device isn't ready to be matched to a driver
+ */
+static int device_pre_probe(struct device *dev)
+{
+ int ret;
+
+ if (!dev->bus || !dev->bus->pre_probe)
+ return 1;
+
+ ret = dev->bus->pre_probe(dev);
+ if (ret) {
+ if (ret == -EPROBE_DEFER)
+ driver_deferred_probe_add(dev);
+ else
+ dev_warn(dev, "pre-probe failed: %d\n", ret);
+
+ return 0;
+ }
+
+ return 1;
+}
+
static int __device_attach(struct device *dev, bool allow_async)
{
int ret = 0;
@@ -566,6 +597,11 @@ static int __device_attach(struct device *dev, bool allow_async)
if (dev->parent)
pm_runtime_get_sync(dev->parent);
+ if (!device_pre_probe(dev)) {
+ ret = 0;
+ goto out_unlock;
+ }
+
ret = bus_for_each_drv(dev->bus, NULL, &data,
__device_attach_driver);
if (!ret && allow_async && data.have_async) {
@@ -630,6 +666,9 @@ static int __driver_attach(struct device *dev, void *data)
* is an error.
*/
+ if (!device_pre_probe(dev))
+ return 0;
+
if (!driver_match_device(drv, dev))
return 0;
diff --git a/include/linux/device.h b/include/linux/device.h
index 5d7bc6349930..d8be07bc9c3f 100644
--- a/include/linux/device.h
+++ b/include/linux/device.h
@@ -74,6 +74,9 @@ extern void bus_remove_file(struct bus_type *, struct bus_attribute *);
* given device can be handled by the given driver.
* @uevent: Called when a device is added, removed, or a few other things
* that generate uevents to add the environment variables.
+ * @pre_probe: Called when a new device or driver is added to this bus, to
+ * perform any initializations that are needed so the device can
+ * be matched to a driver.
* @probe: Called when a new device or driver add to this bus, and callback
* the specific driver's probe to initial the matched device.
* @remove: Called when a device removed from this bus.
@@ -113,6 +116,7 @@ struct bus_type {
int (*match)(struct device *dev, struct device_driver *drv);
int (*uevent)(struct device *dev, struct kobj_uevent_env *env);
+ int (*pre_probe)(struct device *dev);
int (*probe)(struct device *dev);
int (*remove)(struct device *dev);
void (*shutdown)(struct device *dev);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* Re: [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type
2015-09-17 12:56 ` [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type Tomeu Vizoso
@ 2015-09-17 15:52 ` Alan Stern
2015-09-18 14:56 ` Tomeu Vizoso
0 siblings, 1 reply; 30+ messages in thread
From: Alan Stern @ 2015-09-17 15:52 UTC (permalink / raw)
To: Tomeu Vizoso
Cc: linux-kernel, Rob Herring, Stephen Warren,
Javier Martinez Canillas, Greg Kroah-Hartman, Mark Brown,
Thierry Reding, Rafael J. Wysocki, linux-arm-kernel,
Dmitry Torokhov, devicetree, Linus Walleij, linux-acpi,
Arnd Bergmann
On Thu, 17 Sep 2015, Tomeu Vizoso wrote:
> Some buses (eg. AMBA) need access to some HW resources (it may need a
> clock to be enabled so a device ID can be read) before a device can be
> matched to a driver.
>
> The pre_probe callback allows the device-driver core to request the bus
> to perform this initialization and can defer the probe if any of the
> resources needed are missing.
>
> This gives us more flexibility when setting the order in which devices
> are probed because the resources needed to get the matching information
> don't need to be available by the time that the bus devices are
> registered.
Can't the subsystem do this itself in its matching routine?
Alan Stern
^ permalink raw reply [flat|nested] 30+ messages in thread
* Re: [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type
2015-09-17 15:52 ` Alan Stern
@ 2015-09-18 14:56 ` Tomeu Vizoso
0 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-18 14:56 UTC (permalink / raw)
To: Alan Stern
Cc: linux-kernel@vger.kernel.org, Rob Herring, Stephen Warren,
Javier Martinez Canillas, Greg Kroah-Hartman, Mark Brown,
Thierry Reding, Rafael J. Wysocki,
linux-arm-kernel@lists.infradead.org, Dmitry Torokhov,
devicetree@vger.kernel.org, Linus Walleij, linux-acpi,
Arnd Bergmann
On 17 September 2015 at 17:52, Alan Stern <stern@rowland.harvard.edu> wrote:
> On Thu, 17 Sep 2015, Tomeu Vizoso wrote:
>
>> Some buses (eg. AMBA) need access to some HW resources (it may need a
>> clock to be enabled so a device ID can be read) before a device can be
>> matched to a driver.
>>
>> The pre_probe callback allows the device-driver core to request the bus
>> to perform this initialization and can defer the probe if any of the
>> resources needed are missing.
>>
>> This gives us more flexibility when setting the order in which devices
>> are probed because the resources needed to get the matching information
>> don't need to be available by the time that the bus devices are
>> registered.
>
> Can't the subsystem do this itself in its matching routine?
That seems to work, thanks for the idea!
Regards,
Tomeu
> Alan Stern
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at http://www.tux.org/lkml/
^ permalink raw reply [flat|nested] 30+ messages in thread
* [PATCH v5 02/23] ARM: amba: Move reading of periphid to pre_probe()
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
2015-09-17 12:56 ` [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type Tomeu Vizoso
@ 2015-09-17 12:56 ` Tomeu Vizoso
2015-09-17 12:56 ` [PATCH v5 03/23] of/platform: Point to struct device from device node Tomeu Vizoso
` (19 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:56 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Reading the periphid when the Primecell device is registered means that
the apb pclk must be available by then or the device won't be registered
at all.
By moving this code to pre_probe (to be called by the device-driver core
before the device is matched to a driver) we remove any order
requirements.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4:
- Added bus.pre_probe callback so the probes of Primecell devices can be
deferred if their device IDs cannot be yet read because of the clock
driver not having probed when they are registered. Maybe this goes
overboard and the matching information should be in the DT if there is
one.
Changes in v3: None
Changes in v2: None
drivers/amba/bus.c | 78 ++++++++++++++++++++++++++----------------------------
1 file changed, 38 insertions(+), 40 deletions(-)
diff --git a/drivers/amba/bus.c b/drivers/amba/bus.c
index f0099360039e..1cbe43b4acd3 100644
--- a/drivers/amba/bus.c
+++ b/drivers/amba/bus.c
@@ -24,6 +24,8 @@
#define to_amba_driver(d) container_of(d, struct amba_driver, drv)
+static int amba_pre_probe(struct device *dev);
+
static const struct amba_id *
amba_lookup(const struct amba_id *table, struct amba_device *dev)
{
@@ -193,6 +195,7 @@ struct bus_type amba_bustype = {
.name = "amba",
.dev_attrs = amba_dev_attrs,
.match = amba_match,
+ .pre_probe = amba_pre_probe,
.uevent = amba_uevent,
.pm = &amba_pm,
};
@@ -336,44 +339,26 @@ static void amba_device_release(struct device *dev)
kfree(d);
}
-/**
- * amba_device_add - add a previously allocated AMBA device structure
- * @dev: AMBA device allocated by amba_device_alloc
- * @parent: resource parent for this devices resources
- *
- * Claim the resource, and read the device cell ID if not already
- * initialized. Register the AMBA device with the Linux device
- * manager.
- */
-int amba_device_add(struct amba_device *dev, struct resource *parent)
+static int amba_pre_probe(struct device *dev)
{
+ struct amba_device *d = to_amba_device(dev);
u32 size;
void __iomem *tmp;
int i, ret;
- WARN_ON(dev->irq[0] == (unsigned int)-1);
- WARN_ON(dev->irq[1] == (unsigned int)-1);
-
- ret = request_resource(parent, &dev->res);
- if (ret)
- goto err_out;
-
- /* Hard-coded primecell ID instead of plug-n-play */
- if (dev->periphid != 0)
- goto skip_probe;
+ if (d->periphid != 0)
+ return 0;
/*
* Dynamically calculate the size of the resource
* and use this for iomap
*/
- size = resource_size(&dev->res);
- tmp = ioremap(dev->res.start, size);
- if (!tmp) {
- ret = -ENOMEM;
- goto err_release;
- }
+ size = resource_size(&d->res);
+ tmp = ioremap(d->res.start, size);
+ if (!tmp)
+ return -ENOMEM;
- ret = amba_get_enable_pclk(dev);
+ ret = amba_get_enable_pclk(d);
if (ret == 0) {
u32 pid, cid;
@@ -388,37 +373,50 @@ int amba_device_add(struct amba_device *dev, struct resource *parent)
cid |= (readl(tmp + size - 0x10 + 4 * i) & 255) <<
(i * 8);
- amba_put_disable_pclk(dev);
+ amba_put_disable_pclk(d);
if (cid == AMBA_CID || cid == CORESIGHT_CID)
- dev->periphid = pid;
+ d->periphid = pid;
- if (!dev->periphid)
+ if (!d->periphid)
ret = -ENODEV;
}
iounmap(tmp);
+ return ret;
+}
+
+/**
+ * amba_device_add - add a previously allocated AMBA device structure
+ * @dev: AMBA device allocated by amba_device_alloc
+ * @parent: resource parent for this devices resources
+ *
+ * Claim the resource, and register the AMBA device with the Linux device
+ * manager.
+ */
+int amba_device_add(struct amba_device *dev, struct resource *parent)
+{
+ int ret;
+
+ WARN_ON(dev->irq[0] == (unsigned int)-1);
+ WARN_ON(dev->irq[1] == (unsigned int)-1);
+
+ ret = request_resource(parent, &dev->res);
if (ret)
- goto err_release;
+ return ret;
- skip_probe:
ret = device_add(&dev->dev);
if (ret)
- goto err_release;
+ return ret;
if (dev->irq[0])
ret = device_create_file(&dev->dev, &dev_attr_irq0);
if (ret == 0 && dev->irq[1])
ret = device_create_file(&dev->dev, &dev_attr_irq1);
- if (ret == 0)
- return ret;
-
- device_unregister(&dev->dev);
+ if (ret)
+ device_unregister(&dev->dev);
- err_release:
- release_resource(&dev->res);
- err_out:
return ret;
}
EXPORT_SYMBOL_GPL(amba_device_add);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 03/23] of/platform: Point to struct device from device node
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
2015-09-17 12:56 ` [PATCH v5 01/23] driver core: Add pre_probe callback to bus_type Tomeu Vizoso
2015-09-17 12:56 ` [PATCH v5 02/23] ARM: amba: Move reading of periphid to pre_probe() Tomeu Vizoso
@ 2015-09-17 12:56 ` Tomeu Vizoso
[not found] ` <1442494637-3674-1-git-send-email-tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
` (18 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:56 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When adding platform and AMBA devices, set the device node's device
member to point to it.
This speeds lookups considerably and is safe because we only create one
of these devices for any given device node.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5:
- Set the pointer to struct device also for AMBA devices
- Unset the pointer to struct device when the platform device is about
to be unregistered
- Increase the reference count of the device before returning from
of_find_device_by_node()
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/of/platform.c | 19 ++++++++++---------
include/linux/of.h | 1 +
2 files changed, 11 insertions(+), 9 deletions(-)
diff --git a/drivers/of/platform.c b/drivers/of/platform.c
index 1001efaedcb8..408d89f1d124 100644
--- a/drivers/of/platform.c
+++ b/drivers/of/platform.c
@@ -32,11 +32,6 @@ const struct of_device_id of_default_bus_match_table[] = {
{} /* Empty terminated list */
};
-static int of_dev_node_match(struct device *dev, void *data)
-{
- return dev->of_node == data;
-}
-
/**
* of_find_device_by_node - Find the platform_device associated with a node
* @np: Pointer to device tree node
@@ -45,10 +40,10 @@ static int of_dev_node_match(struct device *dev, void *data)
*/
struct platform_device *of_find_device_by_node(struct device_node *np)
{
- struct device *dev;
-
- dev = bus_find_device(&platform_bus_type, NULL, np, of_dev_node_match);
- return dev ? to_platform_device(dev) : NULL;
+ if (np->device && np->device->bus == &platform_bus_type &&
+ get_device(np->device))
+ return to_platform_device(np->device);
+ return NULL;
}
EXPORT_SYMBOL(of_find_device_by_node);
@@ -192,6 +187,8 @@ static struct platform_device *of_platform_device_create_pdata(
goto err_clear_flag;
}
+ np->device = &dev->dev;
+
return dev;
err_clear_flag:
@@ -272,6 +269,8 @@ static struct amba_device *of_amba_device_create(struct device_node *node,
goto err_free;
}
+ node->device = &dev->dev;
+
return dev;
err_free:
@@ -476,6 +475,8 @@ static int of_platform_device_destroy(struct device *dev, void *data)
if (of_node_check_flag(dev->of_node, OF_POPULATED_BUS))
device_for_each_child(dev, NULL, of_platform_device_destroy);
+ dev->of_node->device = NULL;
+
if (dev->bus == &platform_bus_type)
platform_device_unregister(to_platform_device(dev));
#ifdef CONFIG_ARM_AMBA
diff --git a/include/linux/of.h b/include/linux/of.h
index 2194b8ca41f9..eb091be0f8ee 100644
--- a/include/linux/of.h
+++ b/include/linux/of.h
@@ -52,6 +52,7 @@ struct device_node {
phandle phandle;
const char *full_name;
struct fwnode_handle fwnode;
+ struct device *device;
struct property *properties;
struct property *deadprops; /* removed properties */
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
[parent not found: <1442494637-3674-1-git-send-email-tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>]
* [PATCH v5 04/23] of: add function to allow probing a device from a OF node
[not found] ` <1442494637-3674-1-git-send-email-tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
@ 2015-09-17 12:56 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 10/23] drm: Probe panels on demand Tomeu Vizoso
1 sibling, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:56 UTC (permalink / raw)
To: linux-kernel-u79uwXL29TY76Z2rM5mHXA
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki,
linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r,
Dmitry Torokhov, devicetree-u79uwXL29TY76Z2rM5mHXA, Linus Walleij,
linux-acpi-u79uwXL29TY76Z2rM5mHXA, Arnd Bergmann, Tomeu Vizoso
Walks the OF tree up and finds the closest ancestor that has a struct
device associated with it, probing it if isn't bound to a driver yet.
The above should ensure that the dependency represented by the passed OF
node is available, because probing a device should cause its descendants
to be probed as well (when they get registered).
Subsystems can use this when looking up resources for drivers, to reduce
the chances of deferred probes because of the probing order of devices.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
---
Changes in v5:
- Move the assignment to device_node->device for AMBA devices to another
commit.
- Hold a reference to the struct device while it's in use in
of_device_probe().
Changes in v4:
- Rename of_platform_probe to of_device_probe
- Use device_node.device instead of device_node.platform_dev
Changes in v3:
- Set and use device_node.platform_dev instead of reversing the logic to
find the platform device that encloses a device node.
- Drop the fwnode API to probe firmware nodes and add OF-only API for
now. I think this same scheme could be used for machines with ACPI,
but I haven't been able to find one that had to defer its probes because
of the device probe order.
Changes in v2: None
drivers/of/device.c | 61 +++++++++++++++++++++++++++++++++++++++++++++++
include/linux/of_device.h | 3 +++
2 files changed, 64 insertions(+)
diff --git a/drivers/of/device.c b/drivers/of/device.c
index 8b91ea241b10..836be71fc90e 100644
--- a/drivers/of/device.c
+++ b/drivers/of/device.c
@@ -286,3 +286,64 @@ int of_device_uevent_modalias(struct device *dev, struct kobj_uevent_env *env)
return 0;
}
+
+/**
+ * of_device_probe() - Probe device associated with OF node
+ * @np: node to probe
+ *
+ * Probe the device associated with the passed device node.
+ */
+void of_device_probe(struct device_node *np)
+{
+ struct device_node *target;
+ struct device *dev = NULL;
+
+ if (!of_root || !of_node_check_flag(of_root, OF_POPULATED_BUS))
+ return;
+
+ if (!np)
+ return;
+
+ of_node_get(np);
+
+ /* Find the closest ancestor that has a device associated */
+ for (target = np;
+ !of_node_is_root(target);
+ target = of_get_next_parent(target))
+ if (get_device(target->device)) {
+ dev = target->device;
+ break;
+ }
+
+ of_node_put(target);
+
+ if (!dev) {
+ pr_warn("Couldn't find a device for node '%s'\n",
+ of_node_full_name(np));
+ return;
+ }
+
+ /*
+ * Device is bound or is being probed right now. If we have bad luck
+ * and the dependency isn't ready when it's needed, deferred probe
+ * will save us.
+ */
+ if (dev->driver)
+ goto out;
+
+ /*
+ * Probing a device should cause its descendants to be probed as
+ * well, which includes the passed device node.
+ */
+ if (device_attach(dev) != 1)
+ /*
+ * This cannot be a warning for now because clock nodes have a
+ * compatible string but the clock framework doesn't follow
+ * the device/driver model yet.
+ */
+ dev_dbg(dev, "Probe failed for %s\n", of_node_full_name(np));
+
+out:
+ put_device(dev);
+}
+EXPORT_SYMBOL_GPL(of_device_probe);
diff --git a/include/linux/of_device.h b/include/linux/of_device.h
index cc7dd687a89d..da8d489e73ad 100644
--- a/include/linux/of_device.h
+++ b/include/linux/of_device.h
@@ -40,6 +40,7 @@ extern ssize_t of_device_get_modalias(struct device *dev,
extern void of_device_uevent(struct device *dev, struct kobj_uevent_env *env);
extern int of_device_uevent_modalias(struct device *dev, struct kobj_uevent_env *env);
+extern void of_device_probe(struct device_node *np);
static inline void of_device_node_put(struct device *dev)
{
@@ -84,6 +85,8 @@ static inline int of_device_uevent_modalias(struct device *dev,
return -ENODEV;
}
+static inline void of_device_probe(struct device_node *np) { }
+
static inline void of_device_node_put(struct device *dev) { }
static inline const struct of_device_id *__of_match_device(
--
2.4.3
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 10/23] drm: Probe panels on demand
[not found] ` <1442494637-3674-1-git-send-email-tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
2015-09-17 12:56 ` [PATCH v5 04/23] of: add function to allow probing a device from a OF node Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
1 sibling, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel-u79uwXL29TY76Z2rM5mHXA
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki,
linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r,
Dmitry Torokhov, devicetree-u79uwXL29TY76Z2rM5mHXA, Linus Walleij,
linux-acpi-u79uwXL29TY76Z2rM5mHXA, Arnd Bergmann, Tomeu Vizoso
When looking up a panel through its OF node, probe it if it hasn't
already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/gpu/drm/drm_panel.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/gpu/drm/drm_panel.c b/drivers/gpu/drm/drm_panel.c
index 2ef988e037b7..ad79a7b9c74d 100644
--- a/drivers/gpu/drm/drm_panel.c
+++ b/drivers/gpu/drm/drm_panel.c
@@ -23,6 +23,7 @@
#include <linux/err.h>
#include <linux/module.h>
+#include <linux/of_device.h>
#include <drm/drm_crtc.h>
#include <drm/drm_panel.h>
@@ -80,6 +81,8 @@ struct drm_panel *of_drm_find_panel(struct device_node *np)
{
struct drm_panel *panel;
+ of_device_probe(np);
+
mutex_lock(&panel_lock);
list_for_each_entry(panel, &panel_list, list) {
--
2.4.3
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 05/23] gpio: Probe GPIO drivers on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (3 preceding siblings ...)
[not found] ` <1442494637-3674-1-git-send-email-tomeu.vizoso-ZGY8ohtN/8qB+jHODAdFcQ@public.gmane.org>
@ 2015-09-17 12:56 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 06/23] gpio: Probe pinctrl devices " Tomeu Vizoso
` (16 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:56 UTC (permalink / raw)
To: linux-kernel
Cc: Tomeu Vizoso, devicetree, linux-acpi, Arnd Bergmann,
Stephen Warren, Greg Kroah-Hartman, Linus Walleij,
Dmitry Torokhov, Rafael J. Wysocki, Rob Herring,
Javier Martinez Canillas, Mark Brown, Thierry Reding, Alan Stern,
linux-arm-kernel
When looking up a gpiochip through its firmware node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/gpio/gpiolib-of.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/gpio/gpiolib-of.c b/drivers/gpio/gpiolib-of.c
index fa6e3c8823d6..9a439dab7a87 100644
--- a/drivers/gpio/gpiolib-of.c
+++ b/drivers/gpio/gpiolib-of.c
@@ -19,6 +19,7 @@
#include <linux/gpio/consumer.h>
#include <linux/of.h>
#include <linux/of_address.h>
+#include <linux/of_device.h>
#include <linux/of_gpio.h>
#include <linux/pinctrl/pinctrl.h>
#include <linux/slab.h>
@@ -95,6 +96,8 @@ struct gpio_desc *of_get_named_gpiod_flags(struct device_node *np,
return ERR_PTR(ret);
}
+ of_device_probe(gg_data.gpiospec.np);
+
gpiochip_find(&gg_data, of_gpiochip_find_and_xlate);
of_node_put(gg_data.gpiospec.np);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 06/23] gpio: Probe pinctrl devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (4 preceding siblings ...)
2015-09-17 12:56 ` [PATCH v5 05/23] gpio: Probe GPIO drivers " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 07/23] regulator: core: Remove regulator_list Tomeu Vizoso
` (15 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a pin controller through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/gpio/gpiolib-of.c | 2 ++
1 file changed, 2 insertions(+)
diff --git a/drivers/gpio/gpiolib-of.c b/drivers/gpio/gpiolib-of.c
index 9a439dab7a87..05da9a56608d 100644
--- a/drivers/gpio/gpiolib-of.c
+++ b/drivers/gpio/gpiolib-of.c
@@ -359,6 +359,8 @@ static int of_gpiochip_add_pin_range(struct gpio_chip *chip)
if (ret)
break;
+ of_device_probe(pinspec.np);
+
pctldev = of_pinctrl_get(pinspec.np);
if (!pctldev)
return -EPROBE_DEFER;
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 07/23] regulator: core: Remove regulator_list
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (5 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 06/23] gpio: Probe pinctrl devices " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-19 15:01 ` Mark Brown
2015-09-17 12:57 ` [PATCH v5 08/23] regulator: core: Drop redundant locking Tomeu Vizoso
` (14 subsequent siblings)
21 siblings, 1 reply; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
As we are already registering a device with regulator_class for each
regulator device, regulator_list is redundant and can be replaced with
calls to class_find_device() and class_for_each_device().
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/regulator/core.c | 199 +++++++++++++++++++++++++++++------------------
1 file changed, 125 insertions(+), 74 deletions(-)
diff --git a/drivers/regulator/core.c b/drivers/regulator/core.c
index b616d34a4afa..b048af9c1ae4 100644
--- a/drivers/regulator/core.c
+++ b/drivers/regulator/core.c
@@ -51,7 +51,6 @@
pr_debug("%s: " fmt, rdev_get_name(rdev), ##__VA_ARGS__)
static DEFINE_MUTEX(regulator_list_mutex);
-static LIST_HEAD(regulator_list);
static LIST_HEAD(regulator_map_list);
static LIST_HEAD(regulator_ena_gpio_list);
static LIST_HEAD(regulator_supply_alias_list);
@@ -59,6 +58,8 @@ static bool has_full_constraints;
static struct dentry *debugfs_root;
+static struct class regulator_class;
+
/*
* struct regulator_map
*
@@ -1325,6 +1326,36 @@ static void regulator_supply_alias(struct device **dev, const char **supply)
}
}
+static int of_node_match(struct device *dev, const void *data)
+{
+ return dev->of_node == data;
+}
+
+static struct regulator_dev *of_find_regulator_by_node(struct device_node *np)
+{
+ struct device *dev;
+
+ dev = class_find_device(®ulator_class, NULL, np, of_node_match);
+
+ return dev ? dev_to_rdev(dev) : NULL;
+}
+
+static int regulator_match(struct device *dev, const void *data)
+{
+ struct regulator_dev *r = dev_to_rdev(dev);
+
+ return strcmp(rdev_get_name(r), data) == 0;
+}
+
+static struct regulator_dev *regulator_lookup_by_name(const char *name)
+{
+ struct device *dev;
+
+ dev = class_find_device(®ulator_class, NULL, name, regulator_match);
+
+ return dev ? dev_to_rdev(dev) : NULL;
+}
+
static struct regulator_dev *regulator_dev_lookup(struct device *dev,
const char *supply,
int *ret)
@@ -1340,10 +1371,9 @@ static struct regulator_dev *regulator_dev_lookup(struct device *dev,
if (dev && dev->of_node) {
node = of_get_regulator(dev, supply);
if (node) {
- list_for_each_entry(r, ®ulator_list, list)
- if (r->dev.parent &&
- node == r->dev.of_node)
- return r;
+ r = of_find_regulator_by_node(node);
+ if (r)
+ return r;
*ret = -EPROBE_DEFER;
return NULL;
} else {
@@ -1361,9 +1391,9 @@ static struct regulator_dev *regulator_dev_lookup(struct device *dev,
if (dev)
devname = dev_name(dev);
- list_for_each_entry(r, ®ulator_list, list)
- if (strcmp(rdev_get_name(r), supply) == 0)
- return r;
+ r = regulator_lookup_by_name(supply);
+ if (r)
+ return r;
list_for_each_entry(map, ®ulator_map_list, list) {
/* If the mapping has a device set up it must match */
@@ -3806,8 +3836,6 @@ regulator_register(const struct regulator_desc *regulator_desc,
}
}
- list_add(&rdev->list, ®ulator_list);
-
rdev_init_debugfs(rdev);
out:
mutex_unlock(®ulator_list_mutex);
@@ -3861,6 +3889,19 @@ void regulator_unregister(struct regulator_dev *rdev)
}
EXPORT_SYMBOL_GPL(regulator_unregister);
+static int _regulator_suspend_prepare(struct device *dev, void *data)
+{
+ struct regulator_dev *rdev = dev_to_rdev(dev);
+ const suspend_state_t *state = data;
+ int ret;
+
+ mutex_lock(&rdev->mutex);
+ ret = suspend_prepare(rdev, *state);
+ mutex_unlock(&rdev->mutex);
+
+ return ret;
+}
+
/**
* regulator_suspend_prepare - prepare regulators for system wide suspend
* @state: system suspend state
@@ -3870,30 +3911,45 @@ EXPORT_SYMBOL_GPL(regulator_unregister);
*/
int regulator_suspend_prepare(suspend_state_t state)
{
- struct regulator_dev *rdev;
- int ret = 0;
-
/* ON is handled by regulator active state */
if (state == PM_SUSPEND_ON)
return -EINVAL;
- mutex_lock(®ulator_list_mutex);
- list_for_each_entry(rdev, ®ulator_list, list) {
+ return class_for_each_device(®ulator_class, NULL, &state,
+ _regulator_suspend_prepare);
+}
+EXPORT_SYMBOL_GPL(regulator_suspend_prepare);
- mutex_lock(&rdev->mutex);
- ret = suspend_prepare(rdev, state);
- mutex_unlock(&rdev->mutex);
+static int _regulator_suspend_finish(struct device *dev, void *data)
+{
+ struct regulator_dev *rdev = dev_to_rdev(dev);
+ int ret;
- if (ret < 0) {
- rdev_err(rdev, "failed to prepare\n");
- goto out;
+ mutex_lock(&rdev->mutex);
+ if (rdev->use_count > 0 || rdev->constraints->always_on) {
+ if (!_regulator_is_enabled(rdev)) {
+ ret = _regulator_do_enable(rdev);
+ if (ret)
+ dev_err(dev,
+ "Failed to resume regulator %d\n",
+ ret);
}
+ } else {
+ if (!have_full_constraints())
+ goto unlock;
+ if (!_regulator_is_enabled(rdev))
+ goto unlock;
+
+ ret = _regulator_do_disable(rdev);
+ if (ret)
+ dev_err(dev, "Failed to suspend regulator %d\n", ret);
}
-out:
- mutex_unlock(®ulator_list_mutex);
- return ret;
+unlock:
+ mutex_unlock(&rdev->mutex);
+
+ /* Keep processing regulators in spite of any errors */
+ return 0;
}
-EXPORT_SYMBOL_GPL(regulator_suspend_prepare);
/**
* regulator_suspend_finish - resume regulators from system wide suspend
@@ -3903,33 +3959,8 @@ EXPORT_SYMBOL_GPL(regulator_suspend_prepare);
*/
int regulator_suspend_finish(void)
{
- struct regulator_dev *rdev;
- int ret = 0, error;
-
- mutex_lock(®ulator_list_mutex);
- list_for_each_entry(rdev, ®ulator_list, list) {
- mutex_lock(&rdev->mutex);
- if (rdev->use_count > 0 || rdev->constraints->always_on) {
- if (!_regulator_is_enabled(rdev)) {
- error = _regulator_do_enable(rdev);
- if (error)
- ret = error;
- }
- } else {
- if (!have_full_constraints())
- goto unlock;
- if (!_regulator_is_enabled(rdev))
- goto unlock;
-
- error = _regulator_do_disable(rdev);
- if (error)
- ret = error;
- }
-unlock:
- mutex_unlock(&rdev->mutex);
- }
- mutex_unlock(®ulator_list_mutex);
- return ret;
+ return class_for_each_device(®ulator_class, NULL, NULL,
+ _regulator_suspend_finish);
}
EXPORT_SYMBOL_GPL(regulator_suspend_finish);
@@ -4049,14 +4080,35 @@ static const struct file_operations supply_map_fops = {
};
#ifdef CONFIG_DEBUG_FS
+struct summary_data {
+ struct seq_file *s;
+ struct regulator_dev *parent;
+ int level;
+};
+
+static void regulator_summary_show_subtree(struct seq_file *s,
+ struct regulator_dev *rdev,
+ int level);
+
+static int regulator_summary_show_children(struct device *dev, void *data)
+{
+ struct regulator_dev *rdev = dev_to_rdev(dev);
+ struct summary_data *summary_data = data;
+
+ if (rdev->supply && rdev->supply->rdev == summary_data->parent)
+ regulator_summary_show_subtree(summary_data->s, rdev,
+ summary_data->level + 1);
+
+ return 0;
+}
+
static void regulator_summary_show_subtree(struct seq_file *s,
struct regulator_dev *rdev,
int level)
{
- struct list_head *list = s->private;
- struct regulator_dev *child;
struct regulation_constraints *c;
struct regulator *consumer;
+ struct summary_data summary_data;
if (!rdev)
return;
@@ -4106,33 +4158,32 @@ static void regulator_summary_show_subtree(struct seq_file *s,
seq_puts(s, "\n");
}
- list_for_each_entry(child, list, list) {
- /* handle only non-root regulators supplied by current rdev */
- if (!child->supply || child->supply->rdev != rdev)
- continue;
+ summary_data.s = s;
+ summary_data.level = level;
+ summary_data.parent = rdev;
- regulator_summary_show_subtree(s, child, level + 1);
- }
+ class_for_each_device(®ulator_class, NULL, &summary_data,
+ regulator_summary_show_children);
}
-static int regulator_summary_show(struct seq_file *s, void *data)
+static int regulator_summary_show_roots(struct device *dev, void *data)
{
- struct list_head *list = s->private;
- struct regulator_dev *rdev;
-
- seq_puts(s, " regulator use open bypass voltage current min max\n");
- seq_puts(s, "-------------------------------------------------------------------------------\n");
+ struct regulator_dev *rdev = dev_to_rdev(dev);
+ struct seq_file *s = data;
- mutex_lock(®ulator_list_mutex);
+ if (!rdev->supply)
+ regulator_summary_show_subtree(s, rdev, 0);
- list_for_each_entry(rdev, list, list) {
- if (rdev->supply)
- continue;
+ return 0;
+}
- regulator_summary_show_subtree(s, rdev, 0);
- }
+static int regulator_summary_show(struct seq_file *s, void *data)
+{
+ seq_puts(s, " regulator use open bypass voltage current min max\n");
+ seq_puts(s, "-------------------------------------------------------------------------------\n");
- mutex_unlock(®ulator_list_mutex);
+ class_for_each_device(®ulator_class, NULL, s,
+ regulator_summary_show_roots);
return 0;
}
@@ -4166,7 +4217,7 @@ static int __init regulator_init(void)
&supply_map_fops);
debugfs_create_file("regulator_summary", 0444, debugfs_root,
- ®ulator_list, ®ulator_summary_fops);
+ NULL, ®ulator_summary_fops);
regulator_dummy_init();
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* Re: [PATCH v5 07/23] regulator: core: Remove regulator_list
2015-09-17 12:57 ` [PATCH v5 07/23] regulator: core: Remove regulator_list Tomeu Vizoso
@ 2015-09-19 15:01 ` Mark Brown
2015-09-20 20:32 ` Russell King - ARM Linux
2015-09-21 14:08 ` Tomeu Vizoso
0 siblings, 2 replies; 30+ messages in thread
From: Mark Brown @ 2015-09-19 15:01 UTC (permalink / raw)
To: Tomeu Vizoso
Cc: linux-kernel, Rob Herring, Stephen Warren,
Javier Martinez Canillas, Greg Kroah-Hartman, Thierry Reding,
Alan Stern, Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov,
devicetree, Linus Walleij, linux-acpi, Arnd Bergmann
[-- Attachment #1: Type: text/plain, Size: 898 bytes --]
On Thu, Sep 17, 2015 at 02:57:01PM +0200, Tomeu Vizoso wrote:
> As we are already registering a device with regulator_class for each
> regulator device, regulator_list is redundant and can be replaced with
> calls to class_find_device() and class_for_each_device().
This appears to leak references to the struct devices returned by
class_find_device() - it takes a reference before it returns so any
device found using class_find_device() needs to be released with
put_device() and I don't see any new put_device() calls in here.
> Changes in v5: None
> Changes in v4: None
> Changes in v3: None
> Changes in v2: None
This is not the case at all, this patch was newly added. If you want
to include changelogs like this in the patch description try to ensure
that they bear some relationship to reality, if they don't they are
actively harmful as they are likely to mislead or annoy the reader.
[-- Attachment #2: Digital signature --]
[-- Type: application/pgp-signature, Size: 473 bytes --]
^ permalink raw reply [flat|nested] 30+ messages in thread
* Re: [PATCH v5 07/23] regulator: core: Remove regulator_list
2015-09-19 15:01 ` Mark Brown
@ 2015-09-20 20:32 ` Russell King - ARM Linux
2015-09-21 14:08 ` Tomeu Vizoso
2015-09-21 14:08 ` Tomeu Vizoso
1 sibling, 1 reply; 30+ messages in thread
From: Russell King - ARM Linux @ 2015-09-20 20:32 UTC (permalink / raw)
To: Mark Brown
Cc: Tomeu Vizoso, devicetree, linux-acpi, Arnd Bergmann,
Stephen Warren, Greg Kroah-Hartman, Linus Walleij,
Dmitry Torokhov, Rafael J. Wysocki, linux-kernel, Rob Herring,
Javier Martinez Canillas, Thierry Reding, Alan Stern,
linux-arm-kernel
On Sat, Sep 19, 2015 at 08:01:29AM -0700, Mark Brown wrote:
> On Thu, Sep 17, 2015 at 02:57:01PM +0200, Tomeu Vizoso wrote:
> > As we are already registering a device with regulator_class for each
> > regulator device, regulator_list is redundant and can be replaced with
> > calls to class_find_device() and class_for_each_device().
>
> This appears to leak references to the struct devices returned by
> class_find_device() - it takes a reference before it returns so any
> device found using class_find_device() needs to be released with
> put_device() and I don't see any new put_device() calls in here.
When I've been fiding exactly that kind of bug in the PHY code, I've
been adding comments to the docbook function header detailing the
requirement to balance the reference. IMHO, this is a good idea,
because the more places that get it with these APIs, the more likely
people are to potentially read it.
The comment I've been putting in the phy code is:
* If successful, returns a pointer to the phy_device with the embedded
* struct device refcount incremented by one, or NULL on failure. The
* refcount must be dropped by calling phy_disconnect() or phy_detach().
which even goes as far as telling people how they should be dropping
the reference. So there should be no excuse (ignorance is not an
excuse for this!)
--
FTTC broadband for 0.8mile line: currently at 9.6Mbps down 400kbps up
according to speedtest.net.
^ permalink raw reply [flat|nested] 30+ messages in thread
* Re: [PATCH v5 07/23] regulator: core: Remove regulator_list
2015-09-20 20:32 ` Russell King - ARM Linux
@ 2015-09-21 14:08 ` Tomeu Vizoso
0 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-21 14:08 UTC (permalink / raw)
To: Russell King - ARM Linux
Cc: Mark Brown, devicetree@vger.kernel.org, linux-acpi, Arnd Bergmann,
Stephen Warren, Greg Kroah-Hartman, Linus Walleij,
Dmitry Torokhov, Rafael J. Wysocki, linux-kernel@vger.kernel.org,
Rob Herring, Javier Martinez Canillas, Thierry Reding, Alan Stern,
linux-arm-kernel@lists.infradead.org
On 20 September 2015 at 22:32, Russell King - ARM Linux
<linux@arm.linux.org.uk> wrote:
> On Sat, Sep 19, 2015 at 08:01:29AM -0700, Mark Brown wrote:
>> On Thu, Sep 17, 2015 at 02:57:01PM +0200, Tomeu Vizoso wrote:
>> > As we are already registering a device with regulator_class for each
>> > regulator device, regulator_list is redundant and can be replaced with
>> > calls to class_find_device() and class_for_each_device().
>>
>> This appears to leak references to the struct devices returned by
>> class_find_device() - it takes a reference before it returns so any
>> device found using class_find_device() needs to be released with
>> put_device() and I don't see any new put_device() calls in here.
>
> When I've been fiding exactly that kind of bug in the PHY code, I've
> been adding comments to the docbook function header detailing the
> requirement to balance the reference. IMHO, this is a good idea,
> because the more places that get it with these APIs, the more likely
> people are to potentially read it.
>
> The comment I've been putting in the phy code is:
>
> * If successful, returns a pointer to the phy_device with the embedded
> * struct device refcount incremented by one, or NULL on failure. The
> * refcount must be dropped by calling phy_disconnect() or phy_detach().
>
> which even goes as far as telling people how they should be dropping
> the reference. So there should be no excuse (ignorance is not an
> excuse for this!)
Thanks for the suggestion, I have gone with it.
Regards,
Tomeu
> --
> FTTC broadband for 0.8mile line: currently at 9.6Mbps down 400kbps up
> according to speedtest.net.
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at http://www.tux.org/lkml/
^ permalink raw reply [flat|nested] 30+ messages in thread
* Re: [PATCH v5 07/23] regulator: core: Remove regulator_list
2015-09-19 15:01 ` Mark Brown
2015-09-20 20:32 ` Russell King - ARM Linux
@ 2015-09-21 14:08 ` Tomeu Vizoso
1 sibling, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-21 14:08 UTC (permalink / raw)
To: Mark Brown
Cc: linux-kernel@vger.kernel.org, Rob Herring, Stephen Warren,
Javier Martinez Canillas, Greg Kroah-Hartman, Thierry Reding,
Alan Stern, Rafael J. Wysocki,
linux-arm-kernel@lists.infradead.org, Dmitry Torokhov,
devicetree@vger.kernel.org, Linus Walleij, linux-acpi,
Arnd Bergmann
On 19 September 2015 at 17:01, Mark Brown <broonie@kernel.org> wrote:
> On Thu, Sep 17, 2015 at 02:57:01PM +0200, Tomeu Vizoso wrote:
>> As we are already registering a device with regulator_class for each
>> regulator device, regulator_list is redundant and can be replaced with
>> calls to class_find_device() and class_for_each_device().
>
> This appears to leak references to the struct devices returned by
> class_find_device() - it takes a reference before it returns so any
> device found using class_find_device() needs to be released with
> put_device() and I don't see any new put_device() calls in here.
You are right, I have merged the subsequent patch into this one so no
refs are leaked in between.
Thanks,
Tomeu
>> Changes in v5: None
>> Changes in v4: None
>> Changes in v3: None
>> Changes in v2: None
>
> This is not the case at all, this patch was newly added. If you want
> to include changelogs like this in the patch description try to ensure
> that they bear some relationship to reality, if they don't they are
> actively harmful as they are likely to mislead or annoy the reader.
^ permalink raw reply [flat|nested] 30+ messages in thread
* [PATCH v5 08/23] regulator: core: Drop redundant locking
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (6 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 07/23] regulator: core: Remove regulator_list Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 09/23] regulator: core: Probe regulators on demand Tomeu Vizoso
` (13 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Now that we use a klist when looking up regulator devices, we can avoid
locking regulator_list_mutex if we take a reference on the regulator
device, if one is found.
This change would be useful if for example regulator devices could be
registered on demand when a driver requests them. regulator_register()
could end up being called from within _regulator_get while the lock on
regulator_list_mutex is being held, causing a deadlock.
This sequence illustrates the situation described above:
tegra_hdmi_probe
_regulator_get
regulator_dev_lookup
of_device_probe
reg_fixed_voltage_probe
regulator_register
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5:
- Use regulator_class' klist of devices instead of regulator_list to
store and lookup regulator devices.
Changes in v4:
- Take a reference to the regulator's device to prevent dangling
pointers
Changes in v3:
- Avoid unlocking the regulator device's mutex if we don't have a device
Changes in v2:
- Acquire regulator device lock before returning from
regulator_dev_lookup()
drivers/regulator/core.c | 49 +++++++++++++++++++++++++++++-------------------
1 file changed, 30 insertions(+), 19 deletions(-)
diff --git a/drivers/regulator/core.c b/drivers/regulator/core.c
index b048af9c1ae4..24245c78b58c 100644
--- a/drivers/regulator/core.c
+++ b/drivers/regulator/core.c
@@ -1328,7 +1328,7 @@ static void regulator_supply_alias(struct device **dev, const char **supply)
static int of_node_match(struct device *dev, const void *data)
{
- return dev->of_node == data;
+ return dev->of_node == data && get_device(dev);
}
static struct regulator_dev *of_find_regulator_by_node(struct device_node *np)
@@ -1344,7 +1344,7 @@ static int regulator_match(struct device *dev, const void *data)
{
struct regulator_dev *r = dev_to_rdev(dev);
- return strcmp(rdev_get_name(r), data) == 0;
+ return strcmp(rdev_get_name(r), data) == 0 && get_device(dev);
}
static struct regulator_dev *regulator_lookup_by_name(const char *name)
@@ -1395,16 +1395,20 @@ static struct regulator_dev *regulator_dev_lookup(struct device *dev,
if (r)
return r;
+ mutex_lock(®ulator_list_mutex);
list_for_each_entry(map, ®ulator_map_list, list) {
/* If the mapping has a device set up it must match */
if (map->dev_name &&
(!devname || strcmp(map->dev_name, devname)))
continue;
- if (strcmp(map->supply, supply) == 0)
+ if (strcmp(map->supply, supply) == 0 &&
+ get_device(&map->regulator->dev)) {
+ mutex_unlock(®ulator_list_mutex);
return map->regulator;
+ }
}
-
+ mutex_unlock(®ulator_list_mutex);
return NULL;
}
@@ -1435,6 +1439,7 @@ static int regulator_resolve_supply(struct regulator_dev *rdev)
if (!r) {
if (have_full_constraints()) {
r = dummy_regulator_rdev;
+ get_device(&r->dev);
} else {
dev_err(dev, "Failed to resolve %s-supply for %s\n",
rdev->supply_name, rdev->desc->name);
@@ -1444,12 +1449,16 @@ static int regulator_resolve_supply(struct regulator_dev *rdev)
/* Recursively resolve the supply of the supply */
ret = regulator_resolve_supply(r);
- if (ret < 0)
+ if (ret < 0) {
+ put_device(&r->dev);
return ret;
+ }
ret = set_supply(rdev, r);
- if (ret < 0)
+ if (ret < 0) {
+ put_device(&r->dev);
return ret;
+ }
/* Cascade always-on state to supply */
if (_regulator_is_enabled(rdev) && rdev->supply) {
@@ -1485,8 +1494,6 @@ static struct regulator *_regulator_get(struct device *dev, const char *id,
else
ret = -EPROBE_DEFER;
- mutex_lock(®ulator_list_mutex);
-
rdev = regulator_dev_lookup(dev, id, &ret);
if (rdev)
goto found;
@@ -1498,7 +1505,7 @@ static struct regulator *_regulator_get(struct device *dev, const char *id,
* succeed, so, quit with appropriate error value
*/
if (ret && ret != -ENODEV)
- goto out;
+ return regulator;
if (!devname)
devname = "deviceless";
@@ -1512,40 +1519,46 @@ static struct regulator *_regulator_get(struct device *dev, const char *id,
devname, id);
rdev = dummy_regulator_rdev;
+ get_device(&rdev->dev);
goto found;
/* Don't log an error when called from regulator_get_optional() */
} else if (!have_full_constraints() || exclusive) {
dev_warn(dev, "dummy supplies not allowed\n");
}
- mutex_unlock(®ulator_list_mutex);
return regulator;
found:
if (rdev->exclusive) {
regulator = ERR_PTR(-EPERM);
- goto out;
+ put_device(&rdev->dev);
+ return regulator;
}
if (exclusive && rdev->open_count) {
regulator = ERR_PTR(-EBUSY);
- goto out;
+ put_device(&rdev->dev);
+ return regulator;
}
ret = regulator_resolve_supply(rdev);
if (ret < 0) {
regulator = ERR_PTR(ret);
- goto out;
+ put_device(&rdev->dev);
+ return regulator;
}
- if (!try_module_get(rdev->owner))
- goto out;
+ if (!try_module_get(rdev->owner)) {
+ put_device(&rdev->dev);
+ return regulator;
+ }
regulator = create_regulator(rdev, dev, id);
if (regulator == NULL) {
regulator = ERR_PTR(-ENOMEM);
+ put_device(&rdev->dev);
module_put(rdev->owner);
- goto out;
+ return regulator;
}
rdev->open_count++;
@@ -1559,9 +1572,6 @@ found:
rdev->use_count = 0;
}
-out:
- mutex_unlock(®ulator_list_mutex);
-
return regulator;
}
@@ -1659,6 +1669,7 @@ static void _regulator_put(struct regulator *regulator)
rdev->open_count--;
rdev->exclusive = 0;
+ put_device(&rdev->dev);
mutex_unlock(&rdev->mutex);
kfree(regulator->supply_name);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 09/23] regulator: core: Probe regulators on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (7 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 08/23] regulator: core: Drop redundant locking Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 11/23] drm/tegra: Probe dpaux devices " Tomeu Vizoso
` (12 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a regulator through its OF node, probe it if it hasn't
already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/regulator/core.c | 2 ++
1 file changed, 2 insertions(+)
diff --git a/drivers/regulator/core.c b/drivers/regulator/core.c
index 24245c78b58c..54d83974ba5c 100644
--- a/drivers/regulator/core.c
+++ b/drivers/regulator/core.c
@@ -26,6 +26,7 @@
#include <linux/gpio.h>
#include <linux/gpio/consumer.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/regmap.h>
#include <linux/regulator/of_regulator.h>
#include <linux/regulator/consumer.h>
@@ -1371,6 +1372,7 @@ static struct regulator_dev *regulator_dev_lookup(struct device *dev,
if (dev && dev->of_node) {
node = of_get_regulator(dev, supply);
if (node) {
+ of_device_probe(node);
r = of_find_regulator_by_node(node);
if (r)
return r;
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 11/23] drm/tegra: Probe dpaux devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (8 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 09/23] regulator: core: Probe regulators on demand Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 12/23] i2c: core: Probe i2c adapters and " Tomeu Vizoso
` (11 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a dpaux device through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/gpu/drm/tegra/dpaux.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/gpu/drm/tegra/dpaux.c b/drivers/gpu/drm/tegra/dpaux.c
index 6aecb6647313..40bdc2a98548 100644
--- a/drivers/gpu/drm/tegra/dpaux.c
+++ b/drivers/gpu/drm/tegra/dpaux.c
@@ -12,6 +12,7 @@
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/of_gpio.h>
+#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/reset.h>
#include <linux/regulator/consumer.h>
@@ -440,6 +441,8 @@ struct tegra_dpaux *tegra_dpaux_find_by_of_node(struct device_node *np)
{
struct tegra_dpaux *dpaux;
+ of_device_probe(np);
+
mutex_lock(&dpaux_lock);
list_for_each_entry(dpaux, &dpaux_list, list)
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 12/23] i2c: core: Probe i2c adapters and devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (9 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 11/23] drm/tegra: Probe dpaux devices " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 13/23] pwm: Probe PWM chip " Tomeu Vizoso
` (10 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up an i2c adapter or device through its OF node, probe it
if it hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/i2c/i2c-core.c | 4 ++++
1 file changed, 4 insertions(+)
diff --git a/drivers/i2c/i2c-core.c b/drivers/i2c/i2c-core.c
index 5f89f1e3c2f2..02da3acbbd35 100644
--- a/drivers/i2c/i2c-core.c
+++ b/drivers/i2c/i2c-core.c
@@ -1413,6 +1413,8 @@ struct i2c_client *of_find_i2c_device_by_node(struct device_node *node)
struct device *dev;
struct i2c_client *client;
+ of_device_probe(node);
+
dev = bus_find_device(&i2c_bus_type, NULL, node, of_dev_node_match);
if (!dev)
return NULL;
@@ -1431,6 +1433,8 @@ struct i2c_adapter *of_find_i2c_adapter_by_node(struct device_node *node)
struct device *dev;
struct i2c_adapter *adapter;
+ of_device_probe(node);
+
dev = bus_find_device(&i2c_bus_type, NULL, node, of_dev_node_match);
if (!dev)
return NULL;
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 13/23] pwm: Probe PWM chip devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (10 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 12/23] i2c: core: Probe i2c adapters and " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 14/23] backlight: Probe backlight " Tomeu Vizoso
` (9 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Tomeu Vizoso, devicetree, linux-acpi, Arnd Bergmann,
Stephen Warren, Greg Kroah-Hartman, Linus Walleij,
Dmitry Torokhov, Rafael J. Wysocki, Rob Herring,
Javier Martinez Canillas, Mark Brown, Thierry Reding, Alan Stern,
linux-arm-kernel
When looking up a PWM chip through its OF node, probe it if it hasn't
already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/pwm/core.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/pwm/core.c b/drivers/pwm/core.c
index 3f9df3ea3350..794a923df0d8 100644
--- a/drivers/pwm/core.c
+++ b/drivers/pwm/core.c
@@ -29,6 +29,7 @@
#include <linux/device.h>
#include <linux/debugfs.h>
#include <linux/seq_file.h>
+#include <linux/of_device.h>
#include <dt-bindings/pwm/pwm.h>
@@ -516,6 +517,8 @@ static struct pwm_chip *of_node_to_pwmchip(struct device_node *np)
{
struct pwm_chip *chip;
+ of_device_probe(np);
+
mutex_lock(&pwm_lock);
list_for_each_entry(chip, &pwm_chips, list)
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 14/23] backlight: Probe backlight devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (11 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 13/23] pwm: Probe PWM chip " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 15/23] usb: phy: Probe phy " Tomeu Vizoso
` (8 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a backlight device through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/video/backlight/backlight.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/video/backlight/backlight.c b/drivers/video/backlight/backlight.c
index bddc8b17a4d8..9bcdc16eacdf 100644
--- a/drivers/video/backlight/backlight.c
+++ b/drivers/video/backlight/backlight.c
@@ -16,6 +16,7 @@
#include <linux/err.h>
#include <linux/fb.h>
#include <linux/slab.h>
+#include <linux/of_device.h>
#ifdef CONFIG_PMAC_BACKLIGHT
#include <asm/backlight.h>
@@ -559,6 +560,8 @@ struct backlight_device *of_find_backlight_by_node(struct device_node *node)
{
struct device *dev;
+ of_device_probe(node);
+
dev = class_find_device(backlight_class, NULL, node, of_parent_match);
return dev ? to_backlight_device(dev) : NULL;
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 15/23] usb: phy: Probe phy devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (12 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 14/23] backlight: Probe backlight " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 16/23] clk: Probe clk providers " Tomeu Vizoso
` (7 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a phy through its OF node, probe it if it hasn't
already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/usb/phy/phy.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/usb/phy/phy.c b/drivers/usb/phy/phy.c
index 98f75d2842b7..fb0b650bb494 100644
--- a/drivers/usb/phy/phy.c
+++ b/drivers/usb/phy/phy.c
@@ -15,6 +15,7 @@
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/usb/phy.h>
@@ -196,6 +197,8 @@ struct usb_phy *devm_usb_get_phy_by_node(struct device *dev,
goto err0;
}
+ of_device_probe(node);
+
spin_lock_irqsave(&phy_lock, flags);
phy = __of_usb_find_phy(node);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 16/23] clk: Probe clk providers on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (13 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 15/23] usb: phy: Probe phy " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 17/23] pinctrl: Probe pinctrl devices " Tomeu Vizoso
` (6 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a clock through its OF node, probe it if it hasn't
already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/clk/clk.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/clk/clk.c b/drivers/clk/clk.c
index 43e2c3ad6c31..e5fe02a11c36 100644
--- a/drivers/clk/clk.c
+++ b/drivers/clk/clk.c
@@ -19,6 +19,7 @@
#include <linux/list.h>
#include <linux/slab.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/device.h>
#include <linux/init.h>
#include <linux/sched.h>
@@ -3004,6 +3005,8 @@ struct clk *__of_clk_get_from_provider(struct of_phandle_args *clkspec,
if (!clkspec)
return ERR_PTR(-EINVAL);
+ of_device_probe(clkspec->np);
+
/* Check if we have such a provider in our array */
mutex_lock(&of_clk_mutex);
list_for_each_entry(provider, &of_clk_providers, link) {
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 17/23] pinctrl: Probe pinctrl devices on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (14 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 16/23] clk: Probe clk providers " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 18/23] phy: core: Probe phy providers " Tomeu Vizoso
` (5 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a pin controller through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/pinctrl/devicetree.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/pinctrl/devicetree.c b/drivers/pinctrl/devicetree.c
index fe04e748dfe4..f5340b8e1dbe 100644
--- a/drivers/pinctrl/devicetree.c
+++ b/drivers/pinctrl/devicetree.c
@@ -18,6 +18,7 @@
#include <linux/device.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/pinctrl/pinctrl.h>
#include <linux/slab.h>
@@ -110,6 +111,8 @@ static int dt_to_map_one_config(struct pinctrl *p, const char *statename,
struct pinctrl_map *map;
unsigned num_maps;
+ of_device_probe(np_config);
+
/* Find the pin controller containing np_config */
np_pctldev = of_node_get(np_config);
for (;;) {
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 18/23] phy: core: Probe phy providers on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (15 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 17/23] pinctrl: Probe pinctrl devices " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 19/23] dma: of: Probe DMA controllers " Tomeu Vizoso
` (4 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a phy provider through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/phy/phy-core.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/phy/phy-core.c b/drivers/phy/phy-core.c
index fc48fac003a6..94e90031d7f3 100644
--- a/drivers/phy/phy-core.c
+++ b/drivers/phy/phy-core.c
@@ -18,6 +18,7 @@
#include <linux/device.h>
#include <linux/slab.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/phy/phy.h>
#include <linux/idr.h>
#include <linux/pm_runtime.h>
@@ -363,6 +364,8 @@ static struct phy *_of_phy_get(struct device_node *np, int index)
if (ret)
return ERR_PTR(-ENODEV);
+ of_device_probe(args.np);
+
mutex_lock(&phy_provider_mutex);
phy_provider = of_phy_provider_lookup(args.np);
if (IS_ERR(phy_provider) || !try_module_get(phy_provider->owner)) {
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 19/23] dma: of: Probe DMA controllers on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (16 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 18/23] phy: core: Probe phy providers " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 20/23] power-supply: Probe power supplies " Tomeu Vizoso
` (3 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a DMA controller through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/dma/of-dma.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/dma/of-dma.c b/drivers/dma/of-dma.c
index 1e1f2986eba8..e899832f7df3 100644
--- a/drivers/dma/of-dma.c
+++ b/drivers/dma/of-dma.c
@@ -16,6 +16,7 @@
#include <linux/mutex.h>
#include <linux/slab.h>
#include <linux/of.h>
+#include <linux/of_device.h>
#include <linux/of_dma.h>
static LIST_HEAD(of_dma_list);
@@ -263,6 +264,8 @@ struct dma_chan *of_dma_request_slave_channel(struct device_node *np,
if (of_dma_match_channel(np, name, i, &dma_spec))
continue;
+ of_device_probe(dma_spec.np);
+
mutex_lock(&of_dma_lock);
ofdma = of_dma_find_controller(&dma_spec);
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 20/23] power-supply: Probe power supplies on demand
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (17 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 19/23] dma: of: Probe DMA controllers " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 21/23] driver core: Allow deferring probes until late init Tomeu Vizoso
` (2 subsequent siblings)
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
When looking up a power supply through its OF node, probe it if it
hasn't already.
The goal is to reduce deferred probes to a minimum, as it makes it very
cumbersome to find out why a device failed to probe, and can introduce
very big delays in when a critical device is probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4: None
Changes in v3: None
Changes in v2: None
drivers/power/power_supply_core.c | 3 +++
1 file changed, 3 insertions(+)
diff --git a/drivers/power/power_supply_core.c b/drivers/power/power_supply_core.c
index 456987c88baa..80bc89f4ae89 100644
--- a/drivers/power/power_supply_core.c
+++ b/drivers/power/power_supply_core.c
@@ -19,6 +19,7 @@
#include <linux/err.h>
#include <linux/power_supply.h>
#include <linux/thermal.h>
+#include <linux/of_device.h>
#include "power_supply.h"
/* exported for the APM Power driver, APM emulation */
@@ -206,6 +207,8 @@ static int power_supply_find_supply_from_node(struct device_node *supply_node)
{
int error;
+ of_device_probe(supply_node);
+
/*
* class_for_each_device() either returns its own errors or values
* returned by __power_supply_find_supply_from_node().
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 21/23] driver core: Allow deferring probes until late init
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (18 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 20/23] power-supply: Probe power supplies " Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 22/23] driver core: Start processing deferred probes earlier Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 23/23] of/platform: Defer probes of registered devices Tomeu Vizoso
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Add a field to struct device that instructs the device-driver core to
defer the probe of this device until the late_initcall level.
By letting all built-in drivers to register before starting to probe, we
can avoid any deferred probes by probing dependencies on demand.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4:
- Add Kconfig DELAY_DEVICE_PROBES to allow disabling delayed probing in
machines with initcalls that depend on devices probing at a given time.
Changes in v3: None
Changes in v2: None
drivers/base/Kconfig | 18 ++++++++++++++++++
drivers/base/dd.c | 7 +++++++
include/linux/device.h | 2 ++
3 files changed, 27 insertions(+)
diff --git a/drivers/base/Kconfig b/drivers/base/Kconfig
index 98504ec99c7d..44b5d33b1f49 100644
--- a/drivers/base/Kconfig
+++ b/drivers/base/Kconfig
@@ -324,4 +324,22 @@ config CMA_ALIGNMENT
endif
+config DELAY_DEVICE_PROBES
+ bool "Allow delaying the probe of some devices"
+ default y
+ help
+ Devices can be matched to a driver and probed from the moment they
+ are registered, but early during boot their probes are likely to be
+ deferred because some dependency isn't available yet because most
+ drivers haven't been registered yet.
+
+ Enabling this option allows the device registration code to delay the
+ probing of a specific device until device_initcall_sync, when all
+ built-in drivers have been registered already.
+
+ In some platforms there may be implicit assumptions about when some
+ devices are probed, so enabling this option could cause problems there.
+
+ If unsure, say Y here.
+
endmenu
diff --git a/drivers/base/dd.c b/drivers/base/dd.c
index 5caa8478404d..300c59562fc9 100644
--- a/drivers/base/dd.c
+++ b/drivers/base/dd.c
@@ -417,6 +417,13 @@ int driver_probe_device(struct device_driver *drv, struct device *dev)
if (!device_is_registered(dev))
return -ENODEV;
+#if IS_ENABLED(CONFIG_DELAY_DEVICE_PROBES)
+ if (!driver_deferred_probe_enable && dev->probe_late) {
+ driver_deferred_probe_add(dev);
+ return 0;
+ }
+#endif
+
pr_debug("bus: '%s': %s: matched device %s with driver %s\n",
drv->bus->name, __func__, dev_name(dev), drv->name);
diff --git a/include/linux/device.h b/include/linux/device.h
index d8be07bc9c3f..fe5c50ffd7d3 100644
--- a/include/linux/device.h
+++ b/include/linux/device.h
@@ -748,6 +748,7 @@ struct device_dma_parameters {
*
* @offline_disabled: If set, the device is permanently online.
* @offline: Set after successful invocation of bus type's .offline().
+ * @probe_late: If set, device will be probed in the late initcall level.
*
* At the lowest level, every device in a Linux system is represented by an
* instance of struct device. The device structure contains the information
@@ -832,6 +833,7 @@ struct device {
bool offline_disabled:1;
bool offline:1;
+ bool probe_late:1;
};
static inline struct device *kobj_to_dev(struct kobject *kobj)
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 22/23] driver core: Start processing deferred probes earlier
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (19 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 21/23] driver core: Allow deferring probes until late init Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
2015-09-17 12:57 ` [PATCH v5 23/23] of/platform: Defer probes of registered devices Tomeu Vizoso
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Some initcalls in the late level assume that some devices will have
already probed without explicitly checking for that.
After the recent move to defer most device probes when they are
registered, pressure increased in the late initcall level.
By starting the processing of the deferred queue in device_initcall_sync
we increase the chances that the initcalls mentioned before will find
the devices they depend on to have already probed.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4:
- Start processing deferred probes in device_initcall_sync
Changes in v3: None
Changes in v2: None
drivers/base/dd.c | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/drivers/base/dd.c b/drivers/base/dd.c
index 300c59562fc9..5d8572be9b63 100644
--- a/drivers/base/dd.c
+++ b/drivers/base/dd.c
@@ -176,7 +176,7 @@ static void driver_deferred_probe_trigger(void)
*
* We don't want to get in the way when the bulk of drivers are getting probed.
* Instead, this initcall makes sure that deferred probing is delayed until
- * late_initcall time.
+ * device_initcall_sync time.
*/
static int deferred_probe_initcall(void)
{
@@ -190,7 +190,7 @@ static int deferred_probe_initcall(void)
flush_workqueue(deferred_wq);
return 0;
}
-late_initcall(deferred_probe_initcall);
+device_initcall_sync(deferred_probe_initcall);
static void driver_bound(struct device *dev)
{
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread
* [PATCH v5 23/23] of/platform: Defer probes of registered devices
2015-09-17 12:56 [PATCH v5 0/23] On-demand device probing Tomeu Vizoso
` (20 preceding siblings ...)
2015-09-17 12:57 ` [PATCH v5 22/23] driver core: Start processing deferred probes earlier Tomeu Vizoso
@ 2015-09-17 12:57 ` Tomeu Vizoso
21 siblings, 0 replies; 30+ messages in thread
From: Tomeu Vizoso @ 2015-09-17 12:57 UTC (permalink / raw)
To: linux-kernel
Cc: Rob Herring, Stephen Warren, Javier Martinez Canillas,
Greg Kroah-Hartman, Mark Brown, Thierry Reding, Alan Stern,
Rafael J. Wysocki, linux-arm-kernel, Dmitry Torokhov, devicetree,
Linus Walleij, linux-acpi, Arnd Bergmann, Tomeu Vizoso
Instead of trying to match and probe platform and AMBA devices right
after each is registered, delay their probes until device_initcall_sync.
This means that devices will start probing once all built-in drivers
have registered, and after all platform and AMBA devices from the DT
have been registered already.
This allows us to prevent deferred probes by probing dependencies on
demand.
Signed-off-by: Tomeu Vizoso <tomeu.vizoso@collabora.com>
---
Changes in v5: None
Changes in v4:
- Also defer probes of AMBA devices registered from the DT as they can
also request resources.
Changes in v3: None
Changes in v2: None
drivers/of/platform.c | 11 ++++++++---
1 file changed, 8 insertions(+), 3 deletions(-)
diff --git a/drivers/of/platform.c b/drivers/of/platform.c
index 408d89f1d124..7b33e0369374 100644
--- a/drivers/of/platform.c
+++ b/drivers/of/platform.c
@@ -164,7 +164,8 @@ static struct platform_device *of_platform_device_create_pdata(
struct device_node *np,
const char *bus_id,
void *platform_data,
- struct device *parent)
+ struct device *parent,
+ bool probe_late)
{
struct platform_device *dev;
@@ -178,6 +179,7 @@ static struct platform_device *of_platform_device_create_pdata(
dev->dev.bus = &platform_bus_type;
dev->dev.platform_data = platform_data;
+ dev->dev.probe_late = probe_late;
of_dma_configure(&dev->dev, dev->dev.of_node);
of_msi_configure(&dev->dev, dev->dev.of_node);
@@ -209,7 +211,8 @@ struct platform_device *of_platform_device_create(struct device_node *np,
const char *bus_id,
struct device *parent)
{
- return of_platform_device_create_pdata(np, bus_id, NULL, parent);
+ return of_platform_device_create_pdata(np, bus_id, NULL, parent,
+ false);
}
EXPORT_SYMBOL(of_platform_device_create);
@@ -240,6 +243,7 @@ static struct amba_device *of_amba_device_create(struct device_node *node,
dev->dev.of_node = of_node_get(node);
dev->dev.parent = parent ? : &platform_bus;
dev->dev.platform_data = platform_data;
+ dev->dev.probe_late = true;
if (bus_id)
dev_set_name(&dev->dev, "%s", bus_id);
else
@@ -358,7 +362,8 @@ static int of_platform_bus_create(struct device_node *bus,
return 0;
}
- dev = of_platform_device_create_pdata(bus, bus_id, platform_data, parent);
+ dev = of_platform_device_create_pdata(bus, bus_id, platform_data,
+ parent, true);
if (!dev || !of_match_node(matches, bus))
return 0;
--
2.4.3
^ permalink raw reply related [flat|nested] 30+ messages in thread