// SPDX-License-Identifier: GPL-2.0 /* Copyright (C) 2020 Daniel Palmer */ #include #include #include #include #include #include #include #include #include #include #include #include #define DRIVER_NAME "gpio-msc313" #define MSC313_GPIO_IN BIT(0) #define MSC313_GPIO_OUT BIT(4) #define MSC313_GPIO_OEN BIT(5) /* * These bits need to be saved to correctly restore the * gpio state when resuming from suspend to memory. */ #define MSC313_GPIO_BITSTOSAVE (MSC313_GPIO_OUT | MSC313_GPIO_OEN) /* pad names for fuart, same for all SoCs so far */ #define MSC313_PINNAME_FUART_RX "fuart_rx" #define MSC313_PINNAME_FUART_TX "fuart_tx" #define MSC313_PINNAME_FUART_CTS "fuart_cts" #define MSC313_PINNAME_FUART_RTS "fuart_rts" /* pad names for sr, mercury5 is different */ #define MSC313_PINNAME_SR_IO2 "sr_io2" #define MSC313_PINNAME_SR_IO3 "sr_io3" #define MSC313_PINNAME_SR_IO4 "sr_io4" #define MSC313_PINNAME_SR_IO5 "sr_io5" #define MSC313_PINNAME_SR_IO6 "sr_io6" #define MSC313_PINNAME_SR_IO7 "sr_io7" #define MSC313_PINNAME_SR_IO8 "sr_io8" #define MSC313_PINNAME_SR_IO9 "sr_io9" #define MSC313_PINNAME_SR_IO10 "sr_io10" #define MSC313_PINNAME_SR_IO11 "sr_io11" #define MSC313_PINNAME_SR_IO12 "sr_io12" #define MSC313_PINNAME_SR_IO13 "sr_io13" #define MSC313_PINNAME_SR_IO14 "sr_io14" #define MSC313_PINNAME_SR_IO15 "sr_io15" #define MSC313_PINNAME_SR_IO16 "sr_io16" #define MSC313_PINNAME_SR_IO17 "sr_io17" /* pad names for sd, same for all SoCs so far */ #define MSC313_PINNAME_SD_CLK "sd_clk" #define MSC313_PINNAME_SD_CMD "sd_cmd" #define MSC313_PINNAME_SD_D0 "sd_d0" #define MSC313_PINNAME_SD_D1 "sd_d1" #define MSC313_PINNAME_SD_D2 "sd_d2" #define MSC313_PINNAME_SD_D3 "sd_d3" /* pad names for i2c1, same for all SoCs so for */ #define MSC313_PINNAME_I2C1_SCL "i2c1_scl" #define MSC313_PINNAME_I2C1_SCA "i2c1_sda" /* pad names for spi0, same for all SoCs so far */ #define MSC313_PINNAME_SPI0_CZ "spi0_cz" #define MSC313_PINNAME_SPI0_CK "spi0_ck" #define MSC313_PINNAME_SPI0_DI "spi0_di" #define MSC313_PINNAME_SPI0_DO "spi0_do" #define FUART_NAMES \ MSC313_PINNAME_FUART_RX, \ MSC313_PINNAME_FUART_TX, \ MSC313_PINNAME_FUART_CTS, \ MSC313_PINNAME_FUART_RTS #define OFF_FUART_RX 0x50 #define OFF_FUART_TX 0x54 #define OFF_FUART_CTS 0x58 #define OFF_FUART_RTS 0x5c #define FUART_OFFSETS \ OFF_FUART_RX, \ OFF_FUART_TX, \ OFF_FUART_CTS, \ OFF_FUART_RTS #define SR_NAMES \ MSC313_PINNAME_SR_IO2, \ MSC313_PINNAME_SR_IO3, \ MSC313_PINNAME_SR_IO4, \ MSC313_PINNAME_SR_IO5, \ MSC313_PINNAME_SR_IO6, \ MSC313_PINNAME_SR_IO7, \ MSC313_PINNAME_SR_IO8, \ MSC313_PINNAME_SR_IO9, \ MSC313_PINNAME_SR_IO10, \ MSC313_PINNAME_SR_IO11, \ MSC313_PINNAME_SR_IO12, \ MSC313_PINNAME_SR_IO13, \ MSC313_PINNAME_SR_IO14, \ MSC313_PINNAME_SR_IO15, \ MSC313_PINNAME_SR_IO16, \ MSC313_PINNAME_SR_IO17 #define OFF_SR_IO2 0x88 #define OFF_SR_IO3 0x8c #define OFF_SR_IO4 0x90 #define OFF_SR_IO5 0x94 #define OFF_SR_IO6 0x98 #define OFF_SR_IO7 0x9c #define OFF_SR_IO8 0xa0 #define OFF_SR_IO9 0xa4 #define OFF_SR_IO10 0xa8 #define OFF_SR_IO11 0xac #define OFF_SR_IO12 0xb0 #define OFF_SR_IO13 0xb4 #define OFF_SR_IO14 0xb8 #define OFF_SR_IO15 0xbc #define OFF_SR_IO16 0xc0 #define OFF_SR_IO17 0xc4 #define SR_OFFSETS \ OFF_SR_IO2, \ OFF_SR_IO3, \ OFF_SR_IO4, \ OFF_SR_IO5, \ OFF_SR_IO6, \ OFF_SR_IO7, \ OFF_SR_IO8, \ OFF_SR_IO9, \ OFF_SR_IO10, \ OFF_SR_IO11, \ OFF_SR_IO12, \ OFF_SR_IO13, \ OFF_SR_IO14, \ OFF_SR_IO15, \ OFF_SR_IO16, \ OFF_SR_IO17 #define SD_NAMES \ MSC313_PINNAME_SD_CLK, \ MSC313_PINNAME_SD_CMD, \ MSC313_PINNAME_SD_D0, \ MSC313_PINNAME_SD_D1, \ MSC313_PINNAME_SD_D2, \ MSC313_PINNAME_SD_D3 #define OFF_SD_CLK 0x140 #define OFF_SD_CMD 0x144 #define OFF_SD_D0 0x148 #define OFF_SD_D1 0x14c #define OFF_SD_D2 0x150 #define OFF_SD_D3 0x154 #define SD_OFFSETS \ OFF_SD_CLK, \ OFF_SD_CMD, \ OFF_SD_D0, \ OFF_SD_D1, \ OFF_SD_D2, \ OFF_SD_D3 #define I2C1_NAMES \ MSC313_PINNAME_I2C1_SCL, \ MSC313_PINNAME_I2C1_SCA #define OFF_I2C1_SCL 0x188 #define OFF_I2C1_SCA 0x18c #define I2C1_OFFSETS \ OFF_I2C1_SCL, \ OFF_I2C1_SCA #define SPI0_NAMES \ MSC313_PINNAME_SPI0_CZ, \ MSC313_PINNAME_SPI0_CK, \ MSC313_PINNAME_SPI0_DI, \ MSC313_PINNAME_SPI0_DO #define OFF_SPI0_CZ 0x1c0 #define OFF_SPI0_CK 0x1c4 #define OFF_SPI0_DI 0x1c8 #define OFF_SPI0_DO 0x1cc #define SPI0_OFFSETS \ OFF_SPI0_CZ, \ OFF_SPI0_CK, \ OFF_SPI0_DI, \ OFF_SPI0_DO struct msc313_gpio_data { const char * const *names; const unsigned int *offsets; const unsigned int num; }; #define MSC313_GPIO_CHIPDATA(_chip) \ static const struct msc313_gpio_data _chip##_data = { \ .names = _chip##_names, \ .offsets = _chip##_offsets, \ .num = ARRAY_SIZE(_chip##_offsets), \ } #ifdef CONFIG_MACH_INFINITY static const char * const msc313_names[] = { FUART_NAMES, SR_NAMES, SD_NAMES, I2C1_NAMES, SPI0_NAMES, }; static const unsigned int msc313_offsets[] = { FUART_OFFSETS, SR_OFFSETS, SD_OFFSETS, I2C1_OFFSETS, SPI0_OFFSETS, }; MSC313_GPIO_CHIPDATA(msc313); #endif struct msc313_gpio { void __iomem *base; const struct msc313_gpio_data *gpio_data; u8 *saved; }; static void msc313_gpio_set(struct gpio_chip *chip, unsigned int offset, int value) { struct msc313_gpio *gpio = gpiochip_get_data(chip); u8 gpioreg = readb_relaxed(gpio->base + gpio->gpio_data->offsets[offset]); if (value) gpioreg |= MSC313_GPIO_OUT; else gpioreg &= ~MSC313_GPIO_OUT; writeb_relaxed(gpioreg, gpio->base + gpio->gpio_data->offsets[offset]); } static int msc313_gpio_get(struct gpio_chip *chip, unsigned int offset) { struct msc313_gpio *gpio = gpiochip_get_data(chip); return readb_relaxed(gpio->base + gpio->gpio_data->offsets[offset]) & MSC313_GPIO_IN; } static int msc313_gpio_direction_input(struct gpio_chip *chip, unsigned int offset) { struct msc313_gpio *gpio = gpiochip_get_data(chip); u8 gpioreg = readb_relaxed(gpio->base + gpio->gpio_data->offsets[offset]); gpioreg |= MSC313_GPIO_OEN; writeb_relaxed(gpioreg, gpio->base + gpio->gpio_data->offsets[offset]); return 0; } static int msc313_gpio_direction_output(struct gpio_chip *chip, unsigned int offset, int value) { struct msc313_gpio *gpio = gpiochip_get_data(chip); u8 gpioreg = readb_relaxed(gpio->base + gpio->gpio_data->offsets[offset]); gpioreg &= ~MSC313_GPIO_OEN; if (value) gpioreg |= MSC313_GPIO_OUT; else gpioreg &= ~MSC313_GPIO_OUT; writeb_relaxed(gpioreg, gpio->base + gpio->gpio_data->offsets[offset]); return 0; } /* * The interrupt handling happens in the parent interrupt controller, * we don't do anything here. */ static struct irq_chip msc313_gpio_irqchip = { .name = "GPIO", .irq_eoi = irq_chip_eoi_parent, .irq_mask = irq_chip_mask_parent, .irq_unmask = irq_chip_unmask_parent, .irq_set_type = irq_chip_set_type_parent, .irq_set_affinity = irq_chip_set_affinity_parent, }; /* * The parent interrupt controller needs the GIC interrupt type set to GIC_SPI * so we need to provide the fwspec. Essentially gpiochip_populate_parent_fwspec_twocell * that puts GIC_SPI into the first cell. */ static void *msc313_gpio_populate_parent_fwspec(struct gpio_chip *gc, unsigned int parent_hwirq, unsigned int parent_type) { struct irq_fwspec *fwspec; fwspec = kmalloc(sizeof(*fwspec), GFP_KERNEL); if (!fwspec) return NULL; fwspec->fwnode = gc->irq.parent_domain->fwnode; fwspec->param_count = 3; fwspec->param[0] = GIC_SPI; fwspec->param[1] = parent_hwirq; fwspec->param[2] = parent_type; return fwspec; } static int msc313e_gpio_child_to_parent_hwirq(struct gpio_chip *chip, unsigned int child, unsigned int child_type, unsigned int *parent, unsigned int *parent_type) { struct msc313_gpio *priv = gpiochip_get_data(chip); unsigned int offset = priv->gpio_data->offsets[child]; /* * only the spi0 pins have interrupts on the parent * on all of the known chips and so far they are all * mapped to the same place */ if (offset >= OFF_SPI0_CZ && offset <= OFF_SPI0_DO) { *parent_type = child_type; *parent = ((offset - OFF_SPI0_CZ) >> 2) + 28; return 0; } return -EINVAL; } static int msc313_gpio_probe(struct platform_device *pdev) { const struct msc313_gpio_data *match_data; struct msc313_gpio *gpio; struct gpio_chip *gpiochip; struct gpio_irq_chip *gpioirqchip; struct irq_domain *parent_domain; struct device_node *parent_node; struct device *dev = &pdev->dev; match_data = of_device_get_match_data(dev); if (!match_data) return -EINVAL; parent_node = of_irq_find_parent(dev->of_node); if (!parent_node) return -ENODEV; parent_domain = irq_find_host(parent_node); if (!parent_domain) return -ENODEV; gpio = devm_kzalloc(dev, sizeof(*gpio), GFP_KERNEL); if (!gpio) return -ENOMEM; gpio->gpio_data = match_data; gpio->saved = devm_kcalloc(dev, gpio->gpio_data->num, sizeof(*gpio->saved), GFP_KERNEL); if (!gpio->saved) return -ENOMEM; gpio->base = devm_platform_ioremap_resource(pdev, 0); if (IS_ERR(gpio->base)) return PTR_ERR(gpio->base); platform_set_drvdata(pdev, gpio); gpiochip = devm_kzalloc(dev, sizeof(*gpiochip), GFP_KERNEL); if (!gpiochip) return -ENOMEM; gpiochip->label = DRIVER_NAME; gpiochip->parent = dev; gpiochip->request = gpiochip_generic_request; gpiochip->free = gpiochip_generic_free; gpiochip->direction_input = msc313_gpio_direction_input; gpiochip->direction_output = msc313_gpio_direction_output; gpiochip->get = msc313_gpio_get; gpiochip->set = msc313_gpio_set; gpiochip->base = -1; gpiochip->ngpio = gpio->gpio_data->num; gpiochip->names = gpio->gpio_data->names; gpioirqchip = &gpiochip->irq; gpioirqchip->chip = &msc313_gpio_irqchip; gpioirqchip->fwnode = of_node_to_fwnode(dev->of_node); gpioirqchip->parent_domain = parent_domain; gpioirqchip->child_to_parent_hwirq = msc313e_gpio_child_to_parent_hwirq; gpioirqchip->populate_parent_alloc_arg = msc313_gpio_populate_parent_fwspec; gpioirqchip->handler = handle_bad_irq; gpioirqchip->default_type = IRQ_TYPE_NONE; return devm_gpiochip_add_data(dev, gpiochip, gpio); } static int msc313_gpio_remove(struct platform_device *pdev) { return 0; } static const struct of_device_id msc313_gpio_of_match[] = { #ifdef CONFIG_MACH_INFINITY { .compatible = "mstar,msc313-gpio", .data = &msc313_data, }, #endif { } }; /* * The GPIO controller loses the state of the registers when the * SoC goes into suspend to memory mode so we need to save some * of the register bits before suspending and put it back when resuming */ static int __maybe_unused msc313_gpio_suspend(struct device *dev) { struct msc313_gpio *gpio = dev_get_drvdata(dev); int i; for (i = 0; i < gpio->gpio_data->num; i++) gpio->saved[i] = readb_relaxed(gpio->base + gpio->gpio_data->offsets[i]) & MSC313_GPIO_BITSTOSAVE; return 0; } static int __maybe_unused msc313_gpio_resume(struct device *dev) { struct msc313_gpio *gpio = dev_get_drvdata(dev); int i; for (i = 0; i < gpio->gpio_data->num; i++) writeb_relaxed(gpio->saved[i], gpio->base + gpio->gpio_data->offsets[i]); return 0; } static SIMPLE_DEV_PM_OPS(msc313_gpio_ops, msc313_gpio_suspend, msc313_gpio_resume); static struct platform_driver msc313_gpio_driver = { .driver = { .name = DRIVER_NAME, .of_match_table = msc313_gpio_of_match, .pm = &msc313_gpio_ops, }, .probe = msc313_gpio_probe, .remove = msc313_gpio_remove, }; builtin_platform_driver(msc313_gpio_driver);