Commit bcabdef1 authored by Alexandre Courbot's avatar Alexandre Courbot Committed by Grant Likely

gpiolib: check descriptors validity before use

Some functions dereferenced their GPIO descriptor argument without
checking its validity first, potentially leading to an oops when given
an invalid argument.

This patch also makes gpio_get_value() more resilient when given an
invalid GPIO, returning 0 instead of silently crashing.
Signed-off-by: default avatarAlexandre Courbot <acourbot@nvidia.com>
Cc: Ryan Mallon <rmallon@gmail.com>
Signed-off-by: default avatarGrant Likely <grant.likely@secretlab.ca>
parent b0af9cd9
...@@ -174,7 +174,7 @@ static int gpio_ensure_requested(struct gpio_desc *desc) ...@@ -174,7 +174,7 @@ static int gpio_ensure_requested(struct gpio_desc *desc)
/* caller holds gpio_lock *OR* gpio is marked as requested */ /* caller holds gpio_lock *OR* gpio is marked as requested */
static struct gpio_chip *gpiod_to_chip(struct gpio_desc *desc) static struct gpio_chip *gpiod_to_chip(struct gpio_desc *desc)
{ {
return desc->chip; return desc ? desc->chip : NULL;
} }
struct gpio_chip *gpio_to_chip(unsigned gpio) struct gpio_chip *gpio_to_chip(unsigned gpio)
...@@ -654,6 +654,11 @@ static ssize_t export_store(struct class *class, ...@@ -654,6 +654,11 @@ static ssize_t export_store(struct class *class,
goto done; goto done;
desc = gpio_to_desc(gpio); desc = gpio_to_desc(gpio);
/* reject invalid GPIOs */
if (!desc) {
pr_warn("%s: invalid GPIO %ld\n", __func__, gpio);
return -EINVAL;
}
/* No extra locking here; FLAG_SYSFS just signifies that the /* No extra locking here; FLAG_SYSFS just signifies that the
* request and export were done by on behalf of userspace, so * request and export were done by on behalf of userspace, so
...@@ -690,12 +695,14 @@ static ssize_t unexport_store(struct class *class, ...@@ -690,12 +695,14 @@ static ssize_t unexport_store(struct class *class,
if (status < 0) if (status < 0)
goto done; goto done;
status = -EINVAL;
desc = gpio_to_desc(gpio); desc = gpio_to_desc(gpio);
/* reject bogus commands (gpio_unexport ignores them) */ /* reject bogus commands (gpio_unexport ignores them) */
if (!desc) if (!desc) {
goto done; pr_warn("%s: invalid GPIO %ld\n", __func__, gpio);
return -EINVAL;
}
status = -EINVAL;
/* No extra locking here; FLAG_SYSFS just signifies that the /* No extra locking here; FLAG_SYSFS just signifies that the
* request and export were done by on behalf of userspace, so * request and export were done by on behalf of userspace, so
...@@ -846,8 +853,10 @@ static int gpiod_export_link(struct device *dev, const char *name, ...@@ -846,8 +853,10 @@ static int gpiod_export_link(struct device *dev, const char *name,
{ {
int status = -EINVAL; int status = -EINVAL;
if (!desc) if (!desc) {
goto done; pr_warn("%s: invalid GPIO\n", __func__);
return -EINVAL;
}
mutex_lock(&sysfs_lock); mutex_lock(&sysfs_lock);
...@@ -865,7 +874,6 @@ static int gpiod_export_link(struct device *dev, const char *name, ...@@ -865,7 +874,6 @@ static int gpiod_export_link(struct device *dev, const char *name,
mutex_unlock(&sysfs_lock); mutex_unlock(&sysfs_lock);
done:
if (status) if (status)
pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc), pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc),
status); status);
...@@ -896,8 +904,10 @@ static int gpiod_sysfs_set_active_low(struct gpio_desc *desc, int value) ...@@ -896,8 +904,10 @@ static int gpiod_sysfs_set_active_low(struct gpio_desc *desc, int value)
struct device *dev = NULL; struct device *dev = NULL;
int status = -EINVAL; int status = -EINVAL;
if (!desc) if (!desc) {
goto done; pr_warn("%s: invalid GPIO\n", __func__);
return -EINVAL;
}
mutex_lock(&sysfs_lock); mutex_lock(&sysfs_lock);
...@@ -914,7 +924,6 @@ static int gpiod_sysfs_set_active_low(struct gpio_desc *desc, int value) ...@@ -914,7 +924,6 @@ static int gpiod_sysfs_set_active_low(struct gpio_desc *desc, int value)
unlock: unlock:
mutex_unlock(&sysfs_lock); mutex_unlock(&sysfs_lock);
done:
if (status) if (status)
pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc), pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc),
status); status);
...@@ -940,8 +949,8 @@ static void gpiod_unexport(struct gpio_desc *desc) ...@@ -940,8 +949,8 @@ static void gpiod_unexport(struct gpio_desc *desc)
struct device *dev = NULL; struct device *dev = NULL;
if (!desc) { if (!desc) {
status = -EINVAL; pr_warn("%s: invalid GPIO\n", __func__);
goto done; return;
} }
mutex_lock(&sysfs_lock); mutex_lock(&sysfs_lock);
...@@ -962,7 +971,7 @@ static void gpiod_unexport(struct gpio_desc *desc) ...@@ -962,7 +971,7 @@ static void gpiod_unexport(struct gpio_desc *desc)
device_unregister(dev); device_unregister(dev);
put_device(dev); put_device(dev);
} }
done:
if (status) if (status)
pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc), pr_debug("%s: gpio%d status %d\n", __func__, desc_to_gpio(desc),
status); status);
...@@ -1384,12 +1393,13 @@ static int gpiod_request(struct gpio_desc *desc, const char *label) ...@@ -1384,12 +1393,13 @@ static int gpiod_request(struct gpio_desc *desc, const char *label)
int status = -EPROBE_DEFER; int status = -EPROBE_DEFER;
unsigned long flags; unsigned long flags;
spin_lock_irqsave(&gpio_lock, flags);
if (!desc) { if (!desc) {
status = -EINVAL; pr_warn("%s: invalid GPIO\n", __func__);
goto done; return -EINVAL;
} }
spin_lock_irqsave(&gpio_lock, flags);
chip = desc->chip; chip = desc->chip;
if (chip == NULL) if (chip == NULL)
goto done; goto done;
...@@ -1432,8 +1442,7 @@ static int gpiod_request(struct gpio_desc *desc, const char *label) ...@@ -1432,8 +1442,7 @@ static int gpiod_request(struct gpio_desc *desc, const char *label)
done: done:
if (status) if (status)
pr_debug("_gpio_request: gpio-%d (%s) status %d\n", pr_debug("_gpio_request: gpio-%d (%s) status %d\n",
desc ? desc_to_gpio(desc) : -1, desc_to_gpio(desc), label ? : "?", status);
label ? : "?", status);
spin_unlock_irqrestore(&gpio_lock, flags); spin_unlock_irqrestore(&gpio_lock, flags);
return status; return status;
} }
...@@ -1616,10 +1625,13 @@ static int gpiod_direction_input(struct gpio_desc *desc) ...@@ -1616,10 +1625,13 @@ static int gpiod_direction_input(struct gpio_desc *desc)
int status = -EINVAL; int status = -EINVAL;
int offset; int offset;
if (!desc) {
pr_warn("%s: invalid GPIO\n", __func__);
return -EINVAL;
}
spin_lock_irqsave(&gpio_lock, flags); spin_lock_irqsave(&gpio_lock, flags);
if (!desc)
goto fail;
chip = desc->chip; chip = desc->chip;
if (!chip || !chip->get || !chip->direction_input) if (!chip || !chip->get || !chip->direction_input)
goto fail; goto fail;
...@@ -1655,13 +1667,9 @@ static int gpiod_direction_input(struct gpio_desc *desc) ...@@ -1655,13 +1667,9 @@ static int gpiod_direction_input(struct gpio_desc *desc)
return status; return status;
fail: fail:
spin_unlock_irqrestore(&gpio_lock, flags); spin_unlock_irqrestore(&gpio_lock, flags);
if (status) { if (status)
int gpio = -1; pr_debug("%s: gpio-%d status %d\n", __func__,
if (desc) desc_to_gpio(desc), status);
gpio = desc_to_gpio(desc);
pr_debug("%s: gpio-%d status %d\n",
__func__, gpio, status);
}
return status; return status;
} }
...@@ -1678,6 +1686,11 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value) ...@@ -1678,6 +1686,11 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value)
int status = -EINVAL; int status = -EINVAL;
int offset; int offset;
if (!desc) {
pr_warn("%s: invalid GPIO\n", __func__);
return -EINVAL;
}
/* Open drain pin should not be driven to 1 */ /* Open drain pin should not be driven to 1 */
if (value && test_bit(FLAG_OPEN_DRAIN, &desc->flags)) if (value && test_bit(FLAG_OPEN_DRAIN, &desc->flags))
return gpiod_direction_input(desc); return gpiod_direction_input(desc);
...@@ -1688,8 +1701,6 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value) ...@@ -1688,8 +1701,6 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value)
spin_lock_irqsave(&gpio_lock, flags); spin_lock_irqsave(&gpio_lock, flags);
if (!desc)
goto fail;
chip = desc->chip; chip = desc->chip;
if (!chip || !chip->set || !chip->direction_output) if (!chip || !chip->set || !chip->direction_output)
goto fail; goto fail;
...@@ -1725,13 +1736,9 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value) ...@@ -1725,13 +1736,9 @@ static int gpiod_direction_output(struct gpio_desc *desc, int value)
return status; return status;
fail: fail:
spin_unlock_irqrestore(&gpio_lock, flags); spin_unlock_irqrestore(&gpio_lock, flags);
if (status) { if (status)
int gpio = -1; pr_debug("%s: gpio-%d status %d\n", __func__,
if (desc) desc_to_gpio(desc), status);
gpio = desc_to_gpio(desc);
pr_debug("%s: gpio-%d status %d\n",
__func__, gpio, status);
}
return status; return status;
} }
...@@ -1753,10 +1760,13 @@ static int gpiod_set_debounce(struct gpio_desc *desc, unsigned debounce) ...@@ -1753,10 +1760,13 @@ static int gpiod_set_debounce(struct gpio_desc *desc, unsigned debounce)
int status = -EINVAL; int status = -EINVAL;
int offset; int offset;
if (!desc) {
pr_warn("%s: invalid GPIO\n", __func__);
return -EINVAL;
}
spin_lock_irqsave(&gpio_lock, flags); spin_lock_irqsave(&gpio_lock, flags);
if (!desc)
goto fail;
chip = desc->chip; chip = desc->chip;
if (!chip || !chip->set || !chip->set_debounce) if (!chip || !chip->set || !chip->set_debounce)
goto fail; goto fail;
...@@ -1776,13 +1786,9 @@ static int gpiod_set_debounce(struct gpio_desc *desc, unsigned debounce) ...@@ -1776,13 +1786,9 @@ static int gpiod_set_debounce(struct gpio_desc *desc, unsigned debounce)
fail: fail:
spin_unlock_irqrestore(&gpio_lock, flags); spin_unlock_irqrestore(&gpio_lock, flags);
if (status) { if (status)
int gpio = -1; pr_debug("%s: gpio-%d status %d\n", __func__,
if (desc) desc_to_gpio(desc), status);
gpio = desc_to_gpio(desc);
pr_debug("%s: gpio-%d status %d\n",
__func__, gpio, status);
}
return status; return status;
} }
...@@ -1830,6 +1836,8 @@ static int gpiod_get_value(struct gpio_desc *desc) ...@@ -1830,6 +1836,8 @@ static int gpiod_get_value(struct gpio_desc *desc)
int value; int value;
int offset; int offset;
if (!desc)
return 0;
chip = desc->chip; chip = desc->chip;
offset = gpio_chip_hwgpio(desc); offset = gpio_chip_hwgpio(desc);
/* Should be using gpio_get_value_cansleep() */ /* Should be using gpio_get_value_cansleep() */
...@@ -1912,6 +1920,8 @@ static void gpiod_set_value(struct gpio_desc *desc, int value) ...@@ -1912,6 +1920,8 @@ static void gpiod_set_value(struct gpio_desc *desc, int value)
{ {
struct gpio_chip *chip; struct gpio_chip *chip;
if (!desc)
return;
chip = desc->chip; chip = desc->chip;
/* Should be using gpio_set_value_cansleep() */ /* Should be using gpio_set_value_cansleep() */
WARN_ON(chip->can_sleep); WARN_ON(chip->can_sleep);
...@@ -1940,6 +1950,8 @@ EXPORT_SYMBOL_GPL(__gpio_set_value); ...@@ -1940,6 +1950,8 @@ EXPORT_SYMBOL_GPL(__gpio_set_value);
*/ */
static int gpiod_cansleep(struct gpio_desc *desc) static int gpiod_cansleep(struct gpio_desc *desc)
{ {
if (!desc)
return 0;
/* only call this on GPIOs that are valid! */ /* only call this on GPIOs that are valid! */
return desc->chip->can_sleep; return desc->chip->can_sleep;
} }
...@@ -1964,6 +1976,8 @@ static int gpiod_to_irq(struct gpio_desc *desc) ...@@ -1964,6 +1976,8 @@ static int gpiod_to_irq(struct gpio_desc *desc)
struct gpio_chip *chip; struct gpio_chip *chip;
int offset; int offset;
if (!desc)
return -EINVAL;
chip = desc->chip; chip = desc->chip;
offset = gpio_chip_hwgpio(desc); offset = gpio_chip_hwgpio(desc);
return chip->to_irq ? chip->to_irq(chip, offset) : -ENXIO; return chip->to_irq ? chip->to_irq(chip, offset) : -ENXIO;
...@@ -1987,6 +2001,8 @@ static int gpiod_get_value_cansleep(struct gpio_desc *desc) ...@@ -1987,6 +2001,8 @@ static int gpiod_get_value_cansleep(struct gpio_desc *desc)
int offset; int offset;
might_sleep_if(extra_checks); might_sleep_if(extra_checks);
if (!desc)
return 0;
chip = desc->chip; chip = desc->chip;
offset = gpio_chip_hwgpio(desc); offset = gpio_chip_hwgpio(desc);
value = chip->get ? chip->get(chip, offset) : 0; value = chip->get ? chip->get(chip, offset) : 0;
...@@ -2005,6 +2021,8 @@ static void gpiod_set_value_cansleep(struct gpio_desc *desc, int value) ...@@ -2005,6 +2021,8 @@ static void gpiod_set_value_cansleep(struct gpio_desc *desc, int value)
struct gpio_chip *chip; struct gpio_chip *chip;
might_sleep_if(extra_checks); might_sleep_if(extra_checks);
if (!desc)
return;
chip = desc->chip; chip = desc->chip;
trace_gpio_value(desc_to_gpio(desc), 0, value); trace_gpio_value(desc_to_gpio(desc), 0, value);
if (test_bit(FLAG_OPEN_DRAIN, &desc->flags)) if (test_bit(FLAG_OPEN_DRAIN, &desc->flags))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment