diff options
Diffstat (limited to 'drivers/base/core.c')
| -rw-r--r-- | drivers/base/core.c | 48 | 
1 files changed, 40 insertions, 8 deletions
| diff --git a/drivers/base/core.c b/drivers/base/core.c index a4c853411a6b..048ff98dbdfd 100644 --- a/drivers/base/core.c +++ b/drivers/base/core.c @@ -26,7 +26,6 @@  #include <linux/of.h>  #include <linux/of_device.h>  #include <linux/pm_runtime.h> -#include <linux/rcupdate.h>  #include <linux/sched/mm.h>  #include <linux/sched/signal.h>  #include <linux/slab.h> @@ -2634,7 +2633,6 @@ static const char *dev_uevent_name(const struct kobject *kobj)  static int dev_uevent(const struct kobject *kobj, struct kobj_uevent_env *env)  {  	const struct device *dev = kobj_to_dev(kobj); -	struct device_driver *driver;  	int retval = 0;  	/* add device node properties if present */ @@ -2663,12 +2661,8 @@ static int dev_uevent(const struct kobject *kobj, struct kobj_uevent_env *env)  	if (dev->type && dev->type->name)  		add_uevent_var(env, "DEVTYPE=%s", dev->type->name); -	/* Synchronize with module_remove_driver() */ -	rcu_read_lock(); -	driver = READ_ONCE(dev->driver); -	if (driver) -		add_uevent_var(env, "DRIVER=%s", driver->name); -	rcu_read_unlock(); +	if (dev->driver) +		add_uevent_var(env, "DRIVER=%s", dev->driver->name);  	/* Add common DT information about the device */  	of_device_uevent(dev, env); @@ -2738,8 +2732,11 @@ static ssize_t uevent_show(struct device *dev, struct device_attribute *attr,  	if (!env)  		return -ENOMEM; +	/* Synchronize with really_probe() */ +	device_lock(dev);  	/* let the kset specific function add its keys */  	retval = kset->uevent_ops->uevent(&dev->kobj, env); +	device_unlock(dev);  	if (retval)  		goto out; @@ -4038,6 +4035,41 @@ int device_for_each_child_reverse(struct device *parent, void *data,  EXPORT_SYMBOL_GPL(device_for_each_child_reverse);  /** + * device_for_each_child_reverse_from - device child iterator in reversed order. + * @parent: parent struct device. + * @from: optional starting point in child list + * @fn: function to be called for each device. + * @data: data for the callback. + * + * Iterate over @parent's child devices, starting at @from, and call @fn + * for each, passing it @data. This helper is identical to + * device_for_each_child_reverse() when @from is NULL. + * + * @fn is checked each iteration. If it returns anything other than 0, + * iteration stop and that value is returned to the caller of + * device_for_each_child_reverse_from(); + */ +int device_for_each_child_reverse_from(struct device *parent, +				       struct device *from, const void *data, +				       int (*fn)(struct device *, const void *)) +{ +	struct klist_iter i; +	struct device *child; +	int error = 0; + +	if (!parent->p) +		return 0; + +	klist_iter_init_node(&parent->p->klist_children, &i, +			     (from ? &from->p->knode_parent : NULL)); +	while ((child = prev_device(&i)) && !error) +		error = fn(child, data); +	klist_iter_exit(&i); +	return error; +} +EXPORT_SYMBOL_GPL(device_for_each_child_reverse_from); + +/**   * device_find_child - device iterator for locating a particular device.   * @parent: parent struct device   * @match: Callback function to check device | 
