1a1d1e0e3SJonathan Neuschäfer // SPDX-License-Identifier: GPL-2.0
2a1d1e0e3SJonathan Neuschäfer // Copyright (c) 2016-2018 Nuvoton Technology corporation.
3a1d1e0e3SJonathan Neuschäfer // Copyright (c) 2016, Dell Inc
4a1d1e0e3SJonathan Neuschäfer // Copyright (c) 2021-2022 Jonathan Neuschäfer
5a1d1e0e3SJonathan Neuschäfer //
6a1d1e0e3SJonathan Neuschäfer // This driver uses the following registers:
7a1d1e0e3SJonathan Neuschäfer // - Pin mux registers, in the GCR (general control registers) block
8a1d1e0e3SJonathan Neuschäfer // - GPIO registers, specific to each GPIO bank
9a1d1e0e3SJonathan Neuschäfer // - GPIO event (interrupt) registers, located centrally in the GPIO register
10a1d1e0e3SJonathan Neuschäfer // block, shared between all GPIO banks
11a1d1e0e3SJonathan Neuschäfer
12a1d1e0e3SJonathan Neuschäfer #include <linux/device.h>
13a1d1e0e3SJonathan Neuschäfer #include <linux/fwnode.h>
14a1d1e0e3SJonathan Neuschäfer #include <linux/gpio/driver.h>
15a1d1e0e3SJonathan Neuschäfer #include <linux/interrupt.h>
16a1d1e0e3SJonathan Neuschäfer #include <linux/irq.h>
17a1d1e0e3SJonathan Neuschäfer #include <linux/mfd/syscon.h>
18a1d1e0e3SJonathan Neuschäfer #include <linux/module.h>
19a1d1e0e3SJonathan Neuschäfer #include <linux/mod_devicetable.h>
20a1d1e0e3SJonathan Neuschäfer #include <linux/platform_device.h>
21a1d1e0e3SJonathan Neuschäfer #include <linux/regmap.h>
22a1d1e0e3SJonathan Neuschäfer
23a1d1e0e3SJonathan Neuschäfer #include <linux/pinctrl/pinconf.h>
24a1d1e0e3SJonathan Neuschäfer #include <linux/pinctrl/pinconf-generic.h>
25a1d1e0e3SJonathan Neuschäfer #include <linux/pinctrl/pinctrl.h>
26a1d1e0e3SJonathan Neuschäfer #include <linux/pinctrl/pinmux.h>
27a1d1e0e3SJonathan Neuschäfer
28a1d1e0e3SJonathan Neuschäfer #include "../core.h"
29a1d1e0e3SJonathan Neuschäfer
30a1d1e0e3SJonathan Neuschäfer /* GCR registers */
31a1d1e0e3SJonathan Neuschäfer #define WPCM450_GCR_MFSEL1 0x0c
32a1d1e0e3SJonathan Neuschäfer #define WPCM450_GCR_MFSEL2 0x10
33a1d1e0e3SJonathan Neuschäfer #define WPCM450_GCR_NONE 0
34a1d1e0e3SJonathan Neuschäfer
35a1d1e0e3SJonathan Neuschäfer /* GPIO event (interrupt) registers */
36a1d1e0e3SJonathan Neuschäfer #define WPCM450_GPEVTYPE 0x00
37a1d1e0e3SJonathan Neuschäfer #define WPCM450_GPEVPOL 0x04
38a1d1e0e3SJonathan Neuschäfer #define WPCM450_GPEVDBNC 0x08
39a1d1e0e3SJonathan Neuschäfer #define WPCM450_GPEVEN 0x0c
40a1d1e0e3SJonathan Neuschäfer #define WPCM450_GPEVST 0x10
41a1d1e0e3SJonathan Neuschäfer
42a1d1e0e3SJonathan Neuschäfer #define WPCM450_NUM_BANKS 8
43a1d1e0e3SJonathan Neuschäfer #define WPCM450_NUM_GPIOS 128
44a1d1e0e3SJonathan Neuschäfer #define WPCM450_NUM_GPIO_IRQS 4
45a1d1e0e3SJonathan Neuschäfer
46a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl;
47a1d1e0e3SJonathan Neuschäfer struct wpcm450_bank;
48a1d1e0e3SJonathan Neuschäfer
49a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio {
50a1d1e0e3SJonathan Neuschäfer struct gpio_chip gc;
51a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl;
52a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank;
53a1d1e0e3SJonathan Neuschäfer };
54a1d1e0e3SJonathan Neuschäfer
55a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl {
56a1d1e0e3SJonathan Neuschäfer struct pinctrl_dev *pctldev;
57a1d1e0e3SJonathan Neuschäfer struct device *dev;
58a1d1e0e3SJonathan Neuschäfer struct irq_domain *domain;
59a1d1e0e3SJonathan Neuschäfer struct regmap *gcr_regmap;
60a1d1e0e3SJonathan Neuschäfer void __iomem *gpio_base;
61a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio gpio_bank[WPCM450_NUM_BANKS];
62a1d1e0e3SJonathan Neuschäfer unsigned long both_edges;
63a1d1e0e3SJonathan Neuschäfer
64a1d1e0e3SJonathan Neuschäfer /*
65a1d1e0e3SJonathan Neuschäfer * This spin lock protects registers and struct wpcm450_pinctrl fields
66a1d1e0e3SJonathan Neuschäfer * against concurrent access.
67a1d1e0e3SJonathan Neuschäfer */
68a1d1e0e3SJonathan Neuschäfer raw_spinlock_t lock;
69a1d1e0e3SJonathan Neuschäfer };
70a1d1e0e3SJonathan Neuschäfer
71a1d1e0e3SJonathan Neuschäfer struct wpcm450_bank {
72a1d1e0e3SJonathan Neuschäfer /* Range of GPIOs in this port */
73a1d1e0e3SJonathan Neuschäfer u8 base;
74a1d1e0e3SJonathan Neuschäfer u8 length;
75a1d1e0e3SJonathan Neuschäfer
76a1d1e0e3SJonathan Neuschäfer /* Register offsets (0 = register doesn't exist in this port) */
77a1d1e0e3SJonathan Neuschäfer u8 cfg0, cfg1, cfg2;
78a1d1e0e3SJonathan Neuschäfer u8 blink;
79a1d1e0e3SJonathan Neuschäfer u8 dataout, datain;
80a1d1e0e3SJonathan Neuschäfer
81a1d1e0e3SJonathan Neuschäfer /* Interrupt bit mapping */
82a1d1e0e3SJonathan Neuschäfer u8 first_irq_bit; /* First bit in GPEVST that belongs to this bank */
83a1d1e0e3SJonathan Neuschäfer u8 num_irqs; /* Number of IRQ-capable GPIOs in this bank */
84a1d1e0e3SJonathan Neuschäfer u8 first_irq_gpio; /* First IRQ-capable GPIO in this bank */
85a1d1e0e3SJonathan Neuschäfer };
86a1d1e0e3SJonathan Neuschäfer
87a1d1e0e3SJonathan Neuschäfer static const struct wpcm450_bank wpcm450_banks[WPCM450_NUM_BANKS] = {
88a1d1e0e3SJonathan Neuschäfer /* range cfg0 cfg1 cfg2 blink out in IRQ map */
89a1d1e0e3SJonathan Neuschäfer { 0, 16, 0x14, 0x18, 0, 0, 0x1c, 0x20, 0, 16, 0 },
90a1d1e0e3SJonathan Neuschäfer { 16, 16, 0x24, 0x28, 0x2c, 0x30, 0x34, 0x38, 16, 2, 8 },
91a1d1e0e3SJonathan Neuschäfer { 32, 16, 0x3c, 0x40, 0x44, 0, 0x48, 0x4c, 0, 0, 0 },
92a1d1e0e3SJonathan Neuschäfer { 48, 16, 0x50, 0x54, 0x58, 0, 0x5c, 0x60, 0, 0, 0 },
93a1d1e0e3SJonathan Neuschäfer { 64, 16, 0x64, 0x68, 0x6c, 0, 0x70, 0x74, 0, 0, 0 },
94a1d1e0e3SJonathan Neuschäfer { 80, 16, 0x78, 0x7c, 0x80, 0, 0x84, 0x88, 0, 0, 0 },
95a1d1e0e3SJonathan Neuschäfer { 96, 18, 0, 0, 0, 0, 0, 0x8c, 0, 0, 0 },
96a1d1e0e3SJonathan Neuschäfer { 114, 14, 0x90, 0x94, 0x98, 0, 0x9c, 0xa0, 0, 0, 0 },
97a1d1e0e3SJonathan Neuschäfer };
98a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_irq_bitnum(struct wpcm450_gpio * gpio,struct irq_data * d)99a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_irq_bitnum(struct wpcm450_gpio *gpio, struct irq_data *d)
100a1d1e0e3SJonathan Neuschäfer {
101a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank = gpio->bank;
102a1d1e0e3SJonathan Neuschäfer int hwirq = irqd_to_hwirq(d);
103a1d1e0e3SJonathan Neuschäfer
104a1d1e0e3SJonathan Neuschäfer if (hwirq < bank->first_irq_gpio)
105a1d1e0e3SJonathan Neuschäfer return -EINVAL;
106a1d1e0e3SJonathan Neuschäfer
107a1d1e0e3SJonathan Neuschäfer if (hwirq - bank->first_irq_gpio >= bank->num_irqs)
108a1d1e0e3SJonathan Neuschäfer return -EINVAL;
109a1d1e0e3SJonathan Neuschäfer
110a1d1e0e3SJonathan Neuschäfer return hwirq - bank->first_irq_gpio + bank->first_irq_bit;
111a1d1e0e3SJonathan Neuschäfer }
112a1d1e0e3SJonathan Neuschäfer
wpcm450_irq_bitnum_to_gpio(struct wpcm450_gpio * gpio,int bitnum)113a1d1e0e3SJonathan Neuschäfer static int wpcm450_irq_bitnum_to_gpio(struct wpcm450_gpio *gpio, int bitnum)
114a1d1e0e3SJonathan Neuschäfer {
115a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank = gpio->bank;
116a1d1e0e3SJonathan Neuschäfer
117a1d1e0e3SJonathan Neuschäfer if (bitnum < bank->first_irq_bit)
118a1d1e0e3SJonathan Neuschäfer return -EINVAL;
119a1d1e0e3SJonathan Neuschäfer
120a1d1e0e3SJonathan Neuschäfer if (bitnum - bank->first_irq_bit > bank->num_irqs)
121a1d1e0e3SJonathan Neuschäfer return -EINVAL;
122a1d1e0e3SJonathan Neuschäfer
123a1d1e0e3SJonathan Neuschäfer return bitnum - bank->first_irq_bit + bank->first_irq_gpio;
124a1d1e0e3SJonathan Neuschäfer }
125a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_irq_ack(struct irq_data * d)126a1d1e0e3SJonathan Neuschäfer static void wpcm450_gpio_irq_ack(struct irq_data *d)
127a1d1e0e3SJonathan Neuschäfer {
128a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(irq_data_get_irq_chip_data(d));
129a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
130a1d1e0e3SJonathan Neuschäfer unsigned long flags;
131a1d1e0e3SJonathan Neuschäfer int bit;
132a1d1e0e3SJonathan Neuschäfer
133a1d1e0e3SJonathan Neuschäfer bit = wpcm450_gpio_irq_bitnum(gpio, d);
134a1d1e0e3SJonathan Neuschäfer if (bit < 0)
135a1d1e0e3SJonathan Neuschäfer return;
136a1d1e0e3SJonathan Neuschäfer
137a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
138a1d1e0e3SJonathan Neuschäfer iowrite32(BIT(bit), pctrl->gpio_base + WPCM450_GPEVST);
139a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
140a1d1e0e3SJonathan Neuschäfer }
141a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_irq_mask(struct irq_data * d)142a1d1e0e3SJonathan Neuschäfer static void wpcm450_gpio_irq_mask(struct irq_data *d)
143a1d1e0e3SJonathan Neuschäfer {
14497775ebbSJonathan Neuschäfer struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
14597775ebbSJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(gc);
146a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
147a1d1e0e3SJonathan Neuschäfer unsigned long flags;
148a1d1e0e3SJonathan Neuschäfer unsigned long even;
149a1d1e0e3SJonathan Neuschäfer int bit;
150a1d1e0e3SJonathan Neuschäfer
151a1d1e0e3SJonathan Neuschäfer bit = wpcm450_gpio_irq_bitnum(gpio, d);
152a1d1e0e3SJonathan Neuschäfer if (bit < 0)
153a1d1e0e3SJonathan Neuschäfer return;
154a1d1e0e3SJonathan Neuschäfer
155a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
156a1d1e0e3SJonathan Neuschäfer even = ioread32(pctrl->gpio_base + WPCM450_GPEVEN);
157a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &even, 0);
158a1d1e0e3SJonathan Neuschäfer iowrite32(even, pctrl->gpio_base + WPCM450_GPEVEN);
159a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
16097775ebbSJonathan Neuschäfer
16197775ebbSJonathan Neuschäfer gpiochip_disable_irq(gc, irqd_to_hwirq(d));
162a1d1e0e3SJonathan Neuschäfer }
163a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_irq_unmask(struct irq_data * d)164a1d1e0e3SJonathan Neuschäfer static void wpcm450_gpio_irq_unmask(struct irq_data *d)
165a1d1e0e3SJonathan Neuschäfer {
16697775ebbSJonathan Neuschäfer struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
16797775ebbSJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(gc);
168a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
169a1d1e0e3SJonathan Neuschäfer unsigned long flags;
170a1d1e0e3SJonathan Neuschäfer unsigned long even;
171a1d1e0e3SJonathan Neuschäfer int bit;
172a1d1e0e3SJonathan Neuschäfer
173a1d1e0e3SJonathan Neuschäfer bit = wpcm450_gpio_irq_bitnum(gpio, d);
174a1d1e0e3SJonathan Neuschäfer if (bit < 0)
175a1d1e0e3SJonathan Neuschäfer return;
176a1d1e0e3SJonathan Neuschäfer
17797775ebbSJonathan Neuschäfer gpiochip_enable_irq(gc, irqd_to_hwirq(d));
17897775ebbSJonathan Neuschäfer
179a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
180a1d1e0e3SJonathan Neuschäfer even = ioread32(pctrl->gpio_base + WPCM450_GPEVEN);
181a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &even, 1);
182a1d1e0e3SJonathan Neuschäfer iowrite32(even, pctrl->gpio_base + WPCM450_GPEVEN);
183a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
184a1d1e0e3SJonathan Neuschäfer }
185a1d1e0e3SJonathan Neuschäfer
186a1d1e0e3SJonathan Neuschäfer /*
187a1d1e0e3SJonathan Neuschäfer * This is an implementation of the gpio_chip->get() function, for use in
188a1d1e0e3SJonathan Neuschäfer * wpcm450_gpio_fix_evpol. Unfortunately, we can't use the bgpio-provided
189a1d1e0e3SJonathan Neuschäfer * implementation there, because it would require taking gpio_chip->bgpio_lock,
190a1d1e0e3SJonathan Neuschäfer * which is a spin lock, but wpcm450_gpio_fix_evpol must work in contexts where
191a1d1e0e3SJonathan Neuschäfer * a raw spin lock is held.
192a1d1e0e3SJonathan Neuschäfer */
wpcm450_gpio_get(struct wpcm450_gpio * gpio,int offset)193a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_get(struct wpcm450_gpio *gpio, int offset)
194a1d1e0e3SJonathan Neuschäfer {
195a1d1e0e3SJonathan Neuschäfer void __iomem *reg = gpio->pctrl->gpio_base + gpio->bank->datain;
196a1d1e0e3SJonathan Neuschäfer unsigned long flags;
197a1d1e0e3SJonathan Neuschäfer u32 level;
198a1d1e0e3SJonathan Neuschäfer
199a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&gpio->pctrl->lock, flags);
200a1d1e0e3SJonathan Neuschäfer level = !!(ioread32(reg) & BIT(offset));
201a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&gpio->pctrl->lock, flags);
202a1d1e0e3SJonathan Neuschäfer
203a1d1e0e3SJonathan Neuschäfer return level;
204a1d1e0e3SJonathan Neuschäfer }
205a1d1e0e3SJonathan Neuschäfer
206a1d1e0e3SJonathan Neuschäfer /*
207a1d1e0e3SJonathan Neuschäfer * Since the GPIO controller does not support dual-edge triggered interrupts
208a1d1e0e3SJonathan Neuschäfer * (IRQ_TYPE_EDGE_BOTH), they are emulated using rising/falling edge triggered
209a1d1e0e3SJonathan Neuschäfer * interrupts. wpcm450_gpio_fix_evpol sets the interrupt polarity for the
210a1d1e0e3SJonathan Neuschäfer * specified emulated dual-edge triggered interrupts, so that the next edge can
211a1d1e0e3SJonathan Neuschäfer * be detected.
212a1d1e0e3SJonathan Neuschäfer */
wpcm450_gpio_fix_evpol(struct wpcm450_gpio * gpio,unsigned long all)213a1d1e0e3SJonathan Neuschäfer static void wpcm450_gpio_fix_evpol(struct wpcm450_gpio *gpio, unsigned long all)
214a1d1e0e3SJonathan Neuschäfer {
215a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
216a1d1e0e3SJonathan Neuschäfer unsigned int bit;
217a1d1e0e3SJonathan Neuschäfer
218a1d1e0e3SJonathan Neuschäfer for_each_set_bit(bit, &all, 32) {
219a1d1e0e3SJonathan Neuschäfer int offset = wpcm450_irq_bitnum_to_gpio(gpio, bit);
220a1d1e0e3SJonathan Neuschäfer unsigned long evpol;
221a1d1e0e3SJonathan Neuschäfer unsigned long flags;
222a1d1e0e3SJonathan Neuschäfer int level;
223a1d1e0e3SJonathan Neuschäfer
224a1d1e0e3SJonathan Neuschäfer do {
225a1d1e0e3SJonathan Neuschäfer level = wpcm450_gpio_get(gpio, offset);
226a1d1e0e3SJonathan Neuschäfer
227a1d1e0e3SJonathan Neuschäfer /* Switch event polarity to the opposite of the current level */
228a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
229a1d1e0e3SJonathan Neuschäfer evpol = ioread32(pctrl->gpio_base + WPCM450_GPEVPOL);
230a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evpol, !level);
231a1d1e0e3SJonathan Neuschäfer iowrite32(evpol, pctrl->gpio_base + WPCM450_GPEVPOL);
232a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
233a1d1e0e3SJonathan Neuschäfer
234a1d1e0e3SJonathan Neuschäfer } while (wpcm450_gpio_get(gpio, offset) != level);
235a1d1e0e3SJonathan Neuschäfer }
236a1d1e0e3SJonathan Neuschäfer }
237a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_set_irq_type(struct irq_data * d,unsigned int flow_type)238a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_set_irq_type(struct irq_data *d, unsigned int flow_type)
239a1d1e0e3SJonathan Neuschäfer {
240a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(irq_data_get_irq_chip_data(d));
241a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
242a1d1e0e3SJonathan Neuschäfer unsigned long evtype, evpol;
243a1d1e0e3SJonathan Neuschäfer unsigned long flags;
244a1d1e0e3SJonathan Neuschäfer int ret = 0;
245a1d1e0e3SJonathan Neuschäfer int bit;
246a1d1e0e3SJonathan Neuschäfer
247a1d1e0e3SJonathan Neuschäfer bit = wpcm450_gpio_irq_bitnum(gpio, d);
248a1d1e0e3SJonathan Neuschäfer if (bit < 0)
249a1d1e0e3SJonathan Neuschäfer return bit;
250a1d1e0e3SJonathan Neuschäfer
251a1d1e0e3SJonathan Neuschäfer irq_set_handler_locked(d, handle_level_irq);
252a1d1e0e3SJonathan Neuschäfer
253a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
254a1d1e0e3SJonathan Neuschäfer evtype = ioread32(pctrl->gpio_base + WPCM450_GPEVTYPE);
255a1d1e0e3SJonathan Neuschäfer evpol = ioread32(pctrl->gpio_base + WPCM450_GPEVPOL);
256a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &pctrl->both_edges, 0);
257a1d1e0e3SJonathan Neuschäfer switch (flow_type) {
258a1d1e0e3SJonathan Neuschäfer case IRQ_TYPE_LEVEL_LOW:
259a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evtype, 1);
260a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evpol, 0);
261a1d1e0e3SJonathan Neuschäfer break;
262a1d1e0e3SJonathan Neuschäfer case IRQ_TYPE_LEVEL_HIGH:
263a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evtype, 1);
264a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evpol, 1);
265a1d1e0e3SJonathan Neuschäfer break;
266a1d1e0e3SJonathan Neuschäfer case IRQ_TYPE_EDGE_FALLING:
267a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evtype, 0);
268a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evpol, 0);
269a1d1e0e3SJonathan Neuschäfer break;
270a1d1e0e3SJonathan Neuschäfer case IRQ_TYPE_EDGE_RISING:
271a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evtype, 0);
272a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evpol, 1);
273a1d1e0e3SJonathan Neuschäfer break;
274a1d1e0e3SJonathan Neuschäfer case IRQ_TYPE_EDGE_BOTH:
275a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &evtype, 0);
276a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, &pctrl->both_edges, 1);
277a1d1e0e3SJonathan Neuschäfer break;
278a1d1e0e3SJonathan Neuschäfer default:
279a1d1e0e3SJonathan Neuschäfer ret = -EINVAL;
280a1d1e0e3SJonathan Neuschäfer }
281a1d1e0e3SJonathan Neuschäfer iowrite32(evtype, pctrl->gpio_base + WPCM450_GPEVTYPE);
282a1d1e0e3SJonathan Neuschäfer iowrite32(evpol, pctrl->gpio_base + WPCM450_GPEVPOL);
283a1d1e0e3SJonathan Neuschäfer
284a1d1e0e3SJonathan Neuschäfer /* clear the event status for good measure */
285a1d1e0e3SJonathan Neuschäfer iowrite32(BIT(bit), pctrl->gpio_base + WPCM450_GPEVST);
286a1d1e0e3SJonathan Neuschäfer
287a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
288a1d1e0e3SJonathan Neuschäfer
289a1d1e0e3SJonathan Neuschäfer /* fix event polarity after clearing event status */
290a1d1e0e3SJonathan Neuschäfer wpcm450_gpio_fix_evpol(gpio, BIT(bit));
291a1d1e0e3SJonathan Neuschäfer
292a1d1e0e3SJonathan Neuschäfer return ret;
293a1d1e0e3SJonathan Neuschäfer }
294a1d1e0e3SJonathan Neuschäfer
295a1d1e0e3SJonathan Neuschäfer static const struct irq_chip wpcm450_gpio_irqchip = {
296a1d1e0e3SJonathan Neuschäfer .name = "WPCM450-GPIO-IRQ",
297a1d1e0e3SJonathan Neuschäfer .irq_ack = wpcm450_gpio_irq_ack,
298a1d1e0e3SJonathan Neuschäfer .irq_unmask = wpcm450_gpio_irq_unmask,
299a1d1e0e3SJonathan Neuschäfer .irq_mask = wpcm450_gpio_irq_mask,
300a1d1e0e3SJonathan Neuschäfer .irq_set_type = wpcm450_gpio_set_irq_type,
30197775ebbSJonathan Neuschäfer .flags = IRQCHIP_IMMUTABLE,
30297775ebbSJonathan Neuschäfer GPIOCHIP_IRQ_RESOURCE_HELPERS,
303a1d1e0e3SJonathan Neuschäfer };
304a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_irqhandler(struct irq_desc * desc)305a1d1e0e3SJonathan Neuschäfer static void wpcm450_gpio_irqhandler(struct irq_desc *desc)
306a1d1e0e3SJonathan Neuschäfer {
307a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(irq_desc_get_handler_data(desc));
308a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = gpio->pctrl;
309a1d1e0e3SJonathan Neuschäfer struct irq_chip *chip = irq_desc_get_chip(desc);
310a1d1e0e3SJonathan Neuschäfer unsigned long pending;
311a1d1e0e3SJonathan Neuschäfer unsigned long flags;
312a1d1e0e3SJonathan Neuschäfer unsigned long ours;
313a1d1e0e3SJonathan Neuschäfer unsigned int bit;
314a1d1e0e3SJonathan Neuschäfer
315a1d1e0e3SJonathan Neuschäfer ours = GENMASK(gpio->bank->num_irqs - 1, 0) << gpio->bank->first_irq_bit;
316a1d1e0e3SJonathan Neuschäfer
317a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
318a1d1e0e3SJonathan Neuschäfer
319a1d1e0e3SJonathan Neuschäfer pending = ioread32(pctrl->gpio_base + WPCM450_GPEVST);
320a1d1e0e3SJonathan Neuschäfer pending &= ioread32(pctrl->gpio_base + WPCM450_GPEVEN);
321a1d1e0e3SJonathan Neuschäfer pending &= ours;
322a1d1e0e3SJonathan Neuschäfer
323a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
324a1d1e0e3SJonathan Neuschäfer
325a1d1e0e3SJonathan Neuschäfer if (pending & pctrl->both_edges)
326a1d1e0e3SJonathan Neuschäfer wpcm450_gpio_fix_evpol(gpio, pending & pctrl->both_edges);
327a1d1e0e3SJonathan Neuschäfer
328a1d1e0e3SJonathan Neuschäfer chained_irq_enter(chip, desc);
329a1d1e0e3SJonathan Neuschäfer for_each_set_bit(bit, &pending, 32) {
330a1d1e0e3SJonathan Neuschäfer int offset = wpcm450_irq_bitnum_to_gpio(gpio, bit);
331a1d1e0e3SJonathan Neuschäfer
332a1d1e0e3SJonathan Neuschäfer generic_handle_domain_irq(gpio->gc.irq.domain, offset);
333a1d1e0e3SJonathan Neuschäfer }
334a1d1e0e3SJonathan Neuschäfer chained_irq_exit(chip, desc);
335a1d1e0e3SJonathan Neuschäfer }
336a1d1e0e3SJonathan Neuschäfer
337a1d1e0e3SJonathan Neuschäfer static int smb0_pins[] = { 115, 114 };
338a1d1e0e3SJonathan Neuschäfer static int smb1_pins[] = { 117, 116 };
339a1d1e0e3SJonathan Neuschäfer static int smb2_pins[] = { 119, 118 };
340a1d1e0e3SJonathan Neuschäfer static int smb3_pins[] = { 30, 31 };
341a1d1e0e3SJonathan Neuschäfer static int smb4_pins[] = { 28, 29 };
342a1d1e0e3SJonathan Neuschäfer static int smb5_pins[] = { 26, 27 };
343a1d1e0e3SJonathan Neuschäfer
344a1d1e0e3SJonathan Neuschäfer static int scs1_pins[] = { 32 };
345a1d1e0e3SJonathan Neuschäfer static int scs2_pins[] = { 33 };
346a1d1e0e3SJonathan Neuschäfer static int scs3_pins[] = { 34 };
347a1d1e0e3SJonathan Neuschäfer
348a1d1e0e3SJonathan Neuschäfer static int bsp_pins[] = { 41, 42 };
349a1d1e0e3SJonathan Neuschäfer static int hsp1_pins[] = { 43, 44, 45, 46, 47, 61, 62, 63 };
350a1d1e0e3SJonathan Neuschäfer static int hsp2_pins[] = { 48, 49, 50, 51, 52, 53, 54, 55 };
351a1d1e0e3SJonathan Neuschäfer
352a1d1e0e3SJonathan Neuschäfer static int r1err_pins[] = { 56 };
353a1d1e0e3SJonathan Neuschäfer static int r1md_pins[] = { 57, 58 };
354a1d1e0e3SJonathan Neuschäfer static int rmii2_pins[] = { 84, 85, 86, 87, 88, 89 };
355a1d1e0e3SJonathan Neuschäfer static int r2err_pins[] = { 90 };
356a1d1e0e3SJonathan Neuschäfer static int r2md_pins[] = { 91, 92 };
357a1d1e0e3SJonathan Neuschäfer
358a1d1e0e3SJonathan Neuschäfer static int kbcc_pins[] = { 94, 93 };
359a1d1e0e3SJonathan Neuschäfer static int clko_pins[] = { 96 };
360a1d1e0e3SJonathan Neuschäfer static int smi_pins[] = { 97 };
361a1d1e0e3SJonathan Neuschäfer static int uinc_pins[] = { 19 };
362a1d1e0e3SJonathan Neuschäfer static int mben_pins[] = {};
363a1d1e0e3SJonathan Neuschäfer
364a1d1e0e3SJonathan Neuschäfer static int gspi_pins[] = { 12, 13, 14, 15 };
365a1d1e0e3SJonathan Neuschäfer static int sspi_pins[] = { 12, 13, 14, 15 };
366a1d1e0e3SJonathan Neuschäfer
367a1d1e0e3SJonathan Neuschäfer static int xcs1_pins[] = { 35 };
368a1d1e0e3SJonathan Neuschäfer static int xcs2_pins[] = { 36 };
369a1d1e0e3SJonathan Neuschäfer
370a1d1e0e3SJonathan Neuschäfer static int sdio_pins[] = { 7, 22, 43, 44, 45, 46, 47, 60 };
371a1d1e0e3SJonathan Neuschäfer
372a1d1e0e3SJonathan Neuschäfer static int fi0_pins[] = { 64 };
373a1d1e0e3SJonathan Neuschäfer static int fi1_pins[] = { 65 };
374a1d1e0e3SJonathan Neuschäfer static int fi2_pins[] = { 66 };
375a1d1e0e3SJonathan Neuschäfer static int fi3_pins[] = { 67 };
376a1d1e0e3SJonathan Neuschäfer static int fi4_pins[] = { 68 };
377a1d1e0e3SJonathan Neuschäfer static int fi5_pins[] = { 69 };
378a1d1e0e3SJonathan Neuschäfer static int fi6_pins[] = { 70 };
379a1d1e0e3SJonathan Neuschäfer static int fi7_pins[] = { 71 };
380a1d1e0e3SJonathan Neuschäfer static int fi8_pins[] = { 72 };
381a1d1e0e3SJonathan Neuschäfer static int fi9_pins[] = { 73 };
382a1d1e0e3SJonathan Neuschäfer static int fi10_pins[] = { 74 };
383a1d1e0e3SJonathan Neuschäfer static int fi11_pins[] = { 75 };
384a1d1e0e3SJonathan Neuschäfer static int fi12_pins[] = { 76 };
385a1d1e0e3SJonathan Neuschäfer static int fi13_pins[] = { 77 };
386a1d1e0e3SJonathan Neuschäfer static int fi14_pins[] = { 78 };
387a1d1e0e3SJonathan Neuschäfer static int fi15_pins[] = { 79 };
388a1d1e0e3SJonathan Neuschäfer
389a1d1e0e3SJonathan Neuschäfer static int pwm0_pins[] = { 80 };
390a1d1e0e3SJonathan Neuschäfer static int pwm1_pins[] = { 81 };
391a1d1e0e3SJonathan Neuschäfer static int pwm2_pins[] = { 82 };
392a1d1e0e3SJonathan Neuschäfer static int pwm3_pins[] = { 83 };
393a1d1e0e3SJonathan Neuschäfer static int pwm4_pins[] = { 20 };
394a1d1e0e3SJonathan Neuschäfer static int pwm5_pins[] = { 21 };
395a1d1e0e3SJonathan Neuschäfer static int pwm6_pins[] = { 16 };
396a1d1e0e3SJonathan Neuschäfer static int pwm7_pins[] = { 17 };
397a1d1e0e3SJonathan Neuschäfer
398a1d1e0e3SJonathan Neuschäfer static int hg0_pins[] = { 20 };
399a1d1e0e3SJonathan Neuschäfer static int hg1_pins[] = { 21 };
400a1d1e0e3SJonathan Neuschäfer static int hg2_pins[] = { 22 };
401a1d1e0e3SJonathan Neuschäfer static int hg3_pins[] = { 23 };
402a1d1e0e3SJonathan Neuschäfer static int hg4_pins[] = { 24 };
403a1d1e0e3SJonathan Neuschäfer static int hg5_pins[] = { 25 };
404a1d1e0e3SJonathan Neuschäfer static int hg6_pins[] = { 59 };
405a1d1e0e3SJonathan Neuschäfer static int hg7_pins[] = { 60 };
406a1d1e0e3SJonathan Neuschäfer
407a1d1e0e3SJonathan Neuschäfer #define WPCM450_GRPS \
408a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb3), \
409a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb4), \
410a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb5), \
411a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(scs1), \
412a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(scs2), \
413a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(scs3), \
414a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb0), \
415a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb1), \
416a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smb2), \
417a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(bsp), \
418a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hsp1), \
419a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hsp2), \
420a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(r1err), \
421a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(r1md), \
422a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(rmii2), \
423a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(r2err), \
424a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(r2md), \
425a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(kbcc), \
426a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(clko), \
427a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(smi), \
428a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(uinc), \
429a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(gspi), \
430a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(mben), \
431a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(xcs2), \
432a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(xcs1), \
433a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(sdio), \
434a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(sspi), \
435a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi0), \
436a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi1), \
437a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi2), \
438a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi3), \
439a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi4), \
440a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi5), \
441a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi6), \
442a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi7), \
443a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi8), \
444a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi9), \
445a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi10), \
446a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi11), \
447a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi12), \
448a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi13), \
449a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi14), \
450a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(fi15), \
451a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm0), \
452a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm1), \
453a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm2), \
454a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm3), \
455a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm4), \
456a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm5), \
457a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm6), \
458a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(pwm7), \
459a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg0), \
460a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg1), \
461a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg2), \
462a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg3), \
463a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg4), \
464a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg5), \
465a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg6), \
466a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(hg7), \
467a1d1e0e3SJonathan Neuschäfer
468a1d1e0e3SJonathan Neuschäfer enum {
469a1d1e0e3SJonathan Neuschäfer #define WPCM450_GRP(x) fn_ ## x
470a1d1e0e3SJonathan Neuschäfer WPCM450_GRPS
471a1d1e0e3SJonathan Neuschäfer /* add placeholder for none/gpio */
472a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(gpio),
473a1d1e0e3SJonathan Neuschäfer WPCM450_GRP(none),
474a1d1e0e3SJonathan Neuschäfer #undef WPCM450_GRP
475a1d1e0e3SJonathan Neuschäfer };
476a1d1e0e3SJonathan Neuschäfer
477a1d1e0e3SJonathan Neuschäfer static struct group_desc wpcm450_groups[] = {
478a1d1e0e3SJonathan Neuschäfer #define WPCM450_GRP(x) { .name = #x, .pins = x ## _pins, \
479a1d1e0e3SJonathan Neuschäfer .num_pins = ARRAY_SIZE(x ## _pins) }
480a1d1e0e3SJonathan Neuschäfer WPCM450_GRPS
481a1d1e0e3SJonathan Neuschäfer #undef WPCM450_GRP
482a1d1e0e3SJonathan Neuschäfer };
483a1d1e0e3SJonathan Neuschäfer
484a1d1e0e3SJonathan Neuschäfer #define WPCM450_SFUNC(a) WPCM450_FUNC(a, #a)
485a1d1e0e3SJonathan Neuschäfer #define WPCM450_FUNC(a, b...) static const char *a ## _grp[] = { b }
486a1d1e0e3SJonathan Neuschäfer #define WPCM450_MKFUNC(nm) { .name = #nm, .ngroups = ARRAY_SIZE(nm ## _grp), \
487a1d1e0e3SJonathan Neuschäfer .groups = nm ## _grp }
488a1d1e0e3SJonathan Neuschäfer struct wpcm450_func {
489a1d1e0e3SJonathan Neuschäfer const char *name;
490a1d1e0e3SJonathan Neuschäfer const unsigned int ngroups;
491a1d1e0e3SJonathan Neuschäfer const char *const *groups;
492a1d1e0e3SJonathan Neuschäfer };
493a1d1e0e3SJonathan Neuschäfer
494a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb3);
495a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb4);
496a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb5);
497a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(scs1);
498a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(scs2);
499a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(scs3);
500a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb0);
501a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb1);
502a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smb2);
503a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(bsp);
504a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hsp1);
505a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hsp2);
506a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(r1err);
507a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(r1md);
508a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(rmii2);
509a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(r2err);
510a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(r2md);
511a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(kbcc);
512a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(clko);
513a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(smi);
514a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(uinc);
515a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(gspi);
516a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(mben);
517a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(xcs2);
518a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(xcs1);
519a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(sdio);
520a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(sspi);
521a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi0);
522a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi1);
523a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi2);
524a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi3);
525a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi4);
526a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi5);
527a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi6);
528a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi7);
529a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi8);
530a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi9);
531a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi10);
532a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi11);
533a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi12);
534a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi13);
535a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi14);
536a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(fi15);
537a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm0);
538a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm1);
539a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm2);
540a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm3);
541a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm4);
542a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm5);
543a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm6);
544a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(pwm7);
545a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg0);
546a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg1);
547a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg2);
548a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg3);
549a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg4);
550a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg5);
551a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg6);
552a1d1e0e3SJonathan Neuschäfer WPCM450_SFUNC(hg7);
553a1d1e0e3SJonathan Neuschäfer
554a1d1e0e3SJonathan Neuschäfer #define WPCM450_GRP(x) #x
555a1d1e0e3SJonathan Neuschäfer WPCM450_FUNC(gpio, WPCM450_GRPS);
556a1d1e0e3SJonathan Neuschäfer #undef WPCM450_GRP
557a1d1e0e3SJonathan Neuschäfer
558a1d1e0e3SJonathan Neuschäfer /* Function names */
559a1d1e0e3SJonathan Neuschäfer static struct wpcm450_func wpcm450_funcs[] = {
560a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb3),
561a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb4),
562a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb5),
563a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(scs1),
564a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(scs2),
565a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(scs3),
566a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb0),
567a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb1),
568a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smb2),
569a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(bsp),
570a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hsp1),
571a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hsp2),
572a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(r1err),
573a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(r1md),
574a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(rmii2),
575a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(r2err),
576a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(r2md),
577a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(kbcc),
578a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(clko),
579a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(smi),
580a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(uinc),
581a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(gspi),
582a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(mben),
583a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(xcs2),
584a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(xcs1),
585a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(sdio),
586a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(sspi),
587a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi0),
588a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi1),
589a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi2),
590a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi3),
591a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi4),
592a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi5),
593a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi6),
594a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi7),
595a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi8),
596a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi9),
597a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi10),
598a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi11),
599a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi12),
600a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi13),
601a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi14),
602a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(fi15),
603a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm0),
604a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm1),
605a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm2),
606a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm3),
607a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm4),
608a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm5),
609a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm6),
610a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(pwm7),
611a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg0),
612a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg1),
613a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg2),
614a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg3),
615a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg4),
616a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg5),
617a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg6),
618a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(hg7),
619a1d1e0e3SJonathan Neuschäfer WPCM450_MKFUNC(gpio),
620a1d1e0e3SJonathan Neuschäfer };
621a1d1e0e3SJonathan Neuschäfer
622a1d1e0e3SJonathan Neuschäfer #define WPCM450_PINCFG(a, b, c, d, e, f, g) \
6236ef00b42SLinus Walleij [a] = { .fn0 = fn_ ## b, .reg0 = WPCM450_GCR_ ## c, .bit0 = d, \
624a1d1e0e3SJonathan Neuschäfer .fn1 = fn_ ## e, .reg1 = WPCM450_GCR_ ## f, .bit1 = g }
625a1d1e0e3SJonathan Neuschäfer
626a1d1e0e3SJonathan Neuschäfer struct wpcm450_pincfg {
627a1d1e0e3SJonathan Neuschäfer int fn0, reg0, bit0;
628a1d1e0e3SJonathan Neuschäfer int fn1, reg1, bit1;
629a1d1e0e3SJonathan Neuschäfer };
630a1d1e0e3SJonathan Neuschäfer
6316c98ac42SJonathan Neuschäfer /* Add this value to bit0 or bit1 to indicate that the MFSEL bit is inverted */
6326c98ac42SJonathan Neuschäfer #define INV BIT(5)
6336c98ac42SJonathan Neuschäfer
634a1d1e0e3SJonathan Neuschäfer static const struct wpcm450_pincfg pincfg[] = {
635a1d1e0e3SJonathan Neuschäfer /* PIN FUNCTION 1 FUNCTION 2 */
636a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(0, none, NONE, 0, none, NONE, 0),
637a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(1, none, NONE, 0, none, NONE, 0),
638a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(2, none, NONE, 0, none, NONE, 0),
639a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(3, none, NONE, 0, none, NONE, 0),
640a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(4, none, NONE, 0, none, NONE, 0),
641a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(5, none, NONE, 0, none, NONE, 0),
642a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(6, none, NONE, 0, none, NONE, 0),
643a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(7, none, NONE, 0, sdio, MFSEL1, 30),
644a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(8, none, NONE, 0, none, NONE, 0),
645a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(9, none, NONE, 0, none, NONE, 0),
646a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(10, none, NONE, 0, none, NONE, 0),
647a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(11, none, NONE, 0, none, NONE, 0),
648a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(12, gspi, MFSEL1, 24, sspi, MFSEL1, 31),
649a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(13, gspi, MFSEL1, 24, sspi, MFSEL1, 31),
650a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(14, gspi, MFSEL1, 24, sspi, MFSEL1, 31),
651a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(15, gspi, MFSEL1, 24, sspi, MFSEL1, 31),
652a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(16, none, NONE, 0, pwm6, MFSEL2, 22),
653a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(17, none, NONE, 0, pwm7, MFSEL2, 23),
654a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(18, none, NONE, 0, none, NONE, 0),
655a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(19, uinc, MFSEL1, 23, none, NONE, 0),
656a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(20, hg0, MFSEL2, 24, pwm4, MFSEL2, 20),
657a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(21, hg1, MFSEL2, 25, pwm5, MFSEL2, 21),
658a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(22, hg2, MFSEL2, 26, none, NONE, 0),
659a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(23, hg3, MFSEL2, 27, none, NONE, 0),
660a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(24, hg4, MFSEL2, 28, none, NONE, 0),
661a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(25, hg5, MFSEL2, 29, none, NONE, 0),
662a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(26, smb5, MFSEL1, 2, none, NONE, 0),
663a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(27, smb5, MFSEL1, 2, none, NONE, 0),
664a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(28, smb4, MFSEL1, 1, none, NONE, 0),
665a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(29, smb4, MFSEL1, 1, none, NONE, 0),
666a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(30, smb3, MFSEL1, 0, none, NONE, 0),
667a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(31, smb3, MFSEL1, 0, none, NONE, 0),
668a1d1e0e3SJonathan Neuschäfer
669a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(32, scs1, MFSEL1, 3, none, NONE, 0),
670a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(33, scs2, MFSEL1, 4, none, NONE, 0),
6716c98ac42SJonathan Neuschäfer WPCM450_PINCFG(34, scs3, MFSEL1, 5 | INV, none, NONE, 0),
672a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(35, xcs1, MFSEL1, 29, none, NONE, 0),
673a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(36, xcs2, MFSEL1, 28, none, NONE, 0),
674a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(37, none, NONE, 0, none, NONE, 0), /* DVO */
675a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(38, none, NONE, 0, none, NONE, 0), /* DVO */
676a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(39, none, NONE, 0, none, NONE, 0), /* DVO */
677a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(40, none, NONE, 0, none, NONE, 0), /* DVO */
678a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(41, bsp, MFSEL1, 9, none, NONE, 0),
679a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(42, bsp, MFSEL1, 9, none, NONE, 0),
680a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(43, hsp1, MFSEL1, 10, sdio, MFSEL1, 30),
681a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(44, hsp1, MFSEL1, 10, sdio, MFSEL1, 30),
682a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(45, hsp1, MFSEL1, 10, sdio, MFSEL1, 30),
683a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(46, hsp1, MFSEL1, 10, sdio, MFSEL1, 30),
684a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(47, hsp1, MFSEL1, 10, sdio, MFSEL1, 30),
685a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(48, hsp2, MFSEL1, 11, none, NONE, 0),
686a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(49, hsp2, MFSEL1, 11, none, NONE, 0),
687a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(50, hsp2, MFSEL1, 11, none, NONE, 0),
688a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(51, hsp2, MFSEL1, 11, none, NONE, 0),
689a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(52, hsp2, MFSEL1, 11, none, NONE, 0),
690a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(53, hsp2, MFSEL1, 11, none, NONE, 0),
691a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(54, hsp2, MFSEL1, 11, none, NONE, 0),
692a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(55, hsp2, MFSEL1, 11, none, NONE, 0),
693a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(56, r1err, MFSEL1, 12, none, NONE, 0),
694a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(57, r1md, MFSEL1, 13, none, NONE, 0),
695a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(58, r1md, MFSEL1, 13, none, NONE, 0),
696a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(59, hg6, MFSEL2, 30, none, NONE, 0),
697a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(60, hg7, MFSEL2, 31, sdio, MFSEL1, 30),
698a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(61, hsp1, MFSEL1, 10, none, NONE, 0),
699a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(62, hsp1, MFSEL1, 10, none, NONE, 0),
700a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(63, hsp1, MFSEL1, 10, none, NONE, 0),
701a1d1e0e3SJonathan Neuschäfer
702a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(64, fi0, MFSEL2, 0, none, NONE, 0),
703a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(65, fi1, MFSEL2, 1, none, NONE, 0),
704a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(66, fi2, MFSEL2, 2, none, NONE, 0),
705a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(67, fi3, MFSEL2, 3, none, NONE, 0),
706a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(68, fi4, MFSEL2, 4, none, NONE, 0),
707a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(69, fi5, MFSEL2, 5, none, NONE, 0),
708a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(70, fi6, MFSEL2, 6, none, NONE, 0),
709a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(71, fi7, MFSEL2, 7, none, NONE, 0),
710a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(72, fi8, MFSEL2, 8, none, NONE, 0),
711a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(73, fi9, MFSEL2, 9, none, NONE, 0),
712a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(74, fi10, MFSEL2, 10, none, NONE, 0),
713a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(75, fi11, MFSEL2, 11, none, NONE, 0),
714a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(76, fi12, MFSEL2, 12, none, NONE, 0),
715a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(77, fi13, MFSEL2, 13, none, NONE, 0),
716a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(78, fi14, MFSEL2, 14, none, NONE, 0),
717a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(79, fi15, MFSEL2, 15, none, NONE, 0),
718a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(80, pwm0, MFSEL2, 16, none, NONE, 0),
719a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(81, pwm1, MFSEL2, 17, none, NONE, 0),
720a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(82, pwm2, MFSEL2, 18, none, NONE, 0),
721a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(83, pwm3, MFSEL2, 19, none, NONE, 0),
722a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(84, rmii2, MFSEL1, 14, none, NONE, 0),
723a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(85, rmii2, MFSEL1, 14, none, NONE, 0),
724a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(86, rmii2, MFSEL1, 14, none, NONE, 0),
725a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(87, rmii2, MFSEL1, 14, none, NONE, 0),
726a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(88, rmii2, MFSEL1, 14, none, NONE, 0),
727a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(89, rmii2, MFSEL1, 14, none, NONE, 0),
728a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(90, r2err, MFSEL1, 15, none, NONE, 0),
729a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(91, r2md, MFSEL1, 16, none, NONE, 0),
730a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(92, r2md, MFSEL1, 16, none, NONE, 0),
7316c98ac42SJonathan Neuschäfer WPCM450_PINCFG(93, kbcc, MFSEL1, 17 | INV, none, NONE, 0),
7326c98ac42SJonathan Neuschäfer WPCM450_PINCFG(94, kbcc, MFSEL1, 17 | INV, none, NONE, 0),
733a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(95, none, NONE, 0, none, NONE, 0),
734a1d1e0e3SJonathan Neuschäfer
735a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(96, none, NONE, 0, none, NONE, 0),
736a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(97, none, NONE, 0, none, NONE, 0),
737a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(98, none, NONE, 0, none, NONE, 0),
738a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(99, none, NONE, 0, none, NONE, 0),
739a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(100, none, NONE, 0, none, NONE, 0),
740a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(101, none, NONE, 0, none, NONE, 0),
741a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(102, none, NONE, 0, none, NONE, 0),
742a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(103, none, NONE, 0, none, NONE, 0),
743a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(104, none, NONE, 0, none, NONE, 0),
744a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(105, none, NONE, 0, none, NONE, 0),
745a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(106, none, NONE, 0, none, NONE, 0),
746a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(107, none, NONE, 0, none, NONE, 0),
747a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(108, none, NONE, 0, none, NONE, 0), /* DVO */
748a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(109, none, NONE, 0, none, NONE, 0), /* DVO */
749a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(110, none, NONE, 0, none, NONE, 0), /* DVO */
750a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(111, none, NONE, 0, none, NONE, 0), /* DVO */
751a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(112, none, NONE, 0, none, NONE, 0), /* DVO */
752a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(113, none, NONE, 0, none, NONE, 0), /* DVO */
753a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(114, smb0, MFSEL1, 6, none, NONE, 0),
754a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(115, smb0, MFSEL1, 6, none, NONE, 0),
755a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(116, smb1, MFSEL1, 7, none, NONE, 0),
756a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(117, smb1, MFSEL1, 7, none, NONE, 0),
757a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(118, smb2, MFSEL1, 8, none, NONE, 0),
758a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(119, smb2, MFSEL1, 8, none, NONE, 0),
759a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(120, none, NONE, 0, none, NONE, 0), /* DVO */
760a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(121, none, NONE, 0, none, NONE, 0), /* DVO */
761a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(122, none, NONE, 0, none, NONE, 0), /* DVO */
762a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(123, none, NONE, 0, none, NONE, 0), /* DVO */
763a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(124, none, NONE, 0, none, NONE, 0), /* DVO */
764a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(125, none, NONE, 0, none, NONE, 0), /* DVO */
765a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(126, none, NONE, 0, none, NONE, 0), /* DVO */
766a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(127, none, NONE, 0, none, NONE, 0), /* DVO */
767a1d1e0e3SJonathan Neuschäfer };
768a1d1e0e3SJonathan Neuschäfer
769a1d1e0e3SJonathan Neuschäfer #define WPCM450_PIN(n) PINCTRL_PIN(n, "gpio" #n)
770a1d1e0e3SJonathan Neuschäfer
771a1d1e0e3SJonathan Neuschäfer static const struct pinctrl_pin_desc wpcm450_pins[] = {
772a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(0), WPCM450_PIN(1), WPCM450_PIN(2), WPCM450_PIN(3),
773a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(4), WPCM450_PIN(5), WPCM450_PIN(6), WPCM450_PIN(7),
774a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(8), WPCM450_PIN(9), WPCM450_PIN(10), WPCM450_PIN(11),
775a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(12), WPCM450_PIN(13), WPCM450_PIN(14), WPCM450_PIN(15),
776a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(16), WPCM450_PIN(17), WPCM450_PIN(18), WPCM450_PIN(19),
777a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(20), WPCM450_PIN(21), WPCM450_PIN(22), WPCM450_PIN(23),
778a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(24), WPCM450_PIN(25), WPCM450_PIN(26), WPCM450_PIN(27),
779a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(28), WPCM450_PIN(29), WPCM450_PIN(30), WPCM450_PIN(31),
780a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(32), WPCM450_PIN(33), WPCM450_PIN(34), WPCM450_PIN(35),
781a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(36), WPCM450_PIN(37), WPCM450_PIN(38), WPCM450_PIN(39),
782a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(40), WPCM450_PIN(41), WPCM450_PIN(42), WPCM450_PIN(43),
783a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(44), WPCM450_PIN(45), WPCM450_PIN(46), WPCM450_PIN(47),
784a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(48), WPCM450_PIN(49), WPCM450_PIN(50), WPCM450_PIN(51),
785a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(52), WPCM450_PIN(53), WPCM450_PIN(54), WPCM450_PIN(55),
786a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(56), WPCM450_PIN(57), WPCM450_PIN(58), WPCM450_PIN(59),
787a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(60), WPCM450_PIN(61), WPCM450_PIN(62), WPCM450_PIN(63),
788a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(64), WPCM450_PIN(65), WPCM450_PIN(66), WPCM450_PIN(67),
789a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(68), WPCM450_PIN(69), WPCM450_PIN(70), WPCM450_PIN(71),
790a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(72), WPCM450_PIN(73), WPCM450_PIN(74), WPCM450_PIN(75),
791a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(76), WPCM450_PIN(77), WPCM450_PIN(78), WPCM450_PIN(79),
792a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(80), WPCM450_PIN(81), WPCM450_PIN(82), WPCM450_PIN(83),
793a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(84), WPCM450_PIN(85), WPCM450_PIN(86), WPCM450_PIN(87),
794a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(88), WPCM450_PIN(89), WPCM450_PIN(90), WPCM450_PIN(91),
795a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(92), WPCM450_PIN(93), WPCM450_PIN(94), WPCM450_PIN(95),
796a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(96), WPCM450_PIN(97), WPCM450_PIN(98), WPCM450_PIN(99),
797a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(100), WPCM450_PIN(101), WPCM450_PIN(102), WPCM450_PIN(103),
798a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(104), WPCM450_PIN(105), WPCM450_PIN(106), WPCM450_PIN(107),
799a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(108), WPCM450_PIN(109), WPCM450_PIN(110), WPCM450_PIN(111),
800a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(112), WPCM450_PIN(113), WPCM450_PIN(114), WPCM450_PIN(115),
801a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(116), WPCM450_PIN(117), WPCM450_PIN(118), WPCM450_PIN(119),
802a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(120), WPCM450_PIN(121), WPCM450_PIN(122), WPCM450_PIN(123),
803a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(124), WPCM450_PIN(125), WPCM450_PIN(126), WPCM450_PIN(127),
804a1d1e0e3SJonathan Neuschäfer };
805a1d1e0e3SJonathan Neuschäfer
8064f1d423cSJonathan Neuschäfer /* Helper function to update MFSEL field according to the selected function */
wpcm450_update_mfsel(struct regmap * gcr_regmap,int reg,int bit,int fn,int fn_selected)8074f1d423cSJonathan Neuschäfer static void wpcm450_update_mfsel(struct regmap *gcr_regmap, int reg, int bit, int fn, int fn_selected)
8084f1d423cSJonathan Neuschäfer {
8094f1d423cSJonathan Neuschäfer bool value = (fn == fn_selected);
8104f1d423cSJonathan Neuschäfer
8116c98ac42SJonathan Neuschäfer if (bit & INV) {
8126c98ac42SJonathan Neuschäfer value = !value;
8136c98ac42SJonathan Neuschäfer bit &= ~INV;
8146c98ac42SJonathan Neuschäfer }
8156c98ac42SJonathan Neuschäfer
8164f1d423cSJonathan Neuschäfer regmap_update_bits(gcr_regmap, reg, BIT(bit), value ? BIT(bit) : 0);
8174f1d423cSJonathan Neuschäfer }
8184f1d423cSJonathan Neuschäfer
819a1d1e0e3SJonathan Neuschäfer /* Enable mode in pin group */
wpcm450_setfunc(struct regmap * gcr_regmap,const unsigned int * pin,int npins,int func)820a1d1e0e3SJonathan Neuschäfer static void wpcm450_setfunc(struct regmap *gcr_regmap, const unsigned int *pin,
821a1d1e0e3SJonathan Neuschäfer int npins, int func)
822a1d1e0e3SJonathan Neuschäfer {
823a1d1e0e3SJonathan Neuschäfer const struct wpcm450_pincfg *cfg;
824a1d1e0e3SJonathan Neuschäfer int i;
825a1d1e0e3SJonathan Neuschäfer
826a1d1e0e3SJonathan Neuschäfer for (i = 0; i < npins; i++) {
827a1d1e0e3SJonathan Neuschäfer cfg = &pincfg[pin[i]];
828a1d1e0e3SJonathan Neuschäfer if (func == fn_gpio || cfg->fn0 == func || cfg->fn1 == func) {
829a1d1e0e3SJonathan Neuschäfer if (cfg->reg0)
8304f1d423cSJonathan Neuschäfer wpcm450_update_mfsel(gcr_regmap, cfg->reg0,
8314f1d423cSJonathan Neuschäfer cfg->bit0, cfg->fn0, func);
832a1d1e0e3SJonathan Neuschäfer if (cfg->reg1)
8334f1d423cSJonathan Neuschäfer wpcm450_update_mfsel(gcr_regmap, cfg->reg1,
8344f1d423cSJonathan Neuschäfer cfg->bit1, cfg->fn1, func);
835a1d1e0e3SJonathan Neuschäfer }
836a1d1e0e3SJonathan Neuschäfer }
837a1d1e0e3SJonathan Neuschäfer }
838a1d1e0e3SJonathan Neuschäfer
wpcm450_get_groups_count(struct pinctrl_dev * pctldev)839a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_groups_count(struct pinctrl_dev *pctldev)
840a1d1e0e3SJonathan Neuschäfer {
841a1d1e0e3SJonathan Neuschäfer return ARRAY_SIZE(wpcm450_groups);
842a1d1e0e3SJonathan Neuschäfer }
843a1d1e0e3SJonathan Neuschäfer
wpcm450_get_group_name(struct pinctrl_dev * pctldev,unsigned int selector)844a1d1e0e3SJonathan Neuschäfer static const char *wpcm450_get_group_name(struct pinctrl_dev *pctldev,
845a1d1e0e3SJonathan Neuschäfer unsigned int selector)
846a1d1e0e3SJonathan Neuschäfer {
847a1d1e0e3SJonathan Neuschäfer return wpcm450_groups[selector].name;
848a1d1e0e3SJonathan Neuschäfer }
849a1d1e0e3SJonathan Neuschäfer
wpcm450_get_group_pins(struct pinctrl_dev * pctldev,unsigned int selector,const unsigned int ** pins,unsigned int * npins)850a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_group_pins(struct pinctrl_dev *pctldev,
851a1d1e0e3SJonathan Neuschäfer unsigned int selector,
852a1d1e0e3SJonathan Neuschäfer const unsigned int **pins,
853a1d1e0e3SJonathan Neuschäfer unsigned int *npins)
854a1d1e0e3SJonathan Neuschäfer {
855a1d1e0e3SJonathan Neuschäfer *npins = wpcm450_groups[selector].num_pins;
856a1d1e0e3SJonathan Neuschäfer *pins = wpcm450_groups[selector].pins;
857a1d1e0e3SJonathan Neuschäfer
858a1d1e0e3SJonathan Neuschäfer return 0;
859a1d1e0e3SJonathan Neuschäfer }
860a1d1e0e3SJonathan Neuschäfer
wpcm450_dt_node_to_map(struct pinctrl_dev * pctldev,struct device_node * np_config,struct pinctrl_map ** map,u32 * num_maps)861a1d1e0e3SJonathan Neuschäfer static int wpcm450_dt_node_to_map(struct pinctrl_dev *pctldev,
862a1d1e0e3SJonathan Neuschäfer struct device_node *np_config,
863a1d1e0e3SJonathan Neuschäfer struct pinctrl_map **map,
864a1d1e0e3SJonathan Neuschäfer u32 *num_maps)
865a1d1e0e3SJonathan Neuschäfer {
866a1d1e0e3SJonathan Neuschäfer return pinconf_generic_dt_node_to_map(pctldev, np_config,
867a1d1e0e3SJonathan Neuschäfer map, num_maps,
868a1d1e0e3SJonathan Neuschäfer PIN_MAP_TYPE_INVALID);
869a1d1e0e3SJonathan Neuschäfer }
870a1d1e0e3SJonathan Neuschäfer
wpcm450_dt_free_map(struct pinctrl_dev * pctldev,struct pinctrl_map * map,u32 num_maps)871a1d1e0e3SJonathan Neuschäfer static void wpcm450_dt_free_map(struct pinctrl_dev *pctldev,
872a1d1e0e3SJonathan Neuschäfer struct pinctrl_map *map, u32 num_maps)
873a1d1e0e3SJonathan Neuschäfer {
874a1d1e0e3SJonathan Neuschäfer kfree(map);
875a1d1e0e3SJonathan Neuschäfer }
876a1d1e0e3SJonathan Neuschäfer
877a1d1e0e3SJonathan Neuschäfer static const struct pinctrl_ops wpcm450_pinctrl_ops = {
878a1d1e0e3SJonathan Neuschäfer .get_groups_count = wpcm450_get_groups_count,
879a1d1e0e3SJonathan Neuschäfer .get_group_name = wpcm450_get_group_name,
880a1d1e0e3SJonathan Neuschäfer .get_group_pins = wpcm450_get_group_pins,
881a1d1e0e3SJonathan Neuschäfer .dt_node_to_map = wpcm450_dt_node_to_map,
882a1d1e0e3SJonathan Neuschäfer .dt_free_map = wpcm450_dt_free_map,
883a1d1e0e3SJonathan Neuschäfer };
884a1d1e0e3SJonathan Neuschäfer
wpcm450_get_functions_count(struct pinctrl_dev * pctldev)885a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_functions_count(struct pinctrl_dev *pctldev)
886a1d1e0e3SJonathan Neuschäfer {
887a1d1e0e3SJonathan Neuschäfer return ARRAY_SIZE(wpcm450_funcs);
888a1d1e0e3SJonathan Neuschäfer }
889a1d1e0e3SJonathan Neuschäfer
wpcm450_get_function_name(struct pinctrl_dev * pctldev,unsigned int function)890a1d1e0e3SJonathan Neuschäfer static const char *wpcm450_get_function_name(struct pinctrl_dev *pctldev,
891a1d1e0e3SJonathan Neuschäfer unsigned int function)
892a1d1e0e3SJonathan Neuschäfer {
893a1d1e0e3SJonathan Neuschäfer return wpcm450_funcs[function].name;
894a1d1e0e3SJonathan Neuschäfer }
895a1d1e0e3SJonathan Neuschäfer
wpcm450_get_function_groups(struct pinctrl_dev * pctldev,unsigned int function,const char * const ** groups,unsigned int * const ngroups)896a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_function_groups(struct pinctrl_dev *pctldev,
897a1d1e0e3SJonathan Neuschäfer unsigned int function,
898a1d1e0e3SJonathan Neuschäfer const char * const **groups,
899a1d1e0e3SJonathan Neuschäfer unsigned int * const ngroups)
900a1d1e0e3SJonathan Neuschäfer {
901a1d1e0e3SJonathan Neuschäfer *ngroups = wpcm450_funcs[function].ngroups;
902a1d1e0e3SJonathan Neuschäfer *groups = wpcm450_funcs[function].groups;
903a1d1e0e3SJonathan Neuschäfer
904a1d1e0e3SJonathan Neuschäfer return 0;
905a1d1e0e3SJonathan Neuschäfer }
906a1d1e0e3SJonathan Neuschäfer
wpcm450_pinmux_set_mux(struct pinctrl_dev * pctldev,unsigned int function,unsigned int group)907a1d1e0e3SJonathan Neuschäfer static int wpcm450_pinmux_set_mux(struct pinctrl_dev *pctldev,
908a1d1e0e3SJonathan Neuschäfer unsigned int function,
909a1d1e0e3SJonathan Neuschäfer unsigned int group)
910a1d1e0e3SJonathan Neuschäfer {
911a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
912a1d1e0e3SJonathan Neuschäfer
913a1d1e0e3SJonathan Neuschäfer wpcm450_setfunc(pctrl->gcr_regmap, wpcm450_groups[group].pins,
914a1d1e0e3SJonathan Neuschäfer wpcm450_groups[group].num_pins, function);
915a1d1e0e3SJonathan Neuschäfer
916a1d1e0e3SJonathan Neuschäfer return 0;
917a1d1e0e3SJonathan Neuschäfer }
918a1d1e0e3SJonathan Neuschäfer
919a1d1e0e3SJonathan Neuschäfer static const struct pinmux_ops wpcm450_pinmux_ops = {
920a1d1e0e3SJonathan Neuschäfer .get_functions_count = wpcm450_get_functions_count,
921a1d1e0e3SJonathan Neuschäfer .get_function_name = wpcm450_get_function_name,
922a1d1e0e3SJonathan Neuschäfer .get_function_groups = wpcm450_get_function_groups,
923a1d1e0e3SJonathan Neuschäfer .set_mux = wpcm450_pinmux_set_mux,
924a1d1e0e3SJonathan Neuschäfer };
925a1d1e0e3SJonathan Neuschäfer
debounce_bitnum(int gpio)926a1d1e0e3SJonathan Neuschäfer static int debounce_bitnum(int gpio)
927a1d1e0e3SJonathan Neuschäfer {
928a1d1e0e3SJonathan Neuschäfer if (gpio >= 0 && gpio < 16)
929a1d1e0e3SJonathan Neuschäfer return gpio;
930a1d1e0e3SJonathan Neuschäfer return -EINVAL;
931a1d1e0e3SJonathan Neuschäfer }
932a1d1e0e3SJonathan Neuschäfer
wpcm450_config_get(struct pinctrl_dev * pctldev,unsigned int pin,unsigned long * config)933a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_get(struct pinctrl_dev *pctldev, unsigned int pin,
934a1d1e0e3SJonathan Neuschäfer unsigned long *config)
935a1d1e0e3SJonathan Neuschäfer {
936a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
937a1d1e0e3SJonathan Neuschäfer enum pin_config_param param = pinconf_to_config_param(*config);
938a1d1e0e3SJonathan Neuschäfer unsigned long flags;
939a1d1e0e3SJonathan Neuschäfer int bit;
940a1d1e0e3SJonathan Neuschäfer u32 reg;
941a1d1e0e3SJonathan Neuschäfer
942a1d1e0e3SJonathan Neuschäfer switch (param) {
943a1d1e0e3SJonathan Neuschäfer case PIN_CONFIG_INPUT_DEBOUNCE:
944a1d1e0e3SJonathan Neuschäfer bit = debounce_bitnum(pin);
945a1d1e0e3SJonathan Neuschäfer if (bit < 0)
946a1d1e0e3SJonathan Neuschäfer return bit;
947a1d1e0e3SJonathan Neuschäfer
948a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
949a1d1e0e3SJonathan Neuschäfer reg = ioread32(pctrl->gpio_base + WPCM450_GPEVDBNC);
950a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
951a1d1e0e3SJonathan Neuschäfer
952a1d1e0e3SJonathan Neuschäfer *config = pinconf_to_config_packed(param, !!(reg & BIT(bit)));
953a1d1e0e3SJonathan Neuschäfer return 0;
954a1d1e0e3SJonathan Neuschäfer default:
955a1d1e0e3SJonathan Neuschäfer return -ENOTSUPP;
956a1d1e0e3SJonathan Neuschäfer }
957a1d1e0e3SJonathan Neuschäfer }
958a1d1e0e3SJonathan Neuschäfer
wpcm450_config_set_one(struct wpcm450_pinctrl * pctrl,unsigned int pin,unsigned long config)959a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_set_one(struct wpcm450_pinctrl *pctrl,
960a1d1e0e3SJonathan Neuschäfer unsigned int pin, unsigned long config)
961a1d1e0e3SJonathan Neuschäfer {
962a1d1e0e3SJonathan Neuschäfer enum pin_config_param param = pinconf_to_config_param(config);
963a1d1e0e3SJonathan Neuschäfer unsigned long flags;
964a1d1e0e3SJonathan Neuschäfer unsigned long reg;
965a1d1e0e3SJonathan Neuschäfer int bit;
966a1d1e0e3SJonathan Neuschäfer int arg;
967a1d1e0e3SJonathan Neuschäfer
968a1d1e0e3SJonathan Neuschäfer switch (param) {
969a1d1e0e3SJonathan Neuschäfer case PIN_CONFIG_INPUT_DEBOUNCE:
970a1d1e0e3SJonathan Neuschäfer bit = debounce_bitnum(pin);
971a1d1e0e3SJonathan Neuschäfer if (bit < 0)
972a1d1e0e3SJonathan Neuschäfer return bit;
973a1d1e0e3SJonathan Neuschäfer
974a1d1e0e3SJonathan Neuschäfer arg = pinconf_to_config_argument(config);
975a1d1e0e3SJonathan Neuschäfer
976a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags);
977a1d1e0e3SJonathan Neuschäfer reg = ioread32(pctrl->gpio_base + WPCM450_GPEVDBNC);
978a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, ®, arg);
979a1d1e0e3SJonathan Neuschäfer iowrite32(reg, pctrl->gpio_base + WPCM450_GPEVDBNC);
980a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags);
981a1d1e0e3SJonathan Neuschäfer return 0;
982a1d1e0e3SJonathan Neuschäfer default:
983a1d1e0e3SJonathan Neuschäfer return -ENOTSUPP;
984a1d1e0e3SJonathan Neuschäfer }
985a1d1e0e3SJonathan Neuschäfer }
986a1d1e0e3SJonathan Neuschäfer
wpcm450_config_set(struct pinctrl_dev * pctldev,unsigned int pin,unsigned long * configs,unsigned int num_configs)987a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_set(struct pinctrl_dev *pctldev, unsigned int pin,
988a1d1e0e3SJonathan Neuschäfer unsigned long *configs, unsigned int num_configs)
989a1d1e0e3SJonathan Neuschäfer {
990a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev);
991a1d1e0e3SJonathan Neuschäfer int ret;
992a1d1e0e3SJonathan Neuschäfer
993a1d1e0e3SJonathan Neuschäfer while (num_configs--) {
994a1d1e0e3SJonathan Neuschäfer ret = wpcm450_config_set_one(pctrl, pin, *configs++);
995a1d1e0e3SJonathan Neuschäfer if (ret)
996a1d1e0e3SJonathan Neuschäfer return ret;
997a1d1e0e3SJonathan Neuschäfer }
998a1d1e0e3SJonathan Neuschäfer
999a1d1e0e3SJonathan Neuschäfer return 0;
1000a1d1e0e3SJonathan Neuschäfer }
1001a1d1e0e3SJonathan Neuschäfer
1002a1d1e0e3SJonathan Neuschäfer static const struct pinconf_ops wpcm450_pinconf_ops = {
1003a1d1e0e3SJonathan Neuschäfer .is_generic = true,
1004a1d1e0e3SJonathan Neuschäfer .pin_config_get = wpcm450_config_get,
1005a1d1e0e3SJonathan Neuschäfer .pin_config_set = wpcm450_config_set,
1006a1d1e0e3SJonathan Neuschäfer };
1007a1d1e0e3SJonathan Neuschäfer
1008a1d1e0e3SJonathan Neuschäfer static struct pinctrl_desc wpcm450_pinctrl_desc = {
1009a1d1e0e3SJonathan Neuschäfer .name = "wpcm450-pinctrl",
1010a1d1e0e3SJonathan Neuschäfer .pins = wpcm450_pins,
1011a1d1e0e3SJonathan Neuschäfer .npins = ARRAY_SIZE(wpcm450_pins),
1012a1d1e0e3SJonathan Neuschäfer .pctlops = &wpcm450_pinctrl_ops,
1013a1d1e0e3SJonathan Neuschäfer .pmxops = &wpcm450_pinmux_ops,
1014a1d1e0e3SJonathan Neuschäfer .confops = &wpcm450_pinconf_ops,
1015a1d1e0e3SJonathan Neuschäfer .owner = THIS_MODULE,
1016a1d1e0e3SJonathan Neuschäfer };
1017a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_set_config(struct gpio_chip * chip,unsigned int offset,unsigned long config)1018a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_set_config(struct gpio_chip *chip,
1019a1d1e0e3SJonathan Neuschäfer unsigned int offset, unsigned long config)
1020a1d1e0e3SJonathan Neuschäfer {
1021a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(chip);
1022a1d1e0e3SJonathan Neuschäfer
1023a1d1e0e3SJonathan Neuschäfer return wpcm450_config_set_one(gpio->pctrl, offset, config);
1024a1d1e0e3SJonathan Neuschäfer }
1025a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_add_pin_ranges(struct gpio_chip * chip)1026a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_add_pin_ranges(struct gpio_chip *chip)
1027a1d1e0e3SJonathan Neuschäfer {
1028a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(chip);
1029a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank = gpio->bank;
1030a1d1e0e3SJonathan Neuschäfer
1031a1d1e0e3SJonathan Neuschäfer return gpiochip_add_pin_range(&gpio->gc, dev_name(gpio->pctrl->dev),
1032a1d1e0e3SJonathan Neuschäfer 0, bank->base, bank->length);
1033a1d1e0e3SJonathan Neuschäfer }
1034a1d1e0e3SJonathan Neuschäfer
wpcm450_gpio_register(struct platform_device * pdev,struct wpcm450_pinctrl * pctrl)1035a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_register(struct platform_device *pdev,
1036a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl)
1037a1d1e0e3SJonathan Neuschäfer {
1038a1d1e0e3SJonathan Neuschäfer struct device *dev = &pdev->dev;
1039a1d1e0e3SJonathan Neuschäfer struct fwnode_handle *child;
1040a1d1e0e3SJonathan Neuschäfer int ret;
1041a1d1e0e3SJonathan Neuschäfer
1042a1d1e0e3SJonathan Neuschäfer pctrl->gpio_base = devm_platform_ioremap_resource(pdev, 0);
1043283fb4eaSJialin Zhang if (IS_ERR(pctrl->gpio_base))
1044283fb4eaSJialin Zhang return dev_err_probe(dev, PTR_ERR(pctrl->gpio_base),
1045283fb4eaSJialin Zhang "Resource fail for GPIO controller\n");
1046a1d1e0e3SJonathan Neuschäfer
1047a1d1e0e3SJonathan Neuschäfer device_for_each_child_node(dev, child) {
1048a1d1e0e3SJonathan Neuschäfer void __iomem *dat = NULL;
1049a1d1e0e3SJonathan Neuschäfer void __iomem *set = NULL;
1050a1d1e0e3SJonathan Neuschäfer void __iomem *dirout = NULL;
1051a1d1e0e3SJonathan Neuschäfer unsigned long flags = 0;
1052a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank;
1053a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio;
1054a1d1e0e3SJonathan Neuschäfer struct gpio_irq_chip *girq;
1055a1d1e0e3SJonathan Neuschäfer u32 reg;
1056a1d1e0e3SJonathan Neuschäfer int i;
1057a1d1e0e3SJonathan Neuschäfer
1058a1d1e0e3SJonathan Neuschäfer if (!fwnode_property_read_bool(child, "gpio-controller"))
1059a1d1e0e3SJonathan Neuschäfer continue;
1060a1d1e0e3SJonathan Neuschäfer
1061a1d1e0e3SJonathan Neuschäfer ret = fwnode_property_read_u32(child, "reg", ®);
1062a1d1e0e3SJonathan Neuschäfer if (ret < 0)
1063a1d1e0e3SJonathan Neuschäfer return ret;
1064a1d1e0e3SJonathan Neuschäfer
1065ce2076baSDan Carpenter if (reg >= WPCM450_NUM_BANKS)
1066a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, -EINVAL,
1067a1d1e0e3SJonathan Neuschäfer "GPIO index %d out of range!\n", reg);
1068a1d1e0e3SJonathan Neuschäfer
1069*87d315a3SMikhail Kobuk gpio = &pctrl->gpio_bank[reg];
1070*87d315a3SMikhail Kobuk gpio->pctrl = pctrl;
1071*87d315a3SMikhail Kobuk
1072a1d1e0e3SJonathan Neuschäfer bank = &wpcm450_banks[reg];
1073a1d1e0e3SJonathan Neuschäfer gpio->bank = bank;
1074a1d1e0e3SJonathan Neuschäfer
1075a1d1e0e3SJonathan Neuschäfer dat = pctrl->gpio_base + bank->datain;
1076a1d1e0e3SJonathan Neuschäfer if (bank->dataout) {
1077a1d1e0e3SJonathan Neuschäfer set = pctrl->gpio_base + bank->dataout;
1078a1d1e0e3SJonathan Neuschäfer dirout = pctrl->gpio_base + bank->cfg0;
1079a1d1e0e3SJonathan Neuschäfer } else {
1080a1d1e0e3SJonathan Neuschäfer flags = BGPIOF_NO_OUTPUT;
1081a1d1e0e3SJonathan Neuschäfer }
1082a1d1e0e3SJonathan Neuschäfer ret = bgpio_init(&gpio->gc, dev, 4,
1083a1d1e0e3SJonathan Neuschäfer dat, set, NULL, dirout, NULL, flags);
1084a1d1e0e3SJonathan Neuschäfer if (ret < 0)
1085a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, ret, "GPIO initialization failed\n");
1086a1d1e0e3SJonathan Neuschäfer
1087a1d1e0e3SJonathan Neuschäfer gpio->gc.ngpio = bank->length;
1088a1d1e0e3SJonathan Neuschäfer gpio->gc.set_config = wpcm450_gpio_set_config;
1089a1d1e0e3SJonathan Neuschäfer gpio->gc.fwnode = child;
1090a1d1e0e3SJonathan Neuschäfer gpio->gc.add_pin_ranges = wpcm450_gpio_add_pin_ranges;
1091a1d1e0e3SJonathan Neuschäfer
1092a1d1e0e3SJonathan Neuschäfer girq = &gpio->gc.irq;
109397775ebbSJonathan Neuschäfer gpio_irq_chip_set_chip(girq, &wpcm450_gpio_irqchip);
1094a1d1e0e3SJonathan Neuschäfer girq->parent_handler = wpcm450_gpio_irqhandler;
1095a1d1e0e3SJonathan Neuschäfer girq->parents = devm_kcalloc(dev, WPCM450_NUM_GPIO_IRQS,
1096a1d1e0e3SJonathan Neuschäfer sizeof(*girq->parents), GFP_KERNEL);
1097a1d1e0e3SJonathan Neuschäfer if (!girq->parents)
1098a1d1e0e3SJonathan Neuschäfer return -ENOMEM;
1099a1d1e0e3SJonathan Neuschäfer girq->default_type = IRQ_TYPE_NONE;
1100a1d1e0e3SJonathan Neuschäfer girq->handler = handle_bad_irq;
1101a1d1e0e3SJonathan Neuschäfer
1102a1d1e0e3SJonathan Neuschäfer girq->num_parents = 0;
1103a1d1e0e3SJonathan Neuschäfer for (i = 0; i < WPCM450_NUM_GPIO_IRQS; i++) {
1104f4a31facSAndy Shevchenko int irq;
1105a1d1e0e3SJonathan Neuschäfer
1106f4a31facSAndy Shevchenko irq = fwnode_irq_get(child, i);
1107a1d1e0e3SJonathan Neuschäfer if (irq < 0)
1108a1d1e0e3SJonathan Neuschäfer break;
1109f4a31facSAndy Shevchenko if (!irq)
1110f4a31facSAndy Shevchenko continue;
1111a1d1e0e3SJonathan Neuschäfer
1112a1d1e0e3SJonathan Neuschäfer girq->parents[i] = irq;
1113a1d1e0e3SJonathan Neuschäfer girq->num_parents++;
1114a1d1e0e3SJonathan Neuschäfer }
1115a1d1e0e3SJonathan Neuschäfer
1116a1d1e0e3SJonathan Neuschäfer ret = devm_gpiochip_add_data(dev, &gpio->gc, gpio);
1117a1d1e0e3SJonathan Neuschäfer if (ret)
1118a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, ret, "Failed to add GPIO chip\n");
1119a1d1e0e3SJonathan Neuschäfer }
1120a1d1e0e3SJonathan Neuschäfer
1121a1d1e0e3SJonathan Neuschäfer return 0;
1122a1d1e0e3SJonathan Neuschäfer }
1123a1d1e0e3SJonathan Neuschäfer
wpcm450_pinctrl_probe(struct platform_device * pdev)1124a1d1e0e3SJonathan Neuschäfer static int wpcm450_pinctrl_probe(struct platform_device *pdev)
1125a1d1e0e3SJonathan Neuschäfer {
1126a1d1e0e3SJonathan Neuschäfer struct device *dev = &pdev->dev;
1127a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl;
1128a1d1e0e3SJonathan Neuschäfer int ret;
1129a1d1e0e3SJonathan Neuschäfer
1130a1d1e0e3SJonathan Neuschäfer pctrl = devm_kzalloc(dev, sizeof(*pctrl), GFP_KERNEL);
1131a1d1e0e3SJonathan Neuschäfer if (!pctrl)
1132a1d1e0e3SJonathan Neuschäfer return -ENOMEM;
1133a1d1e0e3SJonathan Neuschäfer
1134a1d1e0e3SJonathan Neuschäfer pctrl->dev = &pdev->dev;
1135a1d1e0e3SJonathan Neuschäfer raw_spin_lock_init(&pctrl->lock);
1136a1d1e0e3SJonathan Neuschäfer dev_set_drvdata(dev, pctrl);
1137a1d1e0e3SJonathan Neuschäfer
1138a1d1e0e3SJonathan Neuschäfer pctrl->gcr_regmap =
1139a1d1e0e3SJonathan Neuschäfer syscon_regmap_lookup_by_compatible("nuvoton,wpcm450-gcr");
1140a1d1e0e3SJonathan Neuschäfer if (IS_ERR(pctrl->gcr_regmap))
1141a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, PTR_ERR(pctrl->gcr_regmap),
1142a1d1e0e3SJonathan Neuschäfer "Failed to find nuvoton,wpcm450-gcr\n");
1143a1d1e0e3SJonathan Neuschäfer
1144a1d1e0e3SJonathan Neuschäfer pctrl->pctldev = devm_pinctrl_register(dev,
1145a1d1e0e3SJonathan Neuschäfer &wpcm450_pinctrl_desc, pctrl);
1146a1d1e0e3SJonathan Neuschäfer if (IS_ERR(pctrl->pctldev))
1147a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, PTR_ERR(pctrl->pctldev),
1148a1d1e0e3SJonathan Neuschäfer "Failed to register pinctrl device\n");
1149a1d1e0e3SJonathan Neuschäfer
1150a1d1e0e3SJonathan Neuschäfer ret = wpcm450_gpio_register(pdev, pctrl);
1151a1d1e0e3SJonathan Neuschäfer if (ret < 0)
1152a1d1e0e3SJonathan Neuschäfer return ret;
1153a1d1e0e3SJonathan Neuschäfer
1154a1d1e0e3SJonathan Neuschäfer return 0;
1155a1d1e0e3SJonathan Neuschäfer }
1156a1d1e0e3SJonathan Neuschäfer
1157a1d1e0e3SJonathan Neuschäfer static const struct of_device_id wpcm450_pinctrl_match[] = {
1158a1d1e0e3SJonathan Neuschäfer { .compatible = "nuvoton,wpcm450-pinctrl" },
1159a1d1e0e3SJonathan Neuschäfer { }
1160a1d1e0e3SJonathan Neuschäfer };
1161a1d1e0e3SJonathan Neuschäfer MODULE_DEVICE_TABLE(of, wpcm450_pinctrl_match);
1162a1d1e0e3SJonathan Neuschäfer
1163a1d1e0e3SJonathan Neuschäfer static struct platform_driver wpcm450_pinctrl_driver = {
1164a1d1e0e3SJonathan Neuschäfer .probe = wpcm450_pinctrl_probe,
1165a1d1e0e3SJonathan Neuschäfer .driver = {
1166a1d1e0e3SJonathan Neuschäfer .name = "wpcm450-pinctrl",
1167a1d1e0e3SJonathan Neuschäfer .of_match_table = wpcm450_pinctrl_match,
1168a1d1e0e3SJonathan Neuschäfer },
1169a1d1e0e3SJonathan Neuschäfer };
1170a1d1e0e3SJonathan Neuschäfer module_platform_driver(wpcm450_pinctrl_driver);
1171a1d1e0e3SJonathan Neuschäfer
1172a1d1e0e3SJonathan Neuschäfer MODULE_LICENSE("GPL v2");
1173a1d1e0e3SJonathan Neuschäfer MODULE_AUTHOR("Jonathan Neuschäfer <j.neuschaefer@gmx.net>");
1174a1d1e0e3SJonathan Neuschäfer MODULE_DESCRIPTION("Nuvoton WPCM450 Pinctrl and GPIO driver");
1175