aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/plat-s5p
diff options
context:
space:
mode:
authorMarek Szyprowski <m.szyprowski@samsung.com>2011-03-15 08:17:43 -0400
committerKukjin Kim <kgene.kim@samsung.com>2011-03-15 08:17:43 -0400
commita43efddc3bf8b143ff9352763cd39d425db14d27 (patch)
tree250d9987ae00298c9f63735728b8dd6d2b1fe4e4 /arch/arm/plat-s5p
parent2de0926204b100409bcd51465404ae846c7f48fd (diff)
ARM: S5P: Add function to register gpio interrupt bank data
This patch removes all global data from common s5p gpio interrupt handler code. This enables to reuse this code on EXYNOS4 platform. Instead of global data (IRQ_GPIOINT interrupt number, S5P_GPIOINT_GROUP_MAXNR groups count), a s5p_register_gpioint_bank() function is introduced. It is aimed to be called from gpiolib init. Signed-off-by: Marek Szyprowski <m.szyprowski@samsung.com> Signed-off-by: Kyungmin Park <kyungmin.park@samsung.com> Signed-off-by: Kukjin Kim <kgene.kim@samsung.com>
Diffstat (limited to 'arch/arm/plat-s5p')
-rw-r--r--arch/arm/plat-s5p/irq-gpioint.c68
1 files changed, 58 insertions, 10 deletions
diff --git a/arch/arm/plat-s5p/irq-gpioint.c b/arch/arm/plat-s5p/irq-gpioint.c
index 79a7b1c5a57..cd87d3256e0 100644
--- a/arch/arm/plat-s5p/irq-gpioint.c
+++ b/arch/arm/plat-s5p/irq-gpioint.c
@@ -17,6 +17,7 @@
17#include <linux/irq.h> 17#include <linux/irq.h>
18#include <linux/io.h> 18#include <linux/io.h>
19#include <linux/gpio.h> 19#include <linux/gpio.h>
20#include <linux/slab.h>
20 21
21#include <mach/map.h> 22#include <mach/map.h>
22#include <plat/gpio-core.h> 23#include <plat/gpio-core.h>
@@ -29,7 +30,16 @@
29#define PEND_OFFSET 0xA00 30#define PEND_OFFSET 0xA00
30#define REG_OFFSET(x) ((x) << 2) 31#define REG_OFFSET(x) ((x) << 2)
31 32
32static struct s3c_gpio_chip *irq_chips[S5P_GPIOINT_GROUP_MAXNR]; 33struct s5p_gpioint_bank {
34 struct list_head list;
35 int start;
36 int nr_groups;
37 int irq;
38 struct s3c_gpio_chip **chips;
39 void (*handler)(unsigned int, struct irq_desc *);
40};
41
42LIST_HEAD(banks);
33 43
34static int s5p_gpioint_get_offset(struct irq_data *data) 44static int s5p_gpioint_get_offset(struct irq_data *data)
35{ 45{
@@ -139,11 +149,12 @@ static struct irq_chip s5p_gpioint = {
139 149
140static void s5p_gpioint_handler(unsigned int irq, struct irq_desc *desc) 150static void s5p_gpioint_handler(unsigned int irq, struct irq_desc *desc)
141{ 151{
152 struct s5p_gpioint_bank *bank = get_irq_data(irq);
142 int group, pend_offset, mask_offset; 153 int group, pend_offset, mask_offset;
143 unsigned int pend, mask; 154 unsigned int pend, mask;
144 155
145 for (group = 0; group < S5P_GPIOINT_GROUP_MAXNR; group++) { 156 for (group = 0; group < bank->nr_groups; group++) {
146 struct s3c_gpio_chip *chip = irq_chips[group]; 157 struct s3c_gpio_chip *chip = bank->chips[group];
147 if (!chip) 158 if (!chip)
148 continue; 159 continue;
149 160
@@ -168,23 +179,44 @@ static void s5p_gpioint_handler(unsigned int irq, struct irq_desc *desc)
168static __init int s5p_gpioint_add(struct s3c_gpio_chip *chip) 179static __init int s5p_gpioint_add(struct s3c_gpio_chip *chip)
169{ 180{
170 static int used_gpioint_groups = 0; 181 static int used_gpioint_groups = 0;
171 static bool handler_registered = 0;
172 int irq, group = chip->group; 182 int irq, group = chip->group;
173 int i; 183 int i;
184 struct s5p_gpioint_bank *bank = NULL;
174 185
175 if (used_gpioint_groups >= S5P_GPIOINT_GROUP_COUNT) 186 if (used_gpioint_groups >= S5P_GPIOINT_GROUP_COUNT)
176 return -ENOMEM; 187 return -ENOMEM;
177 188
189 list_for_each_entry(bank, &banks, list) {
190 if (group >= bank->start &&
191 group < bank->start + bank->nr_groups)
192 break;
193 }
194 if (!bank)
195 return -EINVAL;
196
197 if (!bank->handler) {
198 bank->chips = kzalloc(sizeof(struct s3c_gpio_chip *) *
199 bank->nr_groups, GFP_KERNEL);
200 if (!bank->chips)
201 return -ENOMEM;
202
203 set_irq_chained_handler(bank->irq, s5p_gpioint_handler);
204 set_irq_data(bank->irq, bank);
205 bank->handler = s5p_gpioint_handler;
206 printk(KERN_INFO "Registered chained gpio int handler for interrupt %d.\n",
207 bank->irq);
208 }
209
210 /*
211 * chained GPIO irq has been sucessfully registered, allocate new gpio
212 * int group and assign irq nubmers
213 */
214
178 chip->irq_base = S5P_GPIOINT_BASE + 215 chip->irq_base = S5P_GPIOINT_BASE +
179 used_gpioint_groups * S5P_GPIOINT_GROUP_SIZE; 216 used_gpioint_groups * S5P_GPIOINT_GROUP_SIZE;
180 used_gpioint_groups++; 217 used_gpioint_groups++;
181 218
182 if (!handler_registered) { 219 bank->chips[group - bank->start] = chip;
183 set_irq_chained_handler(IRQ_GPIOINT, s5p_gpioint_handler);
184 handler_registered = 1;
185 }
186
187 irq_chips[group] = chip;
188 for (i = 0; i < chip->chip.ngpio; i++) { 220 for (i = 0; i < chip->chip.ngpio; i++) {
189 irq = chip->irq_base + i; 221 irq = chip->irq_base + i;
190 set_irq_chip(irq, &s5p_gpioint); 222 set_irq_chip(irq, &s5p_gpioint);
@@ -221,3 +253,19 @@ int __init s5p_register_gpio_interrupt(int pin)
221 } 253 }
222 return ret; 254 return ret;
223} 255}
256
257int __init s5p_register_gpioint_bank(int chain_irq, int start, int nr_groups)
258{
259 struct s5p_gpioint_bank *bank;
260
261 bank = kzalloc(sizeof(*bank), GFP_KERNEL);
262 if (!bank)
263 return -ENOMEM;
264
265 bank->start = start;
266 bank->nr_groups = nr_groups;
267 bank->irq = chain_irq;
268
269 list_add_tail(&bank->list, &banks);
270 return 0;
271}