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 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 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 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 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 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 */ 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 */ 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 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 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 631a1d1e0e3SJonathan Neuschäfer static const struct wpcm450_pincfg pincfg[] = { 632a1d1e0e3SJonathan Neuschäfer /* PIN FUNCTION 1 FUNCTION 2 */ 633a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(0, none, NONE, 0, none, NONE, 0), 634a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(1, none, NONE, 0, none, NONE, 0), 635a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(2, none, NONE, 0, none, NONE, 0), 636a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(3, none, NONE, 0, none, NONE, 0), 637a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(4, none, NONE, 0, none, NONE, 0), 638a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(5, none, NONE, 0, none, NONE, 0), 639a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(6, none, NONE, 0, none, NONE, 0), 640a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(7, none, NONE, 0, sdio, MFSEL1, 30), 641a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(8, none, NONE, 0, none, NONE, 0), 642a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(9, none, NONE, 0, none, NONE, 0), 643a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(10, none, NONE, 0, none, NONE, 0), 644a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(11, none, NONE, 0, none, NONE, 0), 645a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(12, gspi, MFSEL1, 24, sspi, MFSEL1, 31), 646a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(13, gspi, MFSEL1, 24, sspi, MFSEL1, 31), 647a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(14, gspi, MFSEL1, 24, sspi, MFSEL1, 31), 648a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(15, gspi, MFSEL1, 24, sspi, MFSEL1, 31), 649a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(16, none, NONE, 0, pwm6, MFSEL2, 22), 650a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(17, none, NONE, 0, pwm7, MFSEL2, 23), 651a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(18, none, NONE, 0, none, NONE, 0), 652a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(19, uinc, MFSEL1, 23, none, NONE, 0), 653a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(20, hg0, MFSEL2, 24, pwm4, MFSEL2, 20), 654a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(21, hg1, MFSEL2, 25, pwm5, MFSEL2, 21), 655a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(22, hg2, MFSEL2, 26, none, NONE, 0), 656a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(23, hg3, MFSEL2, 27, none, NONE, 0), 657a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(24, hg4, MFSEL2, 28, none, NONE, 0), 658a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(25, hg5, MFSEL2, 29, none, NONE, 0), 659a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(26, smb5, MFSEL1, 2, none, NONE, 0), 660a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(27, smb5, MFSEL1, 2, none, NONE, 0), 661a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(28, smb4, MFSEL1, 1, none, NONE, 0), 662a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(29, smb4, MFSEL1, 1, none, NONE, 0), 663a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(30, smb3, MFSEL1, 0, none, NONE, 0), 664a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(31, smb3, MFSEL1, 0, none, NONE, 0), 665a1d1e0e3SJonathan Neuschäfer 666a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(32, scs1, MFSEL1, 3, none, NONE, 0), 667a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(33, scs2, MFSEL1, 4, none, NONE, 0), 668a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(34, scs3, MFSEL1, 5, none, NONE, 0), 669a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(35, xcs1, MFSEL1, 29, none, NONE, 0), 670a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(36, xcs2, MFSEL1, 28, none, NONE, 0), 671a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(37, none, NONE, 0, none, NONE, 0), /* DVO */ 672a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(38, none, NONE, 0, none, NONE, 0), /* DVO */ 673a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(39, none, NONE, 0, none, NONE, 0), /* DVO */ 674a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(40, none, NONE, 0, none, NONE, 0), /* DVO */ 675a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(41, bsp, MFSEL1, 9, none, NONE, 0), 676a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(42, bsp, MFSEL1, 9, none, NONE, 0), 677a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(43, hsp1, MFSEL1, 10, sdio, MFSEL1, 30), 678a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(44, hsp1, MFSEL1, 10, sdio, MFSEL1, 30), 679a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(45, hsp1, MFSEL1, 10, sdio, MFSEL1, 30), 680a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(46, hsp1, MFSEL1, 10, sdio, MFSEL1, 30), 681a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(47, hsp1, MFSEL1, 10, sdio, MFSEL1, 30), 682a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(48, hsp2, MFSEL1, 11, none, NONE, 0), 683a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(49, hsp2, MFSEL1, 11, none, NONE, 0), 684a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(50, hsp2, MFSEL1, 11, none, NONE, 0), 685a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(51, hsp2, MFSEL1, 11, none, NONE, 0), 686a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(52, hsp2, MFSEL1, 11, none, NONE, 0), 687a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(53, hsp2, MFSEL1, 11, none, NONE, 0), 688a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(54, hsp2, MFSEL1, 11, none, NONE, 0), 689a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(55, hsp2, MFSEL1, 11, none, NONE, 0), 690a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(56, r1err, MFSEL1, 12, none, NONE, 0), 691a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(57, r1md, MFSEL1, 13, none, NONE, 0), 692a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(58, r1md, MFSEL1, 13, none, NONE, 0), 693a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(59, hg6, MFSEL2, 30, none, NONE, 0), 694a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(60, hg7, MFSEL2, 31, sdio, MFSEL1, 30), 695a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(61, hsp1, MFSEL1, 10, none, NONE, 0), 696a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(62, hsp1, MFSEL1, 10, none, NONE, 0), 697a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(63, hsp1, MFSEL1, 10, none, NONE, 0), 698a1d1e0e3SJonathan Neuschäfer 699a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(64, fi0, MFSEL2, 0, none, NONE, 0), 700a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(65, fi1, MFSEL2, 1, none, NONE, 0), 701a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(66, fi2, MFSEL2, 2, none, NONE, 0), 702a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(67, fi3, MFSEL2, 3, none, NONE, 0), 703a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(68, fi4, MFSEL2, 4, none, NONE, 0), 704a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(69, fi5, MFSEL2, 5, none, NONE, 0), 705a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(70, fi6, MFSEL2, 6, none, NONE, 0), 706a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(71, fi7, MFSEL2, 7, none, NONE, 0), 707a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(72, fi8, MFSEL2, 8, none, NONE, 0), 708a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(73, fi9, MFSEL2, 9, none, NONE, 0), 709a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(74, fi10, MFSEL2, 10, none, NONE, 0), 710a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(75, fi11, MFSEL2, 11, none, NONE, 0), 711a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(76, fi12, MFSEL2, 12, none, NONE, 0), 712a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(77, fi13, MFSEL2, 13, none, NONE, 0), 713a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(78, fi14, MFSEL2, 14, none, NONE, 0), 714a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(79, fi15, MFSEL2, 15, none, NONE, 0), 715a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(80, pwm0, MFSEL2, 16, none, NONE, 0), 716a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(81, pwm1, MFSEL2, 17, none, NONE, 0), 717a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(82, pwm2, MFSEL2, 18, none, NONE, 0), 718a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(83, pwm3, MFSEL2, 19, none, NONE, 0), 719a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(84, rmii2, MFSEL1, 14, none, NONE, 0), 720a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(85, rmii2, MFSEL1, 14, none, NONE, 0), 721a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(86, rmii2, MFSEL1, 14, none, NONE, 0), 722a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(87, rmii2, MFSEL1, 14, none, NONE, 0), 723a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(88, rmii2, MFSEL1, 14, none, NONE, 0), 724a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(89, rmii2, MFSEL1, 14, none, NONE, 0), 725a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(90, r2err, MFSEL1, 15, none, NONE, 0), 726a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(91, r2md, MFSEL1, 16, none, NONE, 0), 727a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(92, r2md, MFSEL1, 16, none, NONE, 0), 728a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(93, kbcc, MFSEL1, 17, none, NONE, 0), 729a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(94, kbcc, MFSEL1, 17, none, NONE, 0), 730a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(95, none, NONE, 0, none, NONE, 0), 731a1d1e0e3SJonathan Neuschäfer 732a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(96, none, NONE, 0, none, NONE, 0), 733a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(97, none, NONE, 0, none, NONE, 0), 734a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(98, none, NONE, 0, none, NONE, 0), 735a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(99, none, NONE, 0, none, NONE, 0), 736a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(100, none, NONE, 0, none, NONE, 0), 737a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(101, none, NONE, 0, none, NONE, 0), 738a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(102, none, NONE, 0, none, NONE, 0), 739a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(103, none, NONE, 0, none, NONE, 0), 740a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(104, none, NONE, 0, none, NONE, 0), 741a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(105, none, NONE, 0, none, NONE, 0), 742a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(106, none, NONE, 0, none, NONE, 0), 743a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(107, none, NONE, 0, none, NONE, 0), 744a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(108, none, NONE, 0, none, NONE, 0), /* DVO */ 745a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(109, none, NONE, 0, none, NONE, 0), /* DVO */ 746a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(110, none, NONE, 0, none, NONE, 0), /* DVO */ 747a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(111, none, NONE, 0, none, NONE, 0), /* DVO */ 748a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(112, none, NONE, 0, none, NONE, 0), /* DVO */ 749a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(113, none, NONE, 0, none, NONE, 0), /* DVO */ 750a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(114, smb0, MFSEL1, 6, none, NONE, 0), 751a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(115, smb0, MFSEL1, 6, none, NONE, 0), 752a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(116, smb1, MFSEL1, 7, none, NONE, 0), 753a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(117, smb1, MFSEL1, 7, none, NONE, 0), 754a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(118, smb2, MFSEL1, 8, none, NONE, 0), 755a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(119, smb2, MFSEL1, 8, none, NONE, 0), 756a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(120, none, NONE, 0, none, NONE, 0), /* DVO */ 757a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(121, none, NONE, 0, none, NONE, 0), /* DVO */ 758a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(122, none, NONE, 0, none, NONE, 0), /* DVO */ 759a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(123, none, NONE, 0, none, NONE, 0), /* DVO */ 760a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(124, none, NONE, 0, none, NONE, 0), /* DVO */ 761a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(125, none, NONE, 0, none, NONE, 0), /* DVO */ 762a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(126, none, NONE, 0, none, NONE, 0), /* DVO */ 763a1d1e0e3SJonathan Neuschäfer WPCM450_PINCFG(127, none, NONE, 0, none, NONE, 0), /* DVO */ 764a1d1e0e3SJonathan Neuschäfer }; 765a1d1e0e3SJonathan Neuschäfer 766a1d1e0e3SJonathan Neuschäfer #define WPCM450_PIN(n) PINCTRL_PIN(n, "gpio" #n) 767a1d1e0e3SJonathan Neuschäfer 768a1d1e0e3SJonathan Neuschäfer static const struct pinctrl_pin_desc wpcm450_pins[] = { 769a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(0), WPCM450_PIN(1), WPCM450_PIN(2), WPCM450_PIN(3), 770a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(4), WPCM450_PIN(5), WPCM450_PIN(6), WPCM450_PIN(7), 771a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(8), WPCM450_PIN(9), WPCM450_PIN(10), WPCM450_PIN(11), 772a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(12), WPCM450_PIN(13), WPCM450_PIN(14), WPCM450_PIN(15), 773a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(16), WPCM450_PIN(17), WPCM450_PIN(18), WPCM450_PIN(19), 774a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(20), WPCM450_PIN(21), WPCM450_PIN(22), WPCM450_PIN(23), 775a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(24), WPCM450_PIN(25), WPCM450_PIN(26), WPCM450_PIN(27), 776a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(28), WPCM450_PIN(29), WPCM450_PIN(30), WPCM450_PIN(31), 777a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(32), WPCM450_PIN(33), WPCM450_PIN(34), WPCM450_PIN(35), 778a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(36), WPCM450_PIN(37), WPCM450_PIN(38), WPCM450_PIN(39), 779a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(40), WPCM450_PIN(41), WPCM450_PIN(42), WPCM450_PIN(43), 780a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(44), WPCM450_PIN(45), WPCM450_PIN(46), WPCM450_PIN(47), 781a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(48), WPCM450_PIN(49), WPCM450_PIN(50), WPCM450_PIN(51), 782a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(52), WPCM450_PIN(53), WPCM450_PIN(54), WPCM450_PIN(55), 783a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(56), WPCM450_PIN(57), WPCM450_PIN(58), WPCM450_PIN(59), 784a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(60), WPCM450_PIN(61), WPCM450_PIN(62), WPCM450_PIN(63), 785a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(64), WPCM450_PIN(65), WPCM450_PIN(66), WPCM450_PIN(67), 786a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(68), WPCM450_PIN(69), WPCM450_PIN(70), WPCM450_PIN(71), 787a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(72), WPCM450_PIN(73), WPCM450_PIN(74), WPCM450_PIN(75), 788a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(76), WPCM450_PIN(77), WPCM450_PIN(78), WPCM450_PIN(79), 789a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(80), WPCM450_PIN(81), WPCM450_PIN(82), WPCM450_PIN(83), 790a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(84), WPCM450_PIN(85), WPCM450_PIN(86), WPCM450_PIN(87), 791a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(88), WPCM450_PIN(89), WPCM450_PIN(90), WPCM450_PIN(91), 792a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(92), WPCM450_PIN(93), WPCM450_PIN(94), WPCM450_PIN(95), 793a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(96), WPCM450_PIN(97), WPCM450_PIN(98), WPCM450_PIN(99), 794a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(100), WPCM450_PIN(101), WPCM450_PIN(102), WPCM450_PIN(103), 795a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(104), WPCM450_PIN(105), WPCM450_PIN(106), WPCM450_PIN(107), 796a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(108), WPCM450_PIN(109), WPCM450_PIN(110), WPCM450_PIN(111), 797a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(112), WPCM450_PIN(113), WPCM450_PIN(114), WPCM450_PIN(115), 798a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(116), WPCM450_PIN(117), WPCM450_PIN(118), WPCM450_PIN(119), 799a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(120), WPCM450_PIN(121), WPCM450_PIN(122), WPCM450_PIN(123), 800a1d1e0e3SJonathan Neuschäfer WPCM450_PIN(124), WPCM450_PIN(125), WPCM450_PIN(126), WPCM450_PIN(127), 801a1d1e0e3SJonathan Neuschäfer }; 802a1d1e0e3SJonathan Neuschäfer 803*4f1d423cSJonathan Neuschäfer /* Helper function to update MFSEL field according to the selected function */ 804*4f1d423cSJonathan Neuschäfer static void wpcm450_update_mfsel(struct regmap *gcr_regmap, int reg, int bit, int fn, int fn_selected) 805*4f1d423cSJonathan Neuschäfer { 806*4f1d423cSJonathan Neuschäfer bool value = (fn == fn_selected); 807*4f1d423cSJonathan Neuschäfer 808*4f1d423cSJonathan Neuschäfer regmap_update_bits(gcr_regmap, reg, BIT(bit), value ? BIT(bit) : 0); 809*4f1d423cSJonathan Neuschäfer } 810*4f1d423cSJonathan Neuschäfer 811a1d1e0e3SJonathan Neuschäfer /* Enable mode in pin group */ 812a1d1e0e3SJonathan Neuschäfer static void wpcm450_setfunc(struct regmap *gcr_regmap, const unsigned int *pin, 813a1d1e0e3SJonathan Neuschäfer int npins, int func) 814a1d1e0e3SJonathan Neuschäfer { 815a1d1e0e3SJonathan Neuschäfer const struct wpcm450_pincfg *cfg; 816a1d1e0e3SJonathan Neuschäfer int i; 817a1d1e0e3SJonathan Neuschäfer 818a1d1e0e3SJonathan Neuschäfer for (i = 0; i < npins; i++) { 819a1d1e0e3SJonathan Neuschäfer cfg = &pincfg[pin[i]]; 820a1d1e0e3SJonathan Neuschäfer if (func == fn_gpio || cfg->fn0 == func || cfg->fn1 == func) { 821a1d1e0e3SJonathan Neuschäfer if (cfg->reg0) 822*4f1d423cSJonathan Neuschäfer wpcm450_update_mfsel(gcr_regmap, cfg->reg0, 823*4f1d423cSJonathan Neuschäfer cfg->bit0, cfg->fn0, func); 824a1d1e0e3SJonathan Neuschäfer if (cfg->reg1) 825*4f1d423cSJonathan Neuschäfer wpcm450_update_mfsel(gcr_regmap, cfg->reg1, 826*4f1d423cSJonathan Neuschäfer cfg->bit1, cfg->fn1, func); 827a1d1e0e3SJonathan Neuschäfer } 828a1d1e0e3SJonathan Neuschäfer } 829a1d1e0e3SJonathan Neuschäfer } 830a1d1e0e3SJonathan Neuschäfer 831a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_groups_count(struct pinctrl_dev *pctldev) 832a1d1e0e3SJonathan Neuschäfer { 833a1d1e0e3SJonathan Neuschäfer return ARRAY_SIZE(wpcm450_groups); 834a1d1e0e3SJonathan Neuschäfer } 835a1d1e0e3SJonathan Neuschäfer 836a1d1e0e3SJonathan Neuschäfer static const char *wpcm450_get_group_name(struct pinctrl_dev *pctldev, 837a1d1e0e3SJonathan Neuschäfer unsigned int selector) 838a1d1e0e3SJonathan Neuschäfer { 839a1d1e0e3SJonathan Neuschäfer return wpcm450_groups[selector].name; 840a1d1e0e3SJonathan Neuschäfer } 841a1d1e0e3SJonathan Neuschäfer 842a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_group_pins(struct pinctrl_dev *pctldev, 843a1d1e0e3SJonathan Neuschäfer unsigned int selector, 844a1d1e0e3SJonathan Neuschäfer const unsigned int **pins, 845a1d1e0e3SJonathan Neuschäfer unsigned int *npins) 846a1d1e0e3SJonathan Neuschäfer { 847a1d1e0e3SJonathan Neuschäfer *npins = wpcm450_groups[selector].num_pins; 848a1d1e0e3SJonathan Neuschäfer *pins = wpcm450_groups[selector].pins; 849a1d1e0e3SJonathan Neuschäfer 850a1d1e0e3SJonathan Neuschäfer return 0; 851a1d1e0e3SJonathan Neuschäfer } 852a1d1e0e3SJonathan Neuschäfer 853a1d1e0e3SJonathan Neuschäfer static int wpcm450_dt_node_to_map(struct pinctrl_dev *pctldev, 854a1d1e0e3SJonathan Neuschäfer struct device_node *np_config, 855a1d1e0e3SJonathan Neuschäfer struct pinctrl_map **map, 856a1d1e0e3SJonathan Neuschäfer u32 *num_maps) 857a1d1e0e3SJonathan Neuschäfer { 858a1d1e0e3SJonathan Neuschäfer return pinconf_generic_dt_node_to_map(pctldev, np_config, 859a1d1e0e3SJonathan Neuschäfer map, num_maps, 860a1d1e0e3SJonathan Neuschäfer PIN_MAP_TYPE_INVALID); 861a1d1e0e3SJonathan Neuschäfer } 862a1d1e0e3SJonathan Neuschäfer 863a1d1e0e3SJonathan Neuschäfer static void wpcm450_dt_free_map(struct pinctrl_dev *pctldev, 864a1d1e0e3SJonathan Neuschäfer struct pinctrl_map *map, u32 num_maps) 865a1d1e0e3SJonathan Neuschäfer { 866a1d1e0e3SJonathan Neuschäfer kfree(map); 867a1d1e0e3SJonathan Neuschäfer } 868a1d1e0e3SJonathan Neuschäfer 869a1d1e0e3SJonathan Neuschäfer static const struct pinctrl_ops wpcm450_pinctrl_ops = { 870a1d1e0e3SJonathan Neuschäfer .get_groups_count = wpcm450_get_groups_count, 871a1d1e0e3SJonathan Neuschäfer .get_group_name = wpcm450_get_group_name, 872a1d1e0e3SJonathan Neuschäfer .get_group_pins = wpcm450_get_group_pins, 873a1d1e0e3SJonathan Neuschäfer .dt_node_to_map = wpcm450_dt_node_to_map, 874a1d1e0e3SJonathan Neuschäfer .dt_free_map = wpcm450_dt_free_map, 875a1d1e0e3SJonathan Neuschäfer }; 876a1d1e0e3SJonathan Neuschäfer 877a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_functions_count(struct pinctrl_dev *pctldev) 878a1d1e0e3SJonathan Neuschäfer { 879a1d1e0e3SJonathan Neuschäfer return ARRAY_SIZE(wpcm450_funcs); 880a1d1e0e3SJonathan Neuschäfer } 881a1d1e0e3SJonathan Neuschäfer 882a1d1e0e3SJonathan Neuschäfer static const char *wpcm450_get_function_name(struct pinctrl_dev *pctldev, 883a1d1e0e3SJonathan Neuschäfer unsigned int function) 884a1d1e0e3SJonathan Neuschäfer { 885a1d1e0e3SJonathan Neuschäfer return wpcm450_funcs[function].name; 886a1d1e0e3SJonathan Neuschäfer } 887a1d1e0e3SJonathan Neuschäfer 888a1d1e0e3SJonathan Neuschäfer static int wpcm450_get_function_groups(struct pinctrl_dev *pctldev, 889a1d1e0e3SJonathan Neuschäfer unsigned int function, 890a1d1e0e3SJonathan Neuschäfer const char * const **groups, 891a1d1e0e3SJonathan Neuschäfer unsigned int * const ngroups) 892a1d1e0e3SJonathan Neuschäfer { 893a1d1e0e3SJonathan Neuschäfer *ngroups = wpcm450_funcs[function].ngroups; 894a1d1e0e3SJonathan Neuschäfer *groups = wpcm450_funcs[function].groups; 895a1d1e0e3SJonathan Neuschäfer 896a1d1e0e3SJonathan Neuschäfer return 0; 897a1d1e0e3SJonathan Neuschäfer } 898a1d1e0e3SJonathan Neuschäfer 899a1d1e0e3SJonathan Neuschäfer static int wpcm450_pinmux_set_mux(struct pinctrl_dev *pctldev, 900a1d1e0e3SJonathan Neuschäfer unsigned int function, 901a1d1e0e3SJonathan Neuschäfer unsigned int group) 902a1d1e0e3SJonathan Neuschäfer { 903a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev); 904a1d1e0e3SJonathan Neuschäfer 905a1d1e0e3SJonathan Neuschäfer wpcm450_setfunc(pctrl->gcr_regmap, wpcm450_groups[group].pins, 906a1d1e0e3SJonathan Neuschäfer wpcm450_groups[group].num_pins, function); 907a1d1e0e3SJonathan Neuschäfer 908a1d1e0e3SJonathan Neuschäfer return 0; 909a1d1e0e3SJonathan Neuschäfer } 910a1d1e0e3SJonathan Neuschäfer 911a1d1e0e3SJonathan Neuschäfer static const struct pinmux_ops wpcm450_pinmux_ops = { 912a1d1e0e3SJonathan Neuschäfer .get_functions_count = wpcm450_get_functions_count, 913a1d1e0e3SJonathan Neuschäfer .get_function_name = wpcm450_get_function_name, 914a1d1e0e3SJonathan Neuschäfer .get_function_groups = wpcm450_get_function_groups, 915a1d1e0e3SJonathan Neuschäfer .set_mux = wpcm450_pinmux_set_mux, 916a1d1e0e3SJonathan Neuschäfer }; 917a1d1e0e3SJonathan Neuschäfer 918a1d1e0e3SJonathan Neuschäfer static int debounce_bitnum(int gpio) 919a1d1e0e3SJonathan Neuschäfer { 920a1d1e0e3SJonathan Neuschäfer if (gpio >= 0 && gpio < 16) 921a1d1e0e3SJonathan Neuschäfer return gpio; 922a1d1e0e3SJonathan Neuschäfer return -EINVAL; 923a1d1e0e3SJonathan Neuschäfer } 924a1d1e0e3SJonathan Neuschäfer 925a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_get(struct pinctrl_dev *pctldev, unsigned int pin, 926a1d1e0e3SJonathan Neuschäfer unsigned long *config) 927a1d1e0e3SJonathan Neuschäfer { 928a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev); 929a1d1e0e3SJonathan Neuschäfer enum pin_config_param param = pinconf_to_config_param(*config); 930a1d1e0e3SJonathan Neuschäfer unsigned long flags; 931a1d1e0e3SJonathan Neuschäfer int bit; 932a1d1e0e3SJonathan Neuschäfer u32 reg; 933a1d1e0e3SJonathan Neuschäfer 934a1d1e0e3SJonathan Neuschäfer switch (param) { 935a1d1e0e3SJonathan Neuschäfer case PIN_CONFIG_INPUT_DEBOUNCE: 936a1d1e0e3SJonathan Neuschäfer bit = debounce_bitnum(pin); 937a1d1e0e3SJonathan Neuschäfer if (bit < 0) 938a1d1e0e3SJonathan Neuschäfer return bit; 939a1d1e0e3SJonathan Neuschäfer 940a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags); 941a1d1e0e3SJonathan Neuschäfer reg = ioread32(pctrl->gpio_base + WPCM450_GPEVDBNC); 942a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags); 943a1d1e0e3SJonathan Neuschäfer 944a1d1e0e3SJonathan Neuschäfer *config = pinconf_to_config_packed(param, !!(reg & BIT(bit))); 945a1d1e0e3SJonathan Neuschäfer return 0; 946a1d1e0e3SJonathan Neuschäfer default: 947a1d1e0e3SJonathan Neuschäfer return -ENOTSUPP; 948a1d1e0e3SJonathan Neuschäfer } 949a1d1e0e3SJonathan Neuschäfer } 950a1d1e0e3SJonathan Neuschäfer 951a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_set_one(struct wpcm450_pinctrl *pctrl, 952a1d1e0e3SJonathan Neuschäfer unsigned int pin, unsigned long config) 953a1d1e0e3SJonathan Neuschäfer { 954a1d1e0e3SJonathan Neuschäfer enum pin_config_param param = pinconf_to_config_param(config); 955a1d1e0e3SJonathan Neuschäfer unsigned long flags; 956a1d1e0e3SJonathan Neuschäfer unsigned long reg; 957a1d1e0e3SJonathan Neuschäfer int bit; 958a1d1e0e3SJonathan Neuschäfer int arg; 959a1d1e0e3SJonathan Neuschäfer 960a1d1e0e3SJonathan Neuschäfer switch (param) { 961a1d1e0e3SJonathan Neuschäfer case PIN_CONFIG_INPUT_DEBOUNCE: 962a1d1e0e3SJonathan Neuschäfer bit = debounce_bitnum(pin); 963a1d1e0e3SJonathan Neuschäfer if (bit < 0) 964a1d1e0e3SJonathan Neuschäfer return bit; 965a1d1e0e3SJonathan Neuschäfer 966a1d1e0e3SJonathan Neuschäfer arg = pinconf_to_config_argument(config); 967a1d1e0e3SJonathan Neuschäfer 968a1d1e0e3SJonathan Neuschäfer raw_spin_lock_irqsave(&pctrl->lock, flags); 969a1d1e0e3SJonathan Neuschäfer reg = ioread32(pctrl->gpio_base + WPCM450_GPEVDBNC); 970a1d1e0e3SJonathan Neuschäfer __assign_bit(bit, ®, arg); 971a1d1e0e3SJonathan Neuschäfer iowrite32(reg, pctrl->gpio_base + WPCM450_GPEVDBNC); 972a1d1e0e3SJonathan Neuschäfer raw_spin_unlock_irqrestore(&pctrl->lock, flags); 973a1d1e0e3SJonathan Neuschäfer return 0; 974a1d1e0e3SJonathan Neuschäfer default: 975a1d1e0e3SJonathan Neuschäfer return -ENOTSUPP; 976a1d1e0e3SJonathan Neuschäfer } 977a1d1e0e3SJonathan Neuschäfer } 978a1d1e0e3SJonathan Neuschäfer 979a1d1e0e3SJonathan Neuschäfer static int wpcm450_config_set(struct pinctrl_dev *pctldev, unsigned int pin, 980a1d1e0e3SJonathan Neuschäfer unsigned long *configs, unsigned int num_configs) 981a1d1e0e3SJonathan Neuschäfer { 982a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl = pinctrl_dev_get_drvdata(pctldev); 983a1d1e0e3SJonathan Neuschäfer int ret; 984a1d1e0e3SJonathan Neuschäfer 985a1d1e0e3SJonathan Neuschäfer while (num_configs--) { 986a1d1e0e3SJonathan Neuschäfer ret = wpcm450_config_set_one(pctrl, pin, *configs++); 987a1d1e0e3SJonathan Neuschäfer if (ret) 988a1d1e0e3SJonathan Neuschäfer return ret; 989a1d1e0e3SJonathan Neuschäfer } 990a1d1e0e3SJonathan Neuschäfer 991a1d1e0e3SJonathan Neuschäfer return 0; 992a1d1e0e3SJonathan Neuschäfer } 993a1d1e0e3SJonathan Neuschäfer 994a1d1e0e3SJonathan Neuschäfer static const struct pinconf_ops wpcm450_pinconf_ops = { 995a1d1e0e3SJonathan Neuschäfer .is_generic = true, 996a1d1e0e3SJonathan Neuschäfer .pin_config_get = wpcm450_config_get, 997a1d1e0e3SJonathan Neuschäfer .pin_config_set = wpcm450_config_set, 998a1d1e0e3SJonathan Neuschäfer }; 999a1d1e0e3SJonathan Neuschäfer 1000a1d1e0e3SJonathan Neuschäfer static struct pinctrl_desc wpcm450_pinctrl_desc = { 1001a1d1e0e3SJonathan Neuschäfer .name = "wpcm450-pinctrl", 1002a1d1e0e3SJonathan Neuschäfer .pins = wpcm450_pins, 1003a1d1e0e3SJonathan Neuschäfer .npins = ARRAY_SIZE(wpcm450_pins), 1004a1d1e0e3SJonathan Neuschäfer .pctlops = &wpcm450_pinctrl_ops, 1005a1d1e0e3SJonathan Neuschäfer .pmxops = &wpcm450_pinmux_ops, 1006a1d1e0e3SJonathan Neuschäfer .confops = &wpcm450_pinconf_ops, 1007a1d1e0e3SJonathan Neuschäfer .owner = THIS_MODULE, 1008a1d1e0e3SJonathan Neuschäfer }; 1009a1d1e0e3SJonathan Neuschäfer 1010a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_set_config(struct gpio_chip *chip, 1011a1d1e0e3SJonathan Neuschäfer unsigned int offset, unsigned long config) 1012a1d1e0e3SJonathan Neuschäfer { 1013a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(chip); 1014a1d1e0e3SJonathan Neuschäfer 1015a1d1e0e3SJonathan Neuschäfer return wpcm450_config_set_one(gpio->pctrl, offset, config); 1016a1d1e0e3SJonathan Neuschäfer } 1017a1d1e0e3SJonathan Neuschäfer 1018a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_add_pin_ranges(struct gpio_chip *chip) 1019a1d1e0e3SJonathan Neuschäfer { 1020a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio = gpiochip_get_data(chip); 1021a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank = gpio->bank; 1022a1d1e0e3SJonathan Neuschäfer 1023a1d1e0e3SJonathan Neuschäfer return gpiochip_add_pin_range(&gpio->gc, dev_name(gpio->pctrl->dev), 1024a1d1e0e3SJonathan Neuschäfer 0, bank->base, bank->length); 1025a1d1e0e3SJonathan Neuschäfer } 1026a1d1e0e3SJonathan Neuschäfer 1027a1d1e0e3SJonathan Neuschäfer static int wpcm450_gpio_register(struct platform_device *pdev, 1028a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl) 1029a1d1e0e3SJonathan Neuschäfer { 1030a1d1e0e3SJonathan Neuschäfer struct device *dev = &pdev->dev; 1031a1d1e0e3SJonathan Neuschäfer struct fwnode_handle *child; 1032a1d1e0e3SJonathan Neuschäfer int ret; 1033a1d1e0e3SJonathan Neuschäfer 1034a1d1e0e3SJonathan Neuschäfer pctrl->gpio_base = devm_platform_ioremap_resource(pdev, 0); 1035283fb4eaSJialin Zhang if (IS_ERR(pctrl->gpio_base)) 1036283fb4eaSJialin Zhang return dev_err_probe(dev, PTR_ERR(pctrl->gpio_base), 1037283fb4eaSJialin Zhang "Resource fail for GPIO controller\n"); 1038a1d1e0e3SJonathan Neuschäfer 1039a1d1e0e3SJonathan Neuschäfer device_for_each_child_node(dev, child) { 1040a1d1e0e3SJonathan Neuschäfer void __iomem *dat = NULL; 1041a1d1e0e3SJonathan Neuschäfer void __iomem *set = NULL; 1042a1d1e0e3SJonathan Neuschäfer void __iomem *dirout = NULL; 1043a1d1e0e3SJonathan Neuschäfer unsigned long flags = 0; 1044a1d1e0e3SJonathan Neuschäfer const struct wpcm450_bank *bank; 1045a1d1e0e3SJonathan Neuschäfer struct wpcm450_gpio *gpio; 1046a1d1e0e3SJonathan Neuschäfer struct gpio_irq_chip *girq; 1047a1d1e0e3SJonathan Neuschäfer u32 reg; 1048a1d1e0e3SJonathan Neuschäfer int i; 1049a1d1e0e3SJonathan Neuschäfer 1050a1d1e0e3SJonathan Neuschäfer if (!fwnode_property_read_bool(child, "gpio-controller")) 1051a1d1e0e3SJonathan Neuschäfer continue; 1052a1d1e0e3SJonathan Neuschäfer 1053a1d1e0e3SJonathan Neuschäfer ret = fwnode_property_read_u32(child, "reg", ®); 1054a1d1e0e3SJonathan Neuschäfer if (ret < 0) 1055a1d1e0e3SJonathan Neuschäfer return ret; 1056a1d1e0e3SJonathan Neuschäfer 1057a1d1e0e3SJonathan Neuschäfer gpio = &pctrl->gpio_bank[reg]; 1058a1d1e0e3SJonathan Neuschäfer gpio->pctrl = pctrl; 1059a1d1e0e3SJonathan Neuschäfer 1060ce2076baSDan Carpenter if (reg >= WPCM450_NUM_BANKS) 1061a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, -EINVAL, 1062a1d1e0e3SJonathan Neuschäfer "GPIO index %d out of range!\n", reg); 1063a1d1e0e3SJonathan Neuschäfer 1064a1d1e0e3SJonathan Neuschäfer bank = &wpcm450_banks[reg]; 1065a1d1e0e3SJonathan Neuschäfer gpio->bank = bank; 1066a1d1e0e3SJonathan Neuschäfer 1067a1d1e0e3SJonathan Neuschäfer dat = pctrl->gpio_base + bank->datain; 1068a1d1e0e3SJonathan Neuschäfer if (bank->dataout) { 1069a1d1e0e3SJonathan Neuschäfer set = pctrl->gpio_base + bank->dataout; 1070a1d1e0e3SJonathan Neuschäfer dirout = pctrl->gpio_base + bank->cfg0; 1071a1d1e0e3SJonathan Neuschäfer } else { 1072a1d1e0e3SJonathan Neuschäfer flags = BGPIOF_NO_OUTPUT; 1073a1d1e0e3SJonathan Neuschäfer } 1074a1d1e0e3SJonathan Neuschäfer ret = bgpio_init(&gpio->gc, dev, 4, 1075a1d1e0e3SJonathan Neuschäfer dat, set, NULL, dirout, NULL, flags); 1076a1d1e0e3SJonathan Neuschäfer if (ret < 0) 1077a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, ret, "GPIO initialization failed\n"); 1078a1d1e0e3SJonathan Neuschäfer 1079a1d1e0e3SJonathan Neuschäfer gpio->gc.ngpio = bank->length; 1080a1d1e0e3SJonathan Neuschäfer gpio->gc.set_config = wpcm450_gpio_set_config; 1081a1d1e0e3SJonathan Neuschäfer gpio->gc.fwnode = child; 1082a1d1e0e3SJonathan Neuschäfer gpio->gc.add_pin_ranges = wpcm450_gpio_add_pin_ranges; 1083a1d1e0e3SJonathan Neuschäfer 1084a1d1e0e3SJonathan Neuschäfer girq = &gpio->gc.irq; 108597775ebbSJonathan Neuschäfer gpio_irq_chip_set_chip(girq, &wpcm450_gpio_irqchip); 1086a1d1e0e3SJonathan Neuschäfer girq->parent_handler = wpcm450_gpio_irqhandler; 1087a1d1e0e3SJonathan Neuschäfer girq->parents = devm_kcalloc(dev, WPCM450_NUM_GPIO_IRQS, 1088a1d1e0e3SJonathan Neuschäfer sizeof(*girq->parents), GFP_KERNEL); 1089a1d1e0e3SJonathan Neuschäfer if (!girq->parents) 1090a1d1e0e3SJonathan Neuschäfer return -ENOMEM; 1091a1d1e0e3SJonathan Neuschäfer girq->default_type = IRQ_TYPE_NONE; 1092a1d1e0e3SJonathan Neuschäfer girq->handler = handle_bad_irq; 1093a1d1e0e3SJonathan Neuschäfer 1094a1d1e0e3SJonathan Neuschäfer girq->num_parents = 0; 1095a1d1e0e3SJonathan Neuschäfer for (i = 0; i < WPCM450_NUM_GPIO_IRQS; i++) { 1096f4a31facSAndy Shevchenko int irq; 1097a1d1e0e3SJonathan Neuschäfer 1098f4a31facSAndy Shevchenko irq = fwnode_irq_get(child, i); 1099a1d1e0e3SJonathan Neuschäfer if (irq < 0) 1100a1d1e0e3SJonathan Neuschäfer break; 1101f4a31facSAndy Shevchenko if (!irq) 1102f4a31facSAndy Shevchenko continue; 1103a1d1e0e3SJonathan Neuschäfer 1104a1d1e0e3SJonathan Neuschäfer girq->parents[i] = irq; 1105a1d1e0e3SJonathan Neuschäfer girq->num_parents++; 1106a1d1e0e3SJonathan Neuschäfer } 1107a1d1e0e3SJonathan Neuschäfer 1108a1d1e0e3SJonathan Neuschäfer ret = devm_gpiochip_add_data(dev, &gpio->gc, gpio); 1109a1d1e0e3SJonathan Neuschäfer if (ret) 1110a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, ret, "Failed to add GPIO chip\n"); 1111a1d1e0e3SJonathan Neuschäfer } 1112a1d1e0e3SJonathan Neuschäfer 1113a1d1e0e3SJonathan Neuschäfer return 0; 1114a1d1e0e3SJonathan Neuschäfer } 1115a1d1e0e3SJonathan Neuschäfer 1116a1d1e0e3SJonathan Neuschäfer static int wpcm450_pinctrl_probe(struct platform_device *pdev) 1117a1d1e0e3SJonathan Neuschäfer { 1118a1d1e0e3SJonathan Neuschäfer struct device *dev = &pdev->dev; 1119a1d1e0e3SJonathan Neuschäfer struct wpcm450_pinctrl *pctrl; 1120a1d1e0e3SJonathan Neuschäfer int ret; 1121a1d1e0e3SJonathan Neuschäfer 1122a1d1e0e3SJonathan Neuschäfer pctrl = devm_kzalloc(dev, sizeof(*pctrl), GFP_KERNEL); 1123a1d1e0e3SJonathan Neuschäfer if (!pctrl) 1124a1d1e0e3SJonathan Neuschäfer return -ENOMEM; 1125a1d1e0e3SJonathan Neuschäfer 1126a1d1e0e3SJonathan Neuschäfer pctrl->dev = &pdev->dev; 1127a1d1e0e3SJonathan Neuschäfer raw_spin_lock_init(&pctrl->lock); 1128a1d1e0e3SJonathan Neuschäfer dev_set_drvdata(dev, pctrl); 1129a1d1e0e3SJonathan Neuschäfer 1130a1d1e0e3SJonathan Neuschäfer pctrl->gcr_regmap = 1131a1d1e0e3SJonathan Neuschäfer syscon_regmap_lookup_by_compatible("nuvoton,wpcm450-gcr"); 1132a1d1e0e3SJonathan Neuschäfer if (IS_ERR(pctrl->gcr_regmap)) 1133a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, PTR_ERR(pctrl->gcr_regmap), 1134a1d1e0e3SJonathan Neuschäfer "Failed to find nuvoton,wpcm450-gcr\n"); 1135a1d1e0e3SJonathan Neuschäfer 1136a1d1e0e3SJonathan Neuschäfer pctrl->pctldev = devm_pinctrl_register(dev, 1137a1d1e0e3SJonathan Neuschäfer &wpcm450_pinctrl_desc, pctrl); 1138a1d1e0e3SJonathan Neuschäfer if (IS_ERR(pctrl->pctldev)) 1139a1d1e0e3SJonathan Neuschäfer return dev_err_probe(dev, PTR_ERR(pctrl->pctldev), 1140a1d1e0e3SJonathan Neuschäfer "Failed to register pinctrl device\n"); 1141a1d1e0e3SJonathan Neuschäfer 1142a1d1e0e3SJonathan Neuschäfer ret = wpcm450_gpio_register(pdev, pctrl); 1143a1d1e0e3SJonathan Neuschäfer if (ret < 0) 1144a1d1e0e3SJonathan Neuschäfer return ret; 1145a1d1e0e3SJonathan Neuschäfer 1146a1d1e0e3SJonathan Neuschäfer return 0; 1147a1d1e0e3SJonathan Neuschäfer } 1148a1d1e0e3SJonathan Neuschäfer 1149a1d1e0e3SJonathan Neuschäfer static const struct of_device_id wpcm450_pinctrl_match[] = { 1150a1d1e0e3SJonathan Neuschäfer { .compatible = "nuvoton,wpcm450-pinctrl" }, 1151a1d1e0e3SJonathan Neuschäfer { } 1152a1d1e0e3SJonathan Neuschäfer }; 1153a1d1e0e3SJonathan Neuschäfer MODULE_DEVICE_TABLE(of, wpcm450_pinctrl_match); 1154a1d1e0e3SJonathan Neuschäfer 1155a1d1e0e3SJonathan Neuschäfer static struct platform_driver wpcm450_pinctrl_driver = { 1156a1d1e0e3SJonathan Neuschäfer .probe = wpcm450_pinctrl_probe, 1157a1d1e0e3SJonathan Neuschäfer .driver = { 1158a1d1e0e3SJonathan Neuschäfer .name = "wpcm450-pinctrl", 1159a1d1e0e3SJonathan Neuschäfer .of_match_table = wpcm450_pinctrl_match, 1160a1d1e0e3SJonathan Neuschäfer }, 1161a1d1e0e3SJonathan Neuschäfer }; 1162a1d1e0e3SJonathan Neuschäfer module_platform_driver(wpcm450_pinctrl_driver); 1163a1d1e0e3SJonathan Neuschäfer 1164a1d1e0e3SJonathan Neuschäfer MODULE_LICENSE("GPL v2"); 1165a1d1e0e3SJonathan Neuschäfer MODULE_AUTHOR("Jonathan Neuschäfer <j.neuschaefer@gmx.net>"); 1166a1d1e0e3SJonathan Neuschäfer MODULE_DESCRIPTION("Nuvoton WPCM450 Pinctrl and GPIO driver"); 1167