提交 fdeb8e15 编写于 作者: L Linus Walleij

gpio: reflect base and ngpio into gpio_device

Some information about the GPIO chip need to stay around also
after the gpio_chip has been removed and only the gpio_device
persist. The base and ngpio are such things, for example we
don't want a new chip arriving to overlap the number space
of a dangling gpio_device, and the chardev may still query
the device for the number of lines etc.

Note that the code that assigns base and insert gpio_device
into the global list no longer check for a missing gpio_chip:
we respect the number space allocated by any other gpio_device.

As a consequence of the gdev being referenced directly from
the gpio_desc, we need to verify it differently from all
in-kernel API calls that fall through to direct queries to
the gpio_chip vtable: we first check that desc is !NULL, then
that desc->gdev is !NULL, then, if desc->gdev->chip is NULL,
we *BAIL OUT* without any error, so as to manage the case
where operations are requested on a device that is gone.

These checks were non-uniform and partly missing in the past:
so to simplify: create the macros VALIDATE_DESC() that will
return -EINVAL if the desc or desc->gdev is missing and just
0 if the chip is gone, and conversely VALIDATE_DESC_VOID()
for the case where the function does not return an error.
By using these macros, we get warning messages about missing
gdev with reference to the right function in the kernel log.

Despite the macro business this simplifies the code and make
it more readable than if we copy/paste the same descriptor
checking code into all code ABI call sites (IMHO).
Signed-off-by: NLinus Walleij <linus.walleij@linaro.org>
上级 6cee3821
...@@ -180,7 +180,7 @@ static int gpio_sysfs_request_irq(struct device *dev, unsigned char flags) ...@@ -180,7 +180,7 @@ static int gpio_sysfs_request_irq(struct device *dev, unsigned char flags)
* Remove this redundant call (along with the corresponding * Remove this redundant call (along with the corresponding
* unlock) when those drivers have been fixed. * unlock) when those drivers have been fixed.
*/ */
ret = gpiochip_lock_as_irq(desc->chip, gpio_chip_hwgpio(desc)); ret = gpiochip_lock_as_irq(desc->gdev->chip, gpio_chip_hwgpio(desc));
if (ret < 0) if (ret < 0)
goto err_put_kn; goto err_put_kn;
...@@ -194,7 +194,7 @@ static int gpio_sysfs_request_irq(struct device *dev, unsigned char flags) ...@@ -194,7 +194,7 @@ static int gpio_sysfs_request_irq(struct device *dev, unsigned char flags)
return 0; return 0;
err_unlock: err_unlock:
gpiochip_unlock_as_irq(desc->chip, gpio_chip_hwgpio(desc)); gpiochip_unlock_as_irq(desc->gdev->chip, gpio_chip_hwgpio(desc));
err_put_kn: err_put_kn:
sysfs_put(data->value_kn); sysfs_put(data->value_kn);
...@@ -212,7 +212,7 @@ static void gpio_sysfs_free_irq(struct device *dev) ...@@ -212,7 +212,7 @@ static void gpio_sysfs_free_irq(struct device *dev)
data->irq_flags = 0; data->irq_flags = 0;
free_irq(data->irq, data); free_irq(data->irq, data);
gpiochip_unlock_as_irq(desc->chip, gpio_chip_hwgpio(desc)); gpiochip_unlock_as_irq(desc->gdev->chip, gpio_chip_hwgpio(desc));
sysfs_put(data->value_kn); sysfs_put(data->value_kn);
} }
...@@ -566,8 +566,8 @@ int gpiod_export(struct gpio_desc *desc, bool direction_may_change) ...@@ -566,8 +566,8 @@ int gpiod_export(struct gpio_desc *desc, bool direction_may_change)
return -EINVAL; return -EINVAL;
} }
chip = desc->chip; gdev = desc->gdev;
gdev = chip->gpiodev; chip = gdev->chip;
mutex_lock(&sysfs_lock); mutex_lock(&sysfs_lock);
...@@ -765,7 +765,7 @@ void gpiochip_sysfs_unregister(struct gpio_device *gdev) ...@@ -765,7 +765,7 @@ void gpiochip_sysfs_unregister(struct gpio_device *gdev)
/* unregister gpiod class devices owned by sysfs */ /* unregister gpiod class devices owned by sysfs */
for (i = 0; i < chip->ngpio; i++) { for (i = 0; i < chip->ngpio; i++) {
desc = &chip->gpiodev->descs[i]; desc = &gdev->descs[i];
if (test_and_clear_bit(FLAG_SYSFS, &desc->flags)) if (test_and_clear_bit(FLAG_SYSFS, &desc->flags))
gpiod_free(desc); gpiod_free(desc);
} }
......
此差异已折叠。
...@@ -33,6 +33,10 @@ struct acpi_device; ...@@ -33,6 +33,10 @@ struct acpi_device;
* @chip: pointer to the corresponding gpiochip, holding static * @chip: pointer to the corresponding gpiochip, holding static
* data for this device * data for this device
* @descs: array of ngpio descriptors. * @descs: array of ngpio descriptors.
* @ngpio: the number of GPIO lines on this GPIO device, equal to the size
* of the @descs array.
* @base: GPIO base in the DEPRECATED global Linux GPIO numberspace, assigned
* at device creation time.
* @list: links gpio_device:s together for traversal * @list: links gpio_device:s together for traversal
* *
* This state container holds most of the runtime variable data * This state container holds most of the runtime variable data
...@@ -48,6 +52,8 @@ struct gpio_device { ...@@ -48,6 +52,8 @@ struct gpio_device {
struct module *owner; struct module *owner;
struct gpio_chip *chip; struct gpio_chip *chip;
struct gpio_desc *descs; struct gpio_desc *descs;
int base;
u16 ngpio;
struct list_head list; struct list_head list;
}; };
...@@ -125,7 +131,7 @@ extern struct spinlock gpio_lock; ...@@ -125,7 +131,7 @@ extern struct spinlock gpio_lock;
extern struct list_head gpio_devices; extern struct list_head gpio_devices;
struct gpio_desc { struct gpio_desc {
struct gpio_chip *chip; struct gpio_device *gdev;
unsigned long flags; unsigned long flags;
/* flag symbols are bit numbers */ /* flag symbols are bit numbers */
#define FLAG_REQUESTED 0 #define FLAG_REQUESTED 0
...@@ -154,7 +160,7 @@ int gpiod_hog(struct gpio_desc *desc, const char *name, ...@@ -154,7 +160,7 @@ int gpiod_hog(struct gpio_desc *desc, const char *name,
*/ */
static int __maybe_unused gpio_chip_hwgpio(const struct gpio_desc *desc) static int __maybe_unused gpio_chip_hwgpio(const struct gpio_desc *desc)
{ {
return desc - &desc->chip->gpiodev->descs[0]; return desc - &desc->gdev->descs[0];
} }
/* With descriptor prefix */ /* With descriptor prefix */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册