提交 40f9244f 编写于 作者: M Mark Brown 提交者: Liam Girdwood

regulator: Allow consumer supplies to be set up with dev_name()

Follow the approach suggested by Russell King and implemented by him in
the clkdev API and allow consumer device supply mapings to be set up
using the dev_name() for the consumer instead of the struct device.
In order to avoid making existing machines instabuggy and creating merge
issues the use of struct device is still supported for the time being.

This resolves problems working with buses such as I2C which make the
struct device available late providing that the final device name is
known, which is the case for most embedded systems with fixed setups.

Consumers must still use the struct device when calling regulator_get().
Signed-off-by: NMark Brown <broonie@opensource.wolfsonmicro.com>
Signed-off-by: NLiam Girdwood <lrg@slimlogic.co.uk>
上级 a5d2abce
...@@ -37,7 +37,7 @@ static int has_full_constraints; ...@@ -37,7 +37,7 @@ static int has_full_constraints;
*/ */
struct regulator_map { struct regulator_map {
struct list_head list; struct list_head list;
struct device *dev; const char *dev_name; /* The dev_name() for the consumer */
const char *supply; const char *supply;
struct regulator_dev *regulator; struct regulator_dev *regulator;
}; };
...@@ -857,23 +857,33 @@ static int set_supply(struct regulator_dev *rdev, ...@@ -857,23 +857,33 @@ static int set_supply(struct regulator_dev *rdev,
* set_consumer_device_supply: Bind a regulator to a symbolic supply * set_consumer_device_supply: Bind a regulator to a symbolic supply
* @rdev: regulator source * @rdev: regulator source
* @consumer_dev: device the supply applies to * @consumer_dev: device the supply applies to
* @consumer_dev_name: dev_name() string for device supply applies to
* @supply: symbolic name for supply * @supply: symbolic name for supply
* *
* Allows platform initialisation code to map physical regulator * Allows platform initialisation code to map physical regulator
* sources to symbolic names for supplies for use by devices. Devices * sources to symbolic names for supplies for use by devices. Devices
* should use these symbolic names to request regulators, avoiding the * should use these symbolic names to request regulators, avoiding the
* need to provide board-specific regulator names as platform data. * need to provide board-specific regulator names as platform data.
*
* Only one of consumer_dev and consumer_dev_name may be specified.
*/ */
static int set_consumer_device_supply(struct regulator_dev *rdev, static int set_consumer_device_supply(struct regulator_dev *rdev,
struct device *consumer_dev, const char *supply) struct device *consumer_dev, const char *consumer_dev_name,
const char *supply)
{ {
struct regulator_map *node; struct regulator_map *node;
if (consumer_dev && consumer_dev_name)
return -EINVAL;
if (!consumer_dev_name && consumer_dev)
consumer_dev_name = dev_name(consumer_dev);
if (supply == NULL) if (supply == NULL)
return -EINVAL; return -EINVAL;
list_for_each_entry(node, &regulator_map_list, list) { list_for_each_entry(node, &regulator_map_list, list) {
if (consumer_dev != node->dev) if (consumer_dev_name != node->dev_name)
continue; continue;
if (strcmp(node->supply, supply) != 0) if (strcmp(node->supply, supply) != 0)
continue; continue;
...@@ -891,26 +901,39 @@ static int set_consumer_device_supply(struct regulator_dev *rdev, ...@@ -891,26 +901,39 @@ static int set_consumer_device_supply(struct regulator_dev *rdev,
return -ENOMEM; return -ENOMEM;
node->regulator = rdev; node->regulator = rdev;
node->dev = consumer_dev; node->dev_name = kstrdup(consumer_dev_name, GFP_KERNEL);
node->supply = supply; node->supply = supply;
if (node->dev_name == NULL) {
kfree(node);
return -ENOMEM;
}
list_add(&node->list, &regulator_map_list); list_add(&node->list, &regulator_map_list);
return 0; return 0;
} }
static void unset_consumer_device_supply(struct regulator_dev *rdev, static void unset_consumer_device_supply(struct regulator_dev *rdev,
struct device *consumer_dev) const char *consumer_dev_name, struct device *consumer_dev)
{ {
struct regulator_map *node, *n; struct regulator_map *node, *n;
if (consumer_dev && !consumer_dev_name)
consumer_dev_name = dev_name(consumer_dev);
list_for_each_entry_safe(node, n, &regulator_map_list, list) { list_for_each_entry_safe(node, n, &regulator_map_list, list) {
if (rdev == node->regulator && if (rdev != node->regulator)
consumer_dev == node->dev) { continue;
if (consumer_dev_name && node->dev_name &&
strcmp(consumer_dev_name, node->dev_name))
continue;
list_del(&node->list); list_del(&node->list);
kfree(node->dev_name);
kfree(node); kfree(node);
return; return;
} }
}
} }
static void unset_regulator_supplies(struct regulator_dev *rdev) static void unset_regulator_supplies(struct regulator_dev *rdev)
...@@ -920,6 +943,7 @@ static void unset_regulator_supplies(struct regulator_dev *rdev) ...@@ -920,6 +943,7 @@ static void unset_regulator_supplies(struct regulator_dev *rdev)
list_for_each_entry_safe(node, n, &regulator_map_list, list) { list_for_each_entry_safe(node, n, &regulator_map_list, list) {
if (rdev == node->regulator) { if (rdev == node->regulator) {
list_del(&node->list); list_del(&node->list);
kfree(node->dev_name);
kfree(node); kfree(node);
return; return;
} }
...@@ -1019,17 +1043,25 @@ struct regulator *regulator_get(struct device *dev, const char *id) ...@@ -1019,17 +1043,25 @@ struct regulator *regulator_get(struct device *dev, const char *id)
struct regulator_dev *rdev; struct regulator_dev *rdev;
struct regulator_map *map; struct regulator_map *map;
struct regulator *regulator = ERR_PTR(-ENODEV); struct regulator *regulator = ERR_PTR(-ENODEV);
const char *devname = NULL;
if (id == NULL) { if (id == NULL) {
printk(KERN_ERR "regulator: get() with no identifier\n"); printk(KERN_ERR "regulator: get() with no identifier\n");
return regulator; return regulator;
} }
if (dev)
devname = dev_name(dev);
mutex_lock(&regulator_list_mutex); mutex_lock(&regulator_list_mutex);
list_for_each_entry(map, &regulator_map_list, list) { list_for_each_entry(map, &regulator_map_list, list) {
if (dev == map->dev && /* If the mapping has a device set up it must match */
strcmp(map->supply, id) == 0) { if (map->dev_name &&
(!devname || strcmp(map->dev_name, devname)))
continue;
if (strcmp(map->supply, id) == 0) {
rdev = map->regulator; rdev = map->regulator;
goto found; goto found;
} }
...@@ -2091,10 +2123,12 @@ struct regulator_dev *regulator_register(struct regulator_desc *regulator_desc, ...@@ -2091,10 +2123,12 @@ struct regulator_dev *regulator_register(struct regulator_desc *regulator_desc,
for (i = 0; i < init_data->num_consumer_supplies; i++) { for (i = 0; i < init_data->num_consumer_supplies; i++) {
ret = set_consumer_device_supply(rdev, ret = set_consumer_device_supply(rdev,
init_data->consumer_supplies[i].dev, init_data->consumer_supplies[i].dev,
init_data->consumer_supplies[i].dev_name,
init_data->consumer_supplies[i].supply); init_data->consumer_supplies[i].supply);
if (ret < 0) { if (ret < 0) {
for (--i; i >= 0; i--) for (--i; i >= 0; i--)
unset_consumer_device_supply(rdev, unset_consumer_device_supply(rdev,
init_data->consumer_supplies[i].dev_name,
init_data->consumer_supplies[i].dev); init_data->consumer_supplies[i].dev);
goto scrub; goto scrub;
} }
......
...@@ -126,13 +126,18 @@ struct regulation_constraints { ...@@ -126,13 +126,18 @@ struct regulation_constraints {
/** /**
* struct regulator_consumer_supply - supply -> device mapping * struct regulator_consumer_supply - supply -> device mapping
* *
* This maps a supply name to a device. * This maps a supply name to a device. Only one of dev or dev_name
* can be specified. Use of dev_name allows support for buses which
* make struct device available late such as I2C and is the preferred
* form.
* *
* @dev: Device structure for the consumer. * @dev: Device structure for the consumer.
* @dev_name: Result of dev_name() for the consumer.
* @supply: Name for the supply. * @supply: Name for the supply.
*/ */
struct regulator_consumer_supply { struct regulator_consumer_supply {
struct device *dev; /* consumer */ struct device *dev; /* consumer */
const char *dev_name; /* dev_name() for consumer */
const char *supply; /* consumer supply - e.g. "vcc" */ const char *supply; /* consumer supply - e.g. "vcc" */
}; };
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册