diff options
| author | Anton Vorontsov <avorontsov@ru.mvista.com> | 2008-04-28 05:14:47 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2008-04-28 11:58:34 -0400 |
| commit | 169b6a7a6e91e1ea32136681b475cbaf2074bf35 (patch) | |
| tree | bf29842528e61cb07c581e43acbe3619a25746ab | |
| parent | 8d0aab2f16c4fa170f32e7a74a52cd0122bbafef (diff) | |
gpiochip_reserve()
Add a new function gpiochip_reserve() to reserve ranges of gpios that platform
code has pre-allocated. That is, this marks gpio numbers which will be
claimed by drivers that haven't yet been loaded, and thus are not available
for dynamic gpio number allocation.
[akpm@linux-foundation.org: remove unneeded __must_check]
[david-b@pacbell.net: don't export gpiochip_reserve (section fix)]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
| -rw-r--r-- | drivers/gpio/gpiolib.c | 50 | ||||
| -rw-r--r-- | include/asm-generic/gpio.h | 1 |
2 files changed, 48 insertions, 3 deletions
diff --git a/drivers/gpio/gpiolib.c b/drivers/gpio/gpiolib.c index 2ba6127c4fae..24c62b848bf9 100644 --- a/drivers/gpio/gpiolib.c +++ b/drivers/gpio/gpiolib.c | |||
| @@ -43,6 +43,7 @@ struct gpio_desc { | |||
| 43 | /* flag symbols are bit numbers */ | 43 | /* flag symbols are bit numbers */ |
| 44 | #define FLAG_REQUESTED 0 | 44 | #define FLAG_REQUESTED 0 |
| 45 | #define FLAG_IS_OUT 1 | 45 | #define FLAG_IS_OUT 1 |
| 46 | #define FLAG_RESERVED 2 | ||
| 46 | 47 | ||
| 47 | #ifdef CONFIG_DEBUG_FS | 48 | #ifdef CONFIG_DEBUG_FS |
| 48 | const char *label; | 49 | const char *label; |
| @@ -88,9 +89,10 @@ static int gpiochip_find_base(int ngpio) | |||
| 88 | int base = -ENOSPC; | 89 | int base = -ENOSPC; |
| 89 | 90 | ||
| 90 | for (i = ARCH_NR_GPIOS - 1; i >= 0 ; i--) { | 91 | for (i = ARCH_NR_GPIOS - 1; i >= 0 ; i--) { |
| 91 | struct gpio_chip *chip = gpio_desc[i].chip; | 92 | struct gpio_desc *desc = &gpio_desc[i]; |
| 93 | struct gpio_chip *chip = desc->chip; | ||
| 92 | 94 | ||
| 93 | if (!chip) { | 95 | if (!chip && !test_bit(FLAG_RESERVED, &desc->flags)) { |
| 94 | spare++; | 96 | spare++; |
| 95 | if (spare == ngpio) { | 97 | if (spare == ngpio) { |
| 96 | base = i; | 98 | base = i; |
| @@ -98,7 +100,8 @@ static int gpiochip_find_base(int ngpio) | |||
| 98 | } | 100 | } |
| 99 | } else { | 101 | } else { |
| 100 | spare = 0; | 102 | spare = 0; |
| 101 | i -= chip->ngpio - 1; | 103 | if (chip) |
| 104 | i -= chip->ngpio - 1; | ||
| 102 | } | 105 | } |
| 103 | } | 106 | } |
| 104 | 107 | ||
| @@ -108,6 +111,47 @@ static int gpiochip_find_base(int ngpio) | |||
| 108 | } | 111 | } |
| 109 | 112 | ||
| 110 | /** | 113 | /** |
| 114 | * gpiochip_reserve() - reserve range of gpios to use with platform code only | ||
| 115 | * @start: starting gpio number | ||
| 116 | * @ngpio: number of gpios to reserve | ||
| 117 | * Context: platform init, potentially before irqs or kmalloc will work | ||
| 118 | * | ||
| 119 | * Returns a negative errno if any gpio within the range is already reserved | ||
| 120 | * or registered, else returns zero as a success code. Use this function | ||
| 121 | * to mark a range of gpios as unavailable for dynamic gpio number allocation, | ||
| 122 | * for example because its driver support is not yet loaded. | ||
| 123 | */ | ||
| 124 | int __init gpiochip_reserve(int start, int ngpio) | ||
| 125 | { | ||
| 126 | int ret = 0; | ||
| 127 | unsigned long flags; | ||
| 128 | int i; | ||
| 129 | |||
| 130 | if (!gpio_is_valid(start) || !gpio_is_valid(start + ngpio)) | ||
| 131 | return -EINVAL; | ||
| 132 | |||
| 133 | spin_lock_irqsave(&gpio_lock, flags); | ||
| 134 | |||
| 135 | for (i = start; i < start + ngpio; i++) { | ||
| 136 | struct gpio_desc *desc = &gpio_desc[i]; | ||
| 137 | |||
| 138 | if (desc->chip || test_bit(FLAG_RESERVED, &desc->flags)) { | ||
| 139 | ret = -EBUSY; | ||
| 140 | goto err; | ||
| 141 | } | ||
| 142 | |||
| 143 | set_bit(FLAG_RESERVED, &desc->flags); | ||
| 144 | } | ||
| 145 | |||
| 146 | pr_debug("%s: reserved gpios from %d to %d\n", | ||
| 147 | __func__, start, start + ngpio - 1); | ||
| 148 | err: | ||
| 149 | spin_unlock_irqrestore(&gpio_lock, flags); | ||
| 150 | |||
| 151 | return ret; | ||
| 152 | } | ||
| 153 | |||
| 154 | /** | ||
| 111 | * gpiochip_add() - register a gpio_chip | 155 | * gpiochip_add() - register a gpio_chip |
| 112 | * @chip: the chip to register, with chip->base initialized | 156 | * @chip: the chip to register, with chip->base initialized |
| 113 | * Context: potentially before irqs or kmalloc will work | 157 | * Context: potentially before irqs or kmalloc will work |
diff --git a/include/asm-generic/gpio.h b/include/asm-generic/gpio.h index 464c5b334dc2..ecf675a59d21 100644 --- a/include/asm-generic/gpio.h +++ b/include/asm-generic/gpio.h | |||
| @@ -74,6 +74,7 @@ struct gpio_chip { | |||
| 74 | 74 | ||
| 75 | extern const char *gpiochip_is_requested(struct gpio_chip *chip, | 75 | extern const char *gpiochip_is_requested(struct gpio_chip *chip, |
| 76 | unsigned offset); | 76 | unsigned offset); |
| 77 | extern int __init __must_check gpiochip_reserve(int start, int ngpio); | ||
| 77 | 78 | ||
| 78 | /* add/remove chips */ | 79 | /* add/remove chips */ |
| 79 | extern int gpiochip_add(struct gpio_chip *chip); | 80 | extern int gpiochip_add(struct gpio_chip *chip); |
