blob: ca33bda8ec555a37ed7cd729133383cb9dd71e12 [file] [log] [blame]
Rabin Vincentd3425712015-06-06 22:30:40 +02001#include <linux/kernel.h>
2#include <linux/init.h>
3#include <linux/gpio.h>
Rabin Vincent29b53572015-07-31 14:48:57 +02004#include <linux/gpio/driver.h>
Rabin Vincentd3425712015-06-06 22:30:40 +02005#include <linux/of_gpio.h>
6#include <linux/io.h>
Rabin Vincent29b53572015-07-31 14:48:57 +02007#include <linux/interrupt.h>
Rabin Vincentd3425712015-06-06 22:30:40 +02008#include <linux/platform_device.h>
9#include <linux/basic_mmio_gpio.h>
10
11#define ETRAX_FS_rw_pa_dout 0
12#define ETRAX_FS_r_pa_din 4
13#define ETRAX_FS_rw_pa_oe 8
14#define ETRAX_FS_rw_intr_cfg 12
15#define ETRAX_FS_rw_intr_mask 16
16#define ETRAX_FS_rw_ack_intr 20
17#define ETRAX_FS_r_intr 24
Rabin Vincent29b53572015-07-31 14:48:57 +020018#define ETRAX_FS_r_masked_intr 28
Rabin Vincentd3425712015-06-06 22:30:40 +020019#define ETRAX_FS_rw_pb_dout 32
20#define ETRAX_FS_r_pb_din 36
21#define ETRAX_FS_rw_pb_oe 40
22#define ETRAX_FS_rw_pc_dout 48
23#define ETRAX_FS_r_pc_din 52
24#define ETRAX_FS_rw_pc_oe 56
25#define ETRAX_FS_rw_pd_dout 64
26#define ETRAX_FS_r_pd_din 68
27#define ETRAX_FS_rw_pd_oe 72
28#define ETRAX_FS_rw_pe_dout 80
29#define ETRAX_FS_r_pe_din 84
30#define ETRAX_FS_rw_pe_oe 88
31
Rabin Vincentd7050732015-07-22 15:05:19 +020032#define ARTPEC3_r_pa_din 0
33#define ARTPEC3_rw_pa_dout 4
34#define ARTPEC3_rw_pa_oe 8
35#define ARTPEC3_r_pb_din 44
36#define ARTPEC3_rw_pb_dout 48
37#define ARTPEC3_rw_pb_oe 52
38#define ARTPEC3_r_pc_din 88
39#define ARTPEC3_rw_pc_dout 92
40#define ARTPEC3_rw_pc_oe 96
41#define ARTPEC3_r_pd_din 116
Rabin Vincent29b53572015-07-31 14:48:57 +020042#define ARTPEC3_rw_intr_cfg 120
43#define ARTPEC3_rw_intr_pins 124
44#define ARTPEC3_rw_intr_mask 128
45#define ARTPEC3_rw_ack_intr 132
46#define ARTPEC3_r_masked_intr 140
47
48#define GIO_CFG_OFF 0
49#define GIO_CFG_HI 1
50#define GIO_CFG_LO 2
51#define GIO_CFG_SET 3
52#define GIO_CFG_POSEDGE 5
53#define GIO_CFG_NEGEDGE 6
54#define GIO_CFG_ANYEDGE 7
55
56struct etraxfs_gpio_info;
57
58struct etraxfs_gpio_block {
59 spinlock_t lock;
60 u32 mask;
61 u32 cfg;
62 u32 pins;
63 unsigned int group[8];
64
65 void __iomem *regs;
66 const struct etraxfs_gpio_info *info;
67};
68
69struct etraxfs_gpio_chip {
70 struct bgpio_chip bgc;
71 struct etraxfs_gpio_block *block;
72};
Rabin Vincentd7050732015-07-22 15:05:19 +020073
Rabin Vincentd3425712015-06-06 22:30:40 +020074struct etraxfs_gpio_port {
75 const char *label;
76 unsigned int oe;
77 unsigned int dout;
78 unsigned int din;
79 unsigned int ngpio;
80};
81
82struct etraxfs_gpio_info {
83 unsigned int num_ports;
84 const struct etraxfs_gpio_port *ports;
Rabin Vincent29b53572015-07-31 14:48:57 +020085
86 unsigned int rw_ack_intr;
87 unsigned int rw_intr_mask;
88 unsigned int rw_intr_cfg;
89 unsigned int rw_intr_pins;
90 unsigned int r_masked_intr;
Rabin Vincentd3425712015-06-06 22:30:40 +020091};
92
93static const struct etraxfs_gpio_port etraxfs_gpio_etraxfs_ports[] = {
94 {
95 .label = "A",
96 .ngpio = 8,
97 .oe = ETRAX_FS_rw_pa_oe,
98 .dout = ETRAX_FS_rw_pa_dout,
99 .din = ETRAX_FS_r_pa_din,
100 },
101 {
102 .label = "B",
103 .ngpio = 18,
104 .oe = ETRAX_FS_rw_pb_oe,
105 .dout = ETRAX_FS_rw_pb_dout,
106 .din = ETRAX_FS_r_pb_din,
107 },
108 {
109 .label = "C",
110 .ngpio = 18,
111 .oe = ETRAX_FS_rw_pc_oe,
112 .dout = ETRAX_FS_rw_pc_dout,
113 .din = ETRAX_FS_r_pc_din,
114 },
115 {
116 .label = "D",
117 .ngpio = 18,
118 .oe = ETRAX_FS_rw_pd_oe,
119 .dout = ETRAX_FS_rw_pd_dout,
120 .din = ETRAX_FS_r_pd_din,
121 },
122 {
123 .label = "E",
124 .ngpio = 18,
125 .oe = ETRAX_FS_rw_pe_oe,
126 .dout = ETRAX_FS_rw_pe_dout,
127 .din = ETRAX_FS_r_pe_din,
128 },
129};
130
131static const struct etraxfs_gpio_info etraxfs_gpio_etraxfs = {
132 .num_ports = ARRAY_SIZE(etraxfs_gpio_etraxfs_ports),
133 .ports = etraxfs_gpio_etraxfs_ports,
Rabin Vincent29b53572015-07-31 14:48:57 +0200134 .rw_ack_intr = ETRAX_FS_rw_ack_intr,
135 .rw_intr_mask = ETRAX_FS_rw_intr_mask,
136 .rw_intr_cfg = ETRAX_FS_rw_intr_cfg,
137 .r_masked_intr = ETRAX_FS_r_masked_intr,
Rabin Vincentd3425712015-06-06 22:30:40 +0200138};
139
Rabin Vincentd7050732015-07-22 15:05:19 +0200140static const struct etraxfs_gpio_port etraxfs_gpio_artpec3_ports[] = {
141 {
142 .label = "A",
143 .ngpio = 32,
144 .oe = ARTPEC3_rw_pa_oe,
145 .dout = ARTPEC3_rw_pa_dout,
146 .din = ARTPEC3_r_pa_din,
147 },
148 {
149 .label = "B",
150 .ngpio = 32,
151 .oe = ARTPEC3_rw_pb_oe,
152 .dout = ARTPEC3_rw_pb_dout,
153 .din = ARTPEC3_r_pb_din,
154 },
155 {
156 .label = "C",
157 .ngpio = 16,
158 .oe = ARTPEC3_rw_pc_oe,
159 .dout = ARTPEC3_rw_pc_dout,
160 .din = ARTPEC3_r_pc_din,
161 },
162 {
163 .label = "D",
164 .ngpio = 32,
165 .din = ARTPEC3_r_pd_din,
166 },
167};
168
169static const struct etraxfs_gpio_info etraxfs_gpio_artpec3 = {
170 .num_ports = ARRAY_SIZE(etraxfs_gpio_artpec3_ports),
171 .ports = etraxfs_gpio_artpec3_ports,
Rabin Vincent29b53572015-07-31 14:48:57 +0200172 .rw_ack_intr = ARTPEC3_rw_ack_intr,
173 .rw_intr_mask = ARTPEC3_rw_intr_mask,
174 .rw_intr_cfg = ARTPEC3_rw_intr_cfg,
175 .r_masked_intr = ARTPEC3_r_masked_intr,
176 .rw_intr_pins = ARTPEC3_rw_intr_pins,
Rabin Vincentd7050732015-07-22 15:05:19 +0200177};
178
Rabin Vincent29b53572015-07-31 14:48:57 +0200179static unsigned int etraxfs_gpio_chip_to_port(struct gpio_chip *gc)
180{
181 return gc->label[0] - 'A';
182}
183
Rabin Vincentd3425712015-06-06 22:30:40 +0200184static int etraxfs_gpio_of_xlate(struct gpio_chip *gc,
185 const struct of_phandle_args *gpiospec,
186 u32 *flags)
187{
188 /*
189 * Port numbers are A to E, and the properties are integers, so we
190 * specify them as 0xA - 0xE.
191 */
Rabin Vincent29b53572015-07-31 14:48:57 +0200192 if (etraxfs_gpio_chip_to_port(gc) + 0xA != gpiospec->args[2])
Rabin Vincentd3425712015-06-06 22:30:40 +0200193 return -EINVAL;
194
195 return of_gpio_simple_xlate(gc, gpiospec, flags);
196}
197
198static const struct of_device_id etraxfs_gpio_of_table[] = {
199 {
200 .compatible = "axis,etraxfs-gio",
201 .data = &etraxfs_gpio_etraxfs,
202 },
Rabin Vincentd7050732015-07-22 15:05:19 +0200203 {
204 .compatible = "axis,artpec3-gio",
205 .data = &etraxfs_gpio_artpec3,
206 },
Rabin Vincentd3425712015-06-06 22:30:40 +0200207 {},
208};
209
Rabin Vincent29b53572015-07-31 14:48:57 +0200210static unsigned int etraxfs_gpio_to_group_irq(unsigned int gpio)
211{
212 return gpio % 8;
213}
214
215static unsigned int etraxfs_gpio_to_group_pin(struct etraxfs_gpio_chip *chip,
216 unsigned int gpio)
217{
218 return 4 * etraxfs_gpio_chip_to_port(&chip->bgc.gc) + gpio / 8;
219}
220
221static void etraxfs_gpio_irq_ack(struct irq_data *d)
222{
223 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
224 struct etraxfs_gpio_block *block = chip->block;
225 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
226
227 writel(BIT(grpirq), block->regs + block->info->rw_ack_intr);
228}
229
230static void etraxfs_gpio_irq_mask(struct irq_data *d)
231{
232 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
233 struct etraxfs_gpio_block *block = chip->block;
234 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
235
236 spin_lock(&block->lock);
237 block->mask &= ~BIT(grpirq);
238 writel(block->mask, block->regs + block->info->rw_intr_mask);
239 spin_unlock(&block->lock);
240}
241
242static void etraxfs_gpio_irq_unmask(struct irq_data *d)
243{
244 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
245 struct etraxfs_gpio_block *block = chip->block;
246 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
247
248 spin_lock(&block->lock);
249 block->mask |= BIT(grpirq);
250 writel(block->mask, block->regs + block->info->rw_intr_mask);
251 spin_unlock(&block->lock);
252}
253
254static int etraxfs_gpio_irq_set_type(struct irq_data *d, u32 type)
255{
256 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
257 struct etraxfs_gpio_block *block = chip->block;
258 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
259 u32 cfg;
260
261 switch (type) {
262 case IRQ_TYPE_EDGE_RISING:
263 cfg = GIO_CFG_POSEDGE;
264 break;
265 case IRQ_TYPE_EDGE_FALLING:
266 cfg = GIO_CFG_NEGEDGE;
267 break;
268 case IRQ_TYPE_EDGE_BOTH:
269 cfg = GIO_CFG_ANYEDGE;
270 break;
271 case IRQ_TYPE_LEVEL_LOW:
272 cfg = GIO_CFG_LO;
273 break;
274 case IRQ_TYPE_LEVEL_HIGH:
275 cfg = GIO_CFG_HI;
276 break;
277 default:
278 return -EINVAL;
279 }
280
281 spin_lock(&block->lock);
282 block->cfg &= ~(0x7 << (grpirq * 3));
283 block->cfg |= (cfg << (grpirq * 3));
284 writel(block->cfg, block->regs + block->info->rw_intr_cfg);
285 spin_unlock(&block->lock);
286
287 return 0;
288}
289
290static int etraxfs_gpio_irq_request_resources(struct irq_data *d)
291{
292 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
293 struct etraxfs_gpio_block *block = chip->block;
294 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
Rabin Vincent29b53572015-07-31 14:48:57 +0200295
296 spin_lock(&block->lock);
297 if (block->group[grpirq])
298 goto out;
299
300 ret = gpiochip_lock_as_irq(&chip->bgc.gc, d->hwirq);
301 if (ret)
302 goto out;
303
304 block->group[grpirq] = d->irq;
305 if (block->info->rw_intr_pins) {
306 unsigned int pin = etraxfs_gpio_to_group_pin(chip, d->hwirq);
307
308 block->pins &= ~(0xf << (grpirq * 4));
309 block->pins |= (pin << (grpirq * 4));
310
311 writel(block->pins, block->regs + block->info->rw_intr_pins);
312 }
313
314out:
315 spin_unlock(&block->lock);
Julia Lawall5e22ec02015-08-13 17:41:16 +0200316 return -EBUSY;
Rabin Vincent29b53572015-07-31 14:48:57 +0200317}
318
319static void etraxfs_gpio_irq_release_resources(struct irq_data *d)
320{
321 struct etraxfs_gpio_chip *chip = irq_data_get_irq_chip_data(d);
322 struct etraxfs_gpio_block *block = chip->block;
323 unsigned int grpirq = etraxfs_gpio_to_group_irq(d->hwirq);
324
325 spin_lock(&block->lock);
326 block->group[grpirq] = 0;
327 gpiochip_unlock_as_irq(&chip->bgc.gc, d->hwirq);
328 spin_unlock(&block->lock);
329}
330
331static struct irq_chip etraxfs_gpio_irq_chip = {
332 .name = "gpio-etraxfs",
333 .irq_ack = etraxfs_gpio_irq_ack,
334 .irq_mask = etraxfs_gpio_irq_mask,
335 .irq_unmask = etraxfs_gpio_irq_unmask,
336 .irq_set_type = etraxfs_gpio_irq_set_type,
337 .irq_request_resources = etraxfs_gpio_irq_request_resources,
338 .irq_release_resources = etraxfs_gpio_irq_release_resources,
339};
340
341static irqreturn_t etraxfs_gpio_interrupt(int irq, void *dev_id)
342{
343 struct etraxfs_gpio_block *block = dev_id;
344 unsigned long intr = readl(block->regs + block->info->r_masked_intr);
345 int bit;
346
347 for_each_set_bit(bit, &intr, 8)
348 generic_handle_irq(block->group[bit]);
349
350 return IRQ_RETVAL(intr & 0xff);
351}
352
Rabin Vincentd3425712015-06-06 22:30:40 +0200353static int etraxfs_gpio_probe(struct platform_device *pdev)
354{
355 struct device *dev = &pdev->dev;
356 const struct etraxfs_gpio_info *info;
357 const struct of_device_id *match;
Rabin Vincent29b53572015-07-31 14:48:57 +0200358 struct etraxfs_gpio_block *block;
359 struct etraxfs_gpio_chip *chips;
360 struct resource *res, *irq;
361 bool allportsirq = false;
Rabin Vincentd3425712015-06-06 22:30:40 +0200362 void __iomem *regs;
363 int ret;
364 int i;
365
366 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
367 regs = devm_ioremap_resource(dev, res);
Krzysztof Kozlowski01540312015-07-09 22:19:53 +0900368 if (IS_ERR(regs))
369 return PTR_ERR(regs);
Rabin Vincentd3425712015-06-06 22:30:40 +0200370
371 match = of_match_node(etraxfs_gpio_of_table, dev->of_node);
372 if (!match)
373 return -EINVAL;
374
375 info = match->data;
376
377 chips = devm_kzalloc(dev, sizeof(*chips) * info->num_ports, GFP_KERNEL);
378 if (!chips)
379 return -ENOMEM;
380
Rabin Vincent29b53572015-07-31 14:48:57 +0200381 irq = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
382 if (!irq)
383 return -EINVAL;
384
385 block = devm_kzalloc(dev, sizeof(*block), GFP_KERNEL);
386 if (!block)
387 return -ENOMEM;
388
389 spin_lock_init(&block->lock);
390
391 block->regs = regs;
392 block->info = info;
393
394 writel(0, block->regs + info->rw_intr_mask);
395 writel(0, block->regs + info->rw_intr_cfg);
396 if (info->rw_intr_pins) {
397 allportsirq = true;
398 writel(0, block->regs + info->rw_intr_pins);
399 }
400
401 ret = devm_request_irq(dev, irq->start, etraxfs_gpio_interrupt,
402 IRQF_SHARED, dev_name(dev), block);
403 if (ret) {
404 dev_err(dev, "Unable to request irq %d\n", ret);
405 return ret;
406 }
407
Rabin Vincentd3425712015-06-06 22:30:40 +0200408 for (i = 0; i < info->num_ports; i++) {
Rabin Vincent29b53572015-07-31 14:48:57 +0200409 struct etraxfs_gpio_chip *chip = &chips[i];
410 struct bgpio_chip *bgc = &chip->bgc;
Rabin Vincentd3425712015-06-06 22:30:40 +0200411 const struct etraxfs_gpio_port *port = &info->ports[i];
Rabin Vincentd7050732015-07-22 15:05:19 +0200412 unsigned long flags = BGPIOF_READ_OUTPUT_REG_SET;
413 void __iomem *dat = regs + port->din;
414 void __iomem *set = regs + port->dout;
415 void __iomem *dirout = regs + port->oe;
416
Rabin Vincent29b53572015-07-31 14:48:57 +0200417 chip->block = block;
418
Rabin Vincentd7050732015-07-22 15:05:19 +0200419 if (dirout == set) {
420 dirout = set = NULL;
421 flags = BGPIOF_NO_OUTPUT;
422 }
Rabin Vincentd3425712015-06-06 22:30:40 +0200423
424 ret = bgpio_init(bgc, dev, 4,
Rabin Vincentd7050732015-07-22 15:05:19 +0200425 dat, set, NULL, dirout, NULL,
426 flags);
Rabin Vincent29b53572015-07-31 14:48:57 +0200427 if (ret) {
428 dev_err(dev, "Unable to init port %s\n",
429 port->label);
430 continue;
431 }
Rabin Vincentd3425712015-06-06 22:30:40 +0200432
433 bgc->gc.ngpio = port->ngpio;
434 bgc->gc.label = port->label;
435
436 bgc->gc.of_node = dev->of_node;
437 bgc->gc.of_gpio_n_cells = 3;
438 bgc->gc.of_xlate = etraxfs_gpio_of_xlate;
439
440 ret = gpiochip_add(&bgc->gc);
Rabin Vincent29b53572015-07-31 14:48:57 +0200441 if (ret) {
Rabin Vincentd3425712015-06-06 22:30:40 +0200442 dev_err(dev, "Unable to register port %s\n",
443 bgc->gc.label);
Rabin Vincent29b53572015-07-31 14:48:57 +0200444 continue;
445 }
446
447 if (i > 0 && !allportsirq)
448 continue;
449
450 ret = gpiochip_irqchip_add(&bgc->gc, &etraxfs_gpio_irq_chip, 0,
451 handle_level_irq, IRQ_TYPE_NONE);
452 if (ret) {
453 dev_err(dev, "Unable to add irqchip to port %s\n",
454 bgc->gc.label);
455 }
Rabin Vincentd3425712015-06-06 22:30:40 +0200456 }
457
458 return 0;
459}
460
461static struct platform_driver etraxfs_gpio_driver = {
462 .driver = {
463 .name = "etraxfs-gpio",
464 .of_match_table = of_match_ptr(etraxfs_gpio_of_table),
465 },
466 .probe = etraxfs_gpio_probe,
467};
468
469static int __init etraxfs_gpio_init(void)
470{
471 return platform_driver_register(&etraxfs_gpio_driver);
472}
473
474device_initcall(etraxfs_gpio_init);