mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2026-05-16 05:31:37 -04:00
reset: gpio: add a devlink between reset-gpio and its consumer
The device that requests the reset control managed by the reset-gpio device is effectively its consumer but the devlink is only established between it and the GPIO controller exposing the reset pin. Add a devlink between the consumer of the reset control and its supplier. This will allow us to simplify the GPIOLIB code managing shared GPIOs when handling the corner case of reset-gpio and gpiolib-shared interacting. While at it and since we need to store the address of the auxiliary device: don't allocate memory for the device separately but fold it into struct reset_gpio_lookup instead. Reviewed-by: Philipp Zabel <p.zabel@pengutronix.de> Signed-off-by: Bartosz Golaszewski <bartosz.golaszewski@oss.qualcomm.com> Signed-off-by: Philipp Zabel <p.zabel@pengutronix.de>
This commit is contained in:
committed by
Philipp Zabel
parent
fe3da77f2f
commit
a9b95ce36d
@@ -77,11 +77,13 @@ struct reset_control_array {
|
|||||||
* @of_args: phandle to the reset controller with all the args like GPIO number
|
* @of_args: phandle to the reset controller with all the args like GPIO number
|
||||||
* @swnode: Software node containing the reference to the GPIO provider
|
* @swnode: Software node containing the reference to the GPIO provider
|
||||||
* @list: list entry for the reset_gpio_lookup_list
|
* @list: list entry for the reset_gpio_lookup_list
|
||||||
|
* @adev: Auxiliary device representing the reset controller
|
||||||
*/
|
*/
|
||||||
struct reset_gpio_lookup {
|
struct reset_gpio_lookup {
|
||||||
struct of_phandle_args of_args;
|
struct of_phandle_args of_args;
|
||||||
struct fwnode_handle *swnode;
|
struct fwnode_handle *swnode;
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
struct auxiliary_device adev;
|
||||||
};
|
};
|
||||||
|
|
||||||
static const char *rcdev_name(struct reset_controller_dev *rcdev)
|
static const char *rcdev_name(struct reset_controller_dev *rcdev)
|
||||||
@@ -824,49 +826,72 @@ static void __reset_control_put_internal(struct reset_control *rstc)
|
|||||||
|
|
||||||
static void reset_gpio_aux_device_release(struct device *dev)
|
static void reset_gpio_aux_device_release(struct device *dev)
|
||||||
{
|
{
|
||||||
struct auxiliary_device *adev = to_auxiliary_dev(dev);
|
|
||||||
|
|
||||||
kfree(adev);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int reset_add_gpio_aux_device(struct device *parent,
|
static int reset_create_gpio_aux_device(struct reset_gpio_lookup *rgpio_dev,
|
||||||
struct fwnode_handle *swnode,
|
struct device *parent, int id)
|
||||||
int id, void *pdata)
|
|
||||||
{
|
{
|
||||||
struct auxiliary_device *adev;
|
struct auxiliary_device *adev = &rgpio_dev->adev;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
adev = kzalloc_obj(*adev);
|
|
||||||
if (!adev)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
adev->id = id;
|
adev->id = id;
|
||||||
adev->name = "gpio";
|
adev->name = "gpio";
|
||||||
adev->dev.parent = parent;
|
adev->dev.parent = parent;
|
||||||
adev->dev.platform_data = pdata;
|
adev->dev.platform_data = &rgpio_dev->of_args;
|
||||||
adev->dev.release = reset_gpio_aux_device_release;
|
adev->dev.release = reset_gpio_aux_device_release;
|
||||||
device_set_node(&adev->dev, swnode);
|
device_set_node(&adev->dev, rgpio_dev->swnode);
|
||||||
|
|
||||||
ret = auxiliary_device_init(adev);
|
ret = auxiliary_device_init(adev);
|
||||||
if (ret) {
|
if (ret)
|
||||||
kfree(adev);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
|
||||||
|
|
||||||
ret = __auxiliary_device_add(adev, "reset");
|
ret = __auxiliary_device_add(adev, "reset");
|
||||||
if (ret) {
|
if (ret) {
|
||||||
auxiliary_device_uninit(adev);
|
auxiliary_device_uninit(adev);
|
||||||
kfree(adev);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void reset_gpio_add_devlink(struct device_node *np,
|
||||||
|
struct reset_gpio_lookup *rgpio_dev)
|
||||||
|
{
|
||||||
|
struct device *consumer;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We must use get_dev_from_fwnode() and not of_find_device_by_node()
|
||||||
|
* because the latter only considers the platform bus while we want to
|
||||||
|
* get consumers of any kind that can be associated with firmware
|
||||||
|
* nodes: auxiliary, soundwire, etc.
|
||||||
|
*/
|
||||||
|
consumer = get_dev_from_fwnode(of_fwnode_handle(np));
|
||||||
|
if (consumer) {
|
||||||
|
if (!device_link_add(consumer, &rgpio_dev->adev.dev,
|
||||||
|
DL_FLAG_AUTOREMOVE_CONSUMER))
|
||||||
|
pr_warn("Failed to create a device link between reset-gpio and its consumer");
|
||||||
|
|
||||||
|
put_device(consumer);
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
* else { }
|
||||||
|
*
|
||||||
|
* TODO: If ever there's a case where we need to support shared
|
||||||
|
* reset-gpios retrieved from a device node for which there's no
|
||||||
|
* device present yet, this is where we'd set up a notifier waiting
|
||||||
|
* for the device to appear in the system. This would be a lot of code
|
||||||
|
* that would go unused for now so let's cross that bridge when and if
|
||||||
|
* we get there.
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @args: phandle to the GPIO provider with all the args like GPIO number
|
* @np: OF-node associated with the consumer
|
||||||
|
* @args: phandle to the GPIO provider with all the args like GPIO number
|
||||||
*/
|
*/
|
||||||
static int __reset_add_reset_gpio_device(const struct of_phandle_args *args)
|
static int __reset_add_reset_gpio_device(struct device_node *np,
|
||||||
|
const struct of_phandle_args *args)
|
||||||
{
|
{
|
||||||
struct property_entry properties[3] = { };
|
struct property_entry properties[3] = { };
|
||||||
unsigned int offset, of_flags, lflags;
|
unsigned int offset, of_flags, lflags;
|
||||||
@@ -916,8 +941,14 @@ static int __reset_add_reset_gpio_device(const struct of_phandle_args *args)
|
|||||||
|
|
||||||
list_for_each_entry(rgpio_dev, &reset_gpio_lookup_list, list) {
|
list_for_each_entry(rgpio_dev, &reset_gpio_lookup_list, list) {
|
||||||
if (args->np == rgpio_dev->of_args.np) {
|
if (args->np == rgpio_dev->of_args.np) {
|
||||||
if (of_phandle_args_equal(args, &rgpio_dev->of_args))
|
if (of_phandle_args_equal(args, &rgpio_dev->of_args)) {
|
||||||
return 0; /* Already on the list, done */
|
/*
|
||||||
|
* Already on the list, create the device link
|
||||||
|
* and stop here.
|
||||||
|
*/
|
||||||
|
reset_gpio_add_devlink(np, rgpio_dev);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -951,11 +982,11 @@ static int __reset_add_reset_gpio_device(const struct of_phandle_args *args)
|
|||||||
goto err_put_of_node;
|
goto err_put_of_node;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = reset_add_gpio_aux_device(parent, rgpio_dev->swnode, id,
|
ret = reset_create_gpio_aux_device(rgpio_dev, parent, id);
|
||||||
&rgpio_dev->of_args);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_del_swnode;
|
goto err_del_swnode;
|
||||||
|
|
||||||
|
reset_gpio_add_devlink(np, rgpio_dev);
|
||||||
list_add(&rgpio_dev->list, &reset_gpio_lookup_list);
|
list_add(&rgpio_dev->list, &reset_gpio_lookup_list);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@@ -1035,7 +1066,7 @@ __of_reset_control_get(struct device_node *node, const char *id, int index,
|
|||||||
|
|
||||||
gpio_fallback = true;
|
gpio_fallback = true;
|
||||||
|
|
||||||
ret = __reset_add_reset_gpio_device(&args);
|
ret = __reset_add_reset_gpio_device(node, &args);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
rstc = ERR_PTR(ret);
|
rstc = ERR_PTR(ret);
|
||||||
goto out_put;
|
goto out_put;
|
||||||
|
|||||||
Reference in New Issue
Block a user