From patchwork Mon Mar 15 08:23:40 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Sander Vanheule X-Patchwork-Id: 1453083 Return-Path: X-Original-To: incoming@patchwork.ozlabs.org Delivered-To: patchwork-incoming@bilbo.ozlabs.org Authentication-Results: ozlabs.org; spf=pass (sender SPF authorized) smtp.mailfrom=vger.kernel.org (client-ip=23.128.96.18; helo=vger.kernel.org; envelope-from=linux-gpio-owner@vger.kernel.org; receiver=) Authentication-Results: ozlabs.org; dkim=pass (2048-bit key; secure) header.d=svanheule.net header.i=@svanheule.net header.a=rsa-sha256 header.s=mail1707 header.b=xq2c5TIw; dkim-atps=neutral Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by ozlabs.org (Postfix) with ESMTP id 4DzTzL03vvz9sWF for ; Mon, 15 Mar 2021 19:26:37 +1100 (AEDT) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230221AbhCOI0E (ORCPT ); Mon, 15 Mar 2021 04:26:04 -0400 Received: from polaris.svanheule.net ([84.16.241.116]:41490 "EHLO polaris.svanheule.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230205AbhCOIZr (ORCPT ); Mon, 15 Mar 2021 04:25:47 -0400 Received: from terra.local.svanheule.net (unknown [IPv6:2a02:a03f:eaff:9701:b4db:50a7:6f83:328f]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) (Authenticated sender: sander@svanheule.net) by polaris.svanheule.net (Postfix) with ESMTPSA id E02211DDFEA; Mon, 15 Mar 2021 09:25:44 +0100 (CET) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=svanheule.net; s=mail1707; t=1615796745; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=L5b8W1wvXvezpB7EzgV2HdjsVQ16QiHtehmNZ6HrL3s=; b=xq2c5TIwxyA1J4rynIh2JX4do9qWTrOMOuRx58PHIsambxodkAq/PuXwzZsOOYutcgSzLu fkRQ4lOHPqECu/51xH+PN0TuvUkrZJgZEV5IsHNiRRJEV364SAMxeTJYnIJPk4y8+My78C +7bdctW+uzILBDhrs+2780HhyzF4cIUna7BBiVNcip3gfzbIgs86SVVO5a9VddK/g6GxXT qb3rL24OG3ezDHVaQsE2FZDV0gEWUKVVI77QaDy3LYLCj3wbJ08vp6WPT7xT8RwWnPMiHd 6r/45hKkWZzJLhngLwR5hpSFa8p0PT69LBZhLhTMLYRpw/jJ+W7Bn/mJsw3Nuw== From: Sander Vanheule To: linux-gpio@vger.kernel.org, devicetree@vger.kernel.org Cc: Linus Walleij , Bartosz Golaszewski , Rob Herring , Thomas Gleixner , Marc Zyngier , linux-kernel@vger.kernel.org, Sander Vanheule , Bert Vermeulen Subject: [PATCH 1/2] dt-bindings: gpio: Binding for Realtek Otto GPIO Date: Mon, 15 Mar 2021 09:23:40 +0100 Message-Id: <20210315082339.9787-2-sander@svanheule.net> X-Mailer: git-send-email 2.30.2 In-Reply-To: <20210315082339.9787-1-sander@svanheule.net> References: <20210315082339.9787-1-sander@svanheule.net> MIME-Version: 1.0 Precedence: bulk List-ID: X-Mailing-List: linux-gpio@vger.kernel.org Add a binding description for Realtek's GPIO controller found on several of their MIPS-based SoCs (codenamed Otto), such as the RTL838x and RTL839x series of switch SoCs. A fallback binding 'realtek,otto-gpio' is provided for cases where the actual port ordering is not known yet, and enabling the interrupt controller may result in uncaught interrupts. Signed-off-by: Sander Vanheule Reviewed-by: Linus Walleij --- .../bindings/gpio/gpio-realtek-otto.yaml | 80 +++++++++++++++++++ 1 file changed, 80 insertions(+) create mode 100644 Documentation/devicetree/bindings/gpio/gpio-realtek-otto.yaml diff --git a/Documentation/devicetree/bindings/gpio/gpio-realtek-otto.yaml b/Documentation/devicetree/bindings/gpio/gpio-realtek-otto.yaml new file mode 100644 index 000000000000..3e8151e3a169 --- /dev/null +++ b/Documentation/devicetree/bindings/gpio/gpio-realtek-otto.yaml @@ -0,0 +1,80 @@ +# SPDX-License-Identifier: GPL-2.0-only OR BSD-2-Clause +%YAML 1.2 +--- +$id: http://devicetree.org/schemas/gpio/gpio-realtek-otto.yaml# +$schema: http://devicetree.org/meta-schemas/core.yaml# + +title: Realtek Otto GPIO controller + +maintainers: + - Sander Vanheule + - Bert Vermeulen + +description: | + Realtek's GPIO controller on their MIPS switch SoCs (Otto platform) consists + of two banks of 32 GPIOs. These GPIOs can generate edge-triggered interrupts. + Each bank's interrupts are cascased into one interrupt line on the parent + interrupt controller, if provided. + This binding allows defining a single bank in the devicetree. The interrupt + controller is not supported on the fallback compatible name, which only + allows for GPIO port use. + +properties: + $nodename: + pattern: "^gpio@[0-9a-f]+$" + + compatible: + oneOf: + - items: + - enum: + - realtek,rtl8380-gpio + - realtek,rtl8390-gpio + - const: realtek,otto-gpio + - const: realtek,otto-gpio + + reg: + maxItems: 1 + + "#gpio-cells": + const: 2 + + gpio-controller: true + + ngpios: + minimum: 1 + maximum: 32 + + interrupt-controller: true + + "#interrupt-cells": + const: 2 + + interrupts: + maxItems: 1 + +required: + - compatible + - reg + - "#gpio-cells" + - gpio-controller + +additionalProperties: false + +dependencies: + interrupt-controller: [ interrupts ] + +examples: + - | + gpio@3500 { + compatible = "realtek,rtl8380-gpio", "realtek,otto-gpio"; + reg = <0x3500 0x1c>; + gpio-controller; + #gpio-cells = <2>; + ngpios = <24>; + interrupt-controller; + #interrupt-cells = <2>; + interrupt-parent = <&rtlintc>; + interrupts = <23>; + }; + +... From patchwork Mon Mar 15 08:23:42 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Sander Vanheule X-Patchwork-Id: 1453084 Return-Path: X-Original-To: incoming@patchwork.ozlabs.org Delivered-To: patchwork-incoming@bilbo.ozlabs.org Authentication-Results: ozlabs.org; spf=pass (sender SPF authorized) smtp.mailfrom=vger.kernel.org (client-ip=23.128.96.18; helo=vger.kernel.org; envelope-from=linux-gpio-owner@vger.kernel.org; receiver=) Authentication-Results: ozlabs.org; dkim=pass (2048-bit key; secure) header.d=svanheule.net header.i=@svanheule.net header.a=rsa-sha256 header.s=mail1707 header.b=Ds+/W99T; dkim-atps=neutral Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by ozlabs.org (Postfix) with ESMTP id 4DzTzx56cmz9sWL for ; Mon, 15 Mar 2021 19:27:09 +1100 (AEDT) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230300AbhCOI0h (ORCPT ); Mon, 15 Mar 2021 04:26:37 -0400 Received: from polaris.svanheule.net ([84.16.241.116]:41520 "EHLO polaris.svanheule.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230171AbhCOI0a (ORCPT ); Mon, 15 Mar 2021 04:26:30 -0400 Received: from terra.local.svanheule.net (unknown [IPv6:2a02:a03f:eaff:9701:b4db:50a7:6f83:328f]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) (Authenticated sender: sander@svanheule.net) by polaris.svanheule.net (Postfix) with ESMTPSA id 2D4221DDFED; Mon, 15 Mar 2021 09:26:29 +0100 (CET) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=svanheule.net; s=mail1707; t=1615796789; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=yYz3MctYOx4C/yzcGmdSasNUSdDDesIXSSyXE/miHx0=; b=Ds+/W99THAxCq5rBnBf5u9ZUCkS22sljtkR1DDDNoVoYaF0v1xOGeVTXuG7eSNS5AJ/VPA wu2EM3vd3gqXMXOePDKENrqopJEBRs+i5rKsG1xU0wRqa3m3pXtl1Zw5CpA0oAt+BsnCzm 7m9w7pG9oVPfT79POHwOswyzwd6/iUn3w4oOOnQKO7QNV5eJ3pbDk0bko2J5LMK6hn6cEt 2GCBAxa29+6t2lapKwlZEuX3IJCIUo8YKPtCku+z3AWr2q6m6J9qjIuJF8LOwrregfM0k3 msJn7yf2YDlPIyKmdrC7PRtfOGSryT3j2+0Lz5Bcs1WVxeOgKhfEPYKxBc9c6A== From: Sander Vanheule To: linux-gpio@vger.kernel.org, devicetree@vger.kernel.org Cc: Linus Walleij , Bartosz Golaszewski , Rob Herring , Thomas Gleixner , Marc Zyngier , linux-kernel@vger.kernel.org, Sander Vanheule , Bert Vermeulen Subject: [PATCH 2/2] gpio: Add Realtek Otto GPIO support Date: Mon, 15 Mar 2021 09:23:42 +0100 Message-Id: <20210315082339.9787-3-sander@svanheule.net> X-Mailer: git-send-email 2.30.2 In-Reply-To: <20210315082339.9787-1-sander@svanheule.net> References: <20210315082339.9787-1-sander@svanheule.net> MIME-Version: 1.0 Precedence: bulk List-ID: X-Mailing-List: linux-gpio@vger.kernel.org Realtek MIPS SoCs (platform name Otto) have GPIO controllers with up to 64 GPIOs, divided over two banks. Each bank has a set of registers for 32 GPIOs, with support for edge-triggered interrupts. Each GPIO bank consists of four 8-bit GPIO ports (ABCD and EFGH). Most registers pack one bit per GPIO, except for the IMR register, which packs two bits per GPIO (AB-CD). Although the byte order is currently assumed to have port A..D at offset 0x0..0x3, this has been observed to be reversed on other, Lexra-based, SoCs (e.g. RTL8196E/97D/97F). Interrupt support is disabled for the fallback devicetree-compatible 'realtek,otto-gpio'. This allows for quick support of GPIO banks in which the byte order would be unknown. In this case, the port ordering in the IMR registers may not match the reversed order in the other registers (DCBA, and BA-DC or DC-BA). Signed-off-by: Sander Vanheule Reviewed-by: Linus Walleij --- drivers/gpio/Kconfig | 12 ++ drivers/gpio/Makefile | 1 + drivers/gpio/gpio-realtek-otto.c | 320 +++++++++++++++++++++++++++++++ 3 files changed, 333 insertions(+) create mode 100644 drivers/gpio/gpio-realtek-otto.c diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig index e3607ec4c2e8..fedf1e49469e 100644 --- a/drivers/gpio/Kconfig +++ b/drivers/gpio/Kconfig @@ -502,6 +502,18 @@ config GPIO_RDA help Say Y here to support RDA Micro GPIO controller. +config GPIO_REALTEK_OTTO + bool "Realtek Otto GPIO support" + depends on MACH_REALTEK_RTL + depends on OF_GPIO + default MACH_REALTEK_RTL + select GPIO_GENERIC + select GPIOLIB_IRQCHIP + help + The GPIO controller on the Otto MIPS platform supports up to two + banks of 32 GPIOs, with edge triggered interrupts. The 32 GPIOs + are grouped in four 8-bit wide ports. + config GPIO_REG bool help diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile index c58a90a3c3b1..8ace5934e3c3 100644 --- a/drivers/gpio/Makefile +++ b/drivers/gpio/Makefile @@ -124,6 +124,7 @@ obj-$(CONFIG_GPIO_RC5T583) += gpio-rc5t583.o obj-$(CONFIG_GPIO_RCAR) += gpio-rcar.o obj-$(CONFIG_GPIO_RDA) += gpio-rda.o obj-$(CONFIG_GPIO_RDC321X) += gpio-rdc321x.o +obj-$(CONFIG_GPIO_REALTEK_OTTO) += gpio-realtek-otto.o obj-$(CONFIG_GPIO_REG) += gpio-reg.o obj-$(CONFIG_ARCH_SA1100) += gpio-sa1100.o obj-$(CONFIG_GPIO_SAMA5D2_PIOBU) += gpio-sama5d2-piobu.o diff --git a/drivers/gpio/gpio-realtek-otto.c b/drivers/gpio/gpio-realtek-otto.c new file mode 100644 index 000000000000..04c11b2085f8 --- /dev/null +++ b/drivers/gpio/gpio-realtek-otto.c @@ -0,0 +1,320 @@ +// SPDX-License-Identifier: GPL-2.0-only + +#include +#include +#include +#include +#include +#include + +/* + * Total register block size is 0x1C for four ports. + * On the RTL8380/RLT8390 platforms port A, B, and C are implemented. + * RTL8389 and RTL8328 implement a second bank with ports E, F, G, and H. + * + * Port information is stored with the first port at offset 0, followed by the + * second, etc. Most registers store one bit per GPIO and should be read out in + * reversed endian order. The two interrupt mask registers store two bits per + * GPIO, and should be manipulated with swahw32, if required. + */ + +/* + * Pin select: (0) "normal", (1) "dedicate peripheral" + * Not used on RTL8380/RTL8390, peripheral selection is managed by control bits + * in the peripheral registers. + */ +#define REALTEK_GPIO_REG_CNR 0x00 +/* Clear bit (0) for input, set bit (1) for output */ +#define REALTEK_GPIO_REG_DIR 0x08 +#define REALTEK_GPIO_REG_DATA 0x0C +/* Read bit for IRQ status, write 1 to clear IRQ */ +#define REALTEK_GPIO_REG_ISR 0x10 +/* Two bits per GPIO */ +#define REALTEK_GPIO_REG_IMR_AB 0x14 +#define REALTEK_GPIO_REG_IMR_CD 0x18 +#define REALTEK_GPIO_IRQ_EDGE_FALLING 1 +#define REALTEK_GPIO_IRQ_EDGE_RISING 2 +#define REALTEK_GPIO_IRQ_EDGE_BOTH 3 + +#define REALTEK_GPIO_MAX 32 + +/* + * Realtek GPIO driver data + * Because the interrupt mask register (IMR) combines the function of + * IRQ type selection and masking, two extra values are stored. + * intr_mask is used to mask/unmask the interrupts for certain GPIO, + * and intr_type is used to store the selected interrupt types. The + * logical AND of these values is written to IMR on changes. + * + * @dev Parent device + * @gc Associated gpio_chip instance + * @base Base address of the register block + * @lock Lock for accessing the IRQ registers and values + * @intr_mask Mask for GPIO interrupts + * @intr_type GPIO interrupt type selection + */ +struct realtek_gpio_ctrl { + struct device *dev; + struct gpio_chip gc; + void __iomem *base; + raw_spinlock_t lock; + u32 intr_mask[2]; + u32 intr_type[2]; +}; + +enum realtek_gpio_flags { + GPIO_INTERRUPTS = BIT(0), +}; + +static struct realtek_gpio_ctrl *irq_data_to_ctrl(struct irq_data *data) +{ + struct gpio_chip *gc = irq_data_get_irq_chip_data(data); + + return container_of(gc, struct realtek_gpio_ctrl, gc); +} + +static inline u32 realtek_gpio_isr_read(struct realtek_gpio_ctrl *ctrl) +{ + return swab32(readl(ctrl->base + REALTEK_GPIO_REG_ISR)); +} + +static inline void realtek_gpio_isr_clear(struct realtek_gpio_ctrl *ctrl, + unsigned int pin_mask) +{ + writel(swab32(pin_mask), ctrl->base + REALTEK_GPIO_REG_ISR); +} + +static inline void realtek_update_imr(struct realtek_gpio_ctrl *ctrl, + unsigned int offset, u32 type, u32 mask) +{ + unsigned int reg; + + if (offset == 0) + reg = REALTEK_GPIO_REG_IMR_AB; + else + reg = REALTEK_GPIO_REG_IMR_CD; + writel(swahw32(type & mask), ctrl->base + reg); +} + +static inline u32 realtek_gpio_imr_bits(unsigned int pin, u32 value) +{ + return ((value & 0x3) << 2*(pin % 16)); +} + +static void realtek_gpio_irq_ack(struct irq_data *data) +{ + struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data); + u32 pin = irqd_to_hwirq(data); + + realtek_gpio_isr_clear(ctrl, BIT(pin)); +} + +static void realtek_gpio_irq_unmask(struct irq_data *data) +{ + struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data); + unsigned int pin = irqd_to_hwirq(data); + unsigned int offset = pin/16; + unsigned long flags; + u32 m; + + raw_spin_lock_irqsave(&ctrl->lock, flags); + m = ctrl->intr_mask[offset]; + m |= realtek_gpio_imr_bits(pin, 3); + ctrl->intr_mask[offset] = m; + realtek_update_imr(ctrl, offset, ctrl->intr_type[offset], m); + raw_spin_unlock_irqrestore(&ctrl->lock, flags); +} + +static void realtek_gpio_irq_mask(struct irq_data *data) +{ + struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data); + unsigned int pin = irqd_to_hwirq(data); + unsigned int offset = pin/16; + unsigned long flags; + u32 m; + + raw_spin_lock_irqsave(&ctrl->lock, flags); + m = ctrl->intr_mask[offset]; + m &= ~realtek_gpio_imr_bits(pin, 3); + ctrl->intr_mask[offset] = m; + realtek_update_imr(ctrl, offset, ctrl->intr_type[offset], m); + raw_spin_unlock_irqrestore(&ctrl->lock, flags); +} + +static int realtek_gpio_irq_set_type(struct irq_data *data, + unsigned int flow_type) +{ + struct realtek_gpio_ctrl *ctrl = irq_data_to_ctrl(data); + irq_flow_handler_t handler; + unsigned int pin = irqd_to_hwirq(data); + unsigned int offset = pin/16; + unsigned long flags; + u32 type, t; + + switch (flow_type & IRQ_TYPE_SENSE_MASK) { + case IRQ_TYPE_NONE: + type = 0; + handler = handle_bad_irq; + break; + case IRQ_TYPE_EDGE_FALLING: + type = REALTEK_GPIO_IRQ_EDGE_FALLING; + handler = handle_edge_irq; + break; + case IRQ_TYPE_EDGE_RISING: + type = REALTEK_GPIO_IRQ_EDGE_RISING; + handler = handle_edge_irq; + break; + case IRQ_TYPE_EDGE_BOTH: + type = REALTEK_GPIO_IRQ_EDGE_BOTH; + handler = handle_edge_irq; + break; + default: + return -EINVAL; + } + + irq_set_handler_locked(data, handler); + + raw_spin_lock_irqsave(&ctrl->lock, flags); + t = ctrl->intr_type[offset]; + t &= ~realtek_gpio_imr_bits(pin, 3); + t |= realtek_gpio_imr_bits(pin, type); + ctrl->intr_type[offset] = t; + realtek_update_imr(ctrl, offset, t, ctrl->intr_mask[offset]); + raw_spin_unlock_irqrestore(&ctrl->lock, flags); + + return 0; +} + +static void realtek_gpio_irq_handler(struct irq_desc *desc) +{ + struct gpio_chip *gc = irq_desc_get_handler_data(desc); + struct realtek_gpio_ctrl *ctrl = gpiochip_get_data(gc); + struct irq_chip *irq_chip = irq_desc_get_chip(desc); + int offset; + unsigned long status; + + chained_irq_enter(irq_chip, desc); + + status = realtek_gpio_isr_read(ctrl); + for_each_set_bit(offset, &status, gc->ngpio) { + dev_dbg(ctrl->dev, "gpio irq %d\n", offset); + generic_handle_irq(irq_find_mapping(gc->irq.domain, + offset)); + realtek_gpio_isr_clear(ctrl, BIT(offset)); + } + + chained_irq_exit(irq_chip, desc); +} + +static struct irq_chip realtek_gpio_irq_chip = { + .name = "realtek-otto-gpio", + .irq_ack = realtek_gpio_irq_ack, + .irq_mask = realtek_gpio_irq_mask, + .irq_unmask = realtek_gpio_irq_unmask, + .irq_set_type = realtek_gpio_irq_set_type +}; + +static const struct of_device_id realtek_gpio_of_match[] = { + { .compatible = "realtek,otto-gpio" }, + { + .compatible = "realtek,rtl8380-gpio", + .data = (void *)GPIO_INTERRUPTS + }, + { + .compatible = "realtek,rtl8390-gpio", + .data = (void *)GPIO_INTERRUPTS + }, + {}, +}; + +MODULE_DEVICE_TABLE(of, realtek_gpio_of_match); + +static int realtek_gpio_probe(struct platform_device *pdev) +{ + struct device *dev = &pdev->dev; + struct device_node *np = dev->of_node; + struct gpio_irq_chip *girq; + struct realtek_gpio_ctrl *ctrl; + const struct of_device_id *match; + unsigned int match_flags; + u32 ngpios; + int err, irq; + + ctrl = devm_kzalloc(dev, sizeof(*ctrl), GFP_KERNEL); + if (!ctrl) + return -ENOMEM; + + ctrl->dev = dev; + + if (!np) { + dev_err(&pdev->dev, "no DT node found\n"); + return -EINVAL; + } + + match = of_match_node(realtek_gpio_of_match, np); + match_flags = (unsigned int) match->data; + + if (of_property_read_u32(np, "ngpios", &ngpios) != 0) + ngpios = REALTEK_GPIO_MAX; + + if (ngpios > REALTEK_GPIO_MAX) { + dev_err(&pdev->dev, "invalid ngpios (max. %d)\n", + REALTEK_GPIO_MAX); + return -EINVAL; + } + + ctrl->base = devm_platform_ioremap_resource(pdev, 0); + if (IS_ERR(ctrl->base)) + return PTR_ERR(ctrl->base); + + raw_spin_lock_init(&ctrl->lock); + + err = bgpio_init(&ctrl->gc, dev, 4, + ctrl->base + REALTEK_GPIO_REG_DATA, NULL, NULL, + ctrl->base + REALTEK_GPIO_REG_DIR, NULL, + BGPIOF_BIG_ENDIAN_BYTE_ORDER); + if (err) { + dev_err(dev, "unable to init generic GPIO"); + return err; + } + + ctrl->gc.ngpio = ngpios; + ctrl->gc.owner = THIS_MODULE; + + irq = of_irq_get(np, 0); + if ((match_flags & GPIO_INTERRUPTS) && irq > 0) { + girq = &ctrl->gc.irq; + girq->chip = &realtek_gpio_irq_chip; + girq->parent_handler = realtek_gpio_irq_handler; + girq->num_parents = 1; + girq->parents = devm_kcalloc(dev, 1, sizeof(*girq->parents), + GFP_KERNEL); + if (!girq->parents) + return -ENOMEM; + girq->default_type = IRQ_TYPE_NONE; + girq->handler = handle_bad_irq; + girq->parents[0] = irq; + + /* Disable and clear all interrupts */ + realtek_update_imr(ctrl, 0, 0, 0); + realtek_update_imr(ctrl, 1, 0, 0); + realtek_gpio_isr_clear(ctrl, BIT(ngpios)-1); + } + + err = gpiochip_add_data(&ctrl->gc, ctrl); + return err; +} + +static struct platform_driver realtek_gpio_driver = { + .driver = { + .name = "realtek-otto-gpio", + .of_match_table = realtek_gpio_of_match, + }, + .probe = realtek_gpio_probe, +}; + +builtin_platform_driver(realtek_gpio_driver); + +MODULE_DESCRIPTION("Realtek Otto GPIO support"); +MODULE_AUTHOR("Sander Vanheule "); +MODULE_LICENSE("GPL v2");