diff options
Diffstat (limited to 'arch/arm/mach-pxa/gpio.c')
-rw-r--r-- | arch/arm/mach-pxa/gpio.c | 20 |
1 files changed, 18 insertions, 2 deletions
diff --git a/arch/arm/mach-pxa/gpio.c b/arch/arm/mach-pxa/gpio.c index a98b2da4c962..f96cae04f6f5 100644 --- a/arch/arm/mach-pxa/gpio.c +++ b/arch/arm/mach-pxa/gpio.c | |||
@@ -164,6 +164,20 @@ static long GPIO_IRQ_rising_edge[4]; | |||
164 | static long GPIO_IRQ_falling_edge[4]; | 164 | static long GPIO_IRQ_falling_edge[4]; |
165 | static long GPIO_IRQ_mask[4]; | 165 | static long GPIO_IRQ_mask[4]; |
166 | 166 | ||
167 | /* | ||
168 | * On PXA25x and PXA27x, GAFRx and GPDRx together decide the alternate | ||
169 | * function of a GPIO, and GPDRx cannot be altered once configured. It | ||
170 | * is attributed as "occupied" here (I know this terminology isn't | ||
171 | * accurate, you are welcome to propose a better one :-) | ||
172 | */ | ||
173 | static int __gpio_is_occupied(unsigned gpio) | ||
174 | { | ||
175 | if (cpu_is_pxa25x() || cpu_is_pxa27x()) | ||
176 | return GAFR(gpio) & (0x3 << (((gpio) & 0xf) * 2)); | ||
177 | else | ||
178 | return 0; | ||
179 | } | ||
180 | |||
167 | static int pxa_gpio_irq_type(unsigned int irq, unsigned int type) | 181 | static int pxa_gpio_irq_type(unsigned int irq, unsigned int type) |
168 | { | 182 | { |
169 | int gpio, idx; | 183 | int gpio, idx; |
@@ -179,12 +193,14 @@ static int pxa_gpio_irq_type(unsigned int irq, unsigned int type) | |||
179 | GPIO_IRQ_falling_edge[idx] | | 193 | GPIO_IRQ_falling_edge[idx] | |
180 | GPDR(gpio)) & GPIO_bit(gpio)) | 194 | GPDR(gpio)) & GPIO_bit(gpio)) |
181 | return 0; | 195 | return 0; |
182 | if (GAFR(gpio) & (0x3 << (((gpio) & 0xf)*2))) | 196 | |
197 | if (__gpio_is_occupied(gpio)) | ||
183 | return 0; | 198 | return 0; |
199 | |||
184 | type = IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING; | 200 | type = IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING; |
185 | } | 201 | } |
186 | 202 | ||
187 | pxa_gpio_mode(gpio | GPIO_IN); | 203 | GPDR(gpio) &= ~GPIO_bit(gpio); |
188 | 204 | ||
189 | if (type & IRQ_TYPE_EDGE_RISING) | 205 | if (type & IRQ_TYPE_EDGE_RISING) |
190 | __set_bit(gpio, GPIO_IRQ_rising_edge); | 206 | __set_bit(gpio, GPIO_IRQ_rising_edge); |