ACPI / property: Extend device_get_next_child_node() to data-only nodes
Make device_get_next_child_node() work with ACPI data-only subnodes introduced previously. Namely, replace acpi_get_next_child() with acpi_get_next_subnode() that can handle (and return) child device objects as well as child data-only subnodes of the given device and modify the ACPI part of the GPIO subsystem to handle data-only subnodes returned by it. To that end, introduce acpi_node_get_gpiod() taking a struct fwnode_handle pointer as the first argument. That argument may point to an ACPI device object as well as to a data-only subnode and the function should do the right thing (ie. look for the matching GPIO descriptor correctly) in either case. Next, modify fwnode_get_named_gpiod() to use acpi_node_get_gpiod() instead of acpi_get_gpiod_by_index() which automatically causes devm_get_gpiod_from_child() to work with ACPI data-only subnodes that may be returned by device_get_next_child_node() which in turn is required by the users of that function (the gpio_keys_polled and gpio-leds drivers). Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Tested-by: Mika Westerberg <mika.westerberg@linux.intel.com> Acked-by: Linus Walleij <linus.walleij@linaro.org>
This commit is contained in:
@@ -453,7 +453,7 @@ static int acpi_gpio_resource_lookup(struct acpi_gpio_lookup *lookup,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int acpi_gpio_property_lookup(struct acpi_device *adev,
|
||||
static int acpi_gpio_property_lookup(struct fwnode_handle *fwnode,
|
||||
const char *propname, int index,
|
||||
struct acpi_gpio_lookup *lookup)
|
||||
{
|
||||
@@ -461,10 +461,16 @@ static int acpi_gpio_property_lookup(struct acpi_device *adev,
|
||||
int ret;
|
||||
|
||||
memset(&args, 0, sizeof(args));
|
||||
ret = acpi_dev_get_property_reference(adev, propname, index, &args);
|
||||
if (ret && !acpi_get_driver_gpio_data(adev, propname, index, &args))
|
||||
return ret;
|
||||
ret = acpi_node_get_property_reference(fwnode, propname, index, &args);
|
||||
if (ret) {
|
||||
struct acpi_device *adev = to_acpi_device_node(fwnode);
|
||||
|
||||
if (!adev)
|
||||
return ret;
|
||||
|
||||
if (!acpi_get_driver_gpio_data(adev, propname, index, &args))
|
||||
return ret;
|
||||
}
|
||||
/*
|
||||
* The property was found and resolved, so need to lookup the GPIO based
|
||||
* on returned args.
|
||||
@@ -518,7 +524,8 @@ struct gpio_desc *acpi_get_gpiod_by_index(struct acpi_device *adev,
|
||||
if (propname) {
|
||||
dev_dbg(&adev->dev, "GPIO: looking up %s\n", propname);
|
||||
|
||||
ret = acpi_gpio_property_lookup(adev, propname, index, &lookup);
|
||||
ret = acpi_gpio_property_lookup(acpi_fwnode_handle(adev),
|
||||
propname, index, &lookup);
|
||||
if (ret)
|
||||
return ERR_PTR(ret);
|
||||
|
||||
@@ -534,6 +541,47 @@ struct gpio_desc *acpi_get_gpiod_by_index(struct acpi_device *adev,
|
||||
return ret ? ERR_PTR(ret) : lookup.desc;
|
||||
}
|
||||
|
||||
/**
|
||||
* acpi_node_get_gpiod() - get a GPIO descriptor from ACPI resources
|
||||
* @fwnode: pointer to an ACPI firmware node to get the GPIO information from
|
||||
* @propname: Property name of the GPIO
|
||||
* @index: index of GpioIo/GpioInt resource (starting from %0)
|
||||
* @info: info pointer to fill in (optional)
|
||||
*
|
||||
* If @fwnode is an ACPI device object, call %acpi_get_gpiod_by_index() for it.
|
||||
* Otherwise (ie. it is a data-only non-device object), use the property-based
|
||||
* GPIO lookup to get to the GPIO resource with the relevant information and use
|
||||
* that to obtain the GPIO descriptor to return.
|
||||
*/
|
||||
struct gpio_desc *acpi_node_get_gpiod(struct fwnode_handle *fwnode,
|
||||
const char *propname, int index,
|
||||
struct acpi_gpio_info *info)
|
||||
{
|
||||
struct acpi_gpio_lookup lookup;
|
||||
struct acpi_device *adev;
|
||||
int ret;
|
||||
|
||||
adev = to_acpi_device_node(fwnode);
|
||||
if (adev)
|
||||
return acpi_get_gpiod_by_index(adev, propname, index, info);
|
||||
|
||||
if (!is_acpi_data_node(fwnode))
|
||||
return ERR_PTR(-ENODEV);
|
||||
|
||||
if (!propname)
|
||||
return ERR_PTR(-EINVAL);
|
||||
|
||||
memset(&lookup, 0, sizeof(lookup));
|
||||
lookup.index = index;
|
||||
|
||||
ret = acpi_gpio_property_lookup(fwnode, propname, index, &lookup);
|
||||
if (ret)
|
||||
return ERR_PTR(ret);
|
||||
|
||||
ret = acpi_gpio_resource_lookup(&lookup, info);
|
||||
return ret ? ERR_PTR(ret) : lookup.desc;
|
||||
}
|
||||
|
||||
/**
|
||||
* acpi_dev_gpio_irq_get() - Find GpioInt and translate it to Linux IRQ number
|
||||
* @adev: pointer to a ACPI device to get IRQ from
|
||||
|
Reference in New Issue
Block a user