Received: by 2002:ad5:474a:0:0:0:0:0 with SMTP id i10csp2610047imu; Sun, 13 Jan 2019 06:04:16 -0800 (PST) X-Google-Smtp-Source: ALg8bN5rnpj8sesB21bTlROzCtCQDuo1T+e4r4uXJxGUkUovP2QRF/0fLs0ASQ2ELkBRrQab/jYS X-Received: by 2002:a62:c302:: with SMTP id v2mr22030003pfg.155.1547388256701; Sun, 13 Jan 2019 06:04:16 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1547388256; cv=none; d=google.com; s=arc-20160816; b=zYA03eXwEST1vNIf+nuKrjjsNMJlkY3BRJDs3qo4U4ODGG9pGEh+jzxT0vfbQX38JK DYYEA5Yz/7xteAig0fXumc26fOl9n2HV97rauzgRg1RlyWyesNWwYAKvPok/7PhN1T28 R1QkrHn8xEPSw3pMfNrayXsteoQG0xz5vk047EUczp8o0EoAnArFRNG9q7PL1OJGqzAz QsdtS6r+tO6mMDOl5Iz9AuPevN1oNeGiGE5C/qND0TrbRrUUgnB2hKES410/FgTDHIPa N6kvlpOcZjZazfbRdH4yNt1H35k9sW+JbSnwB+FhCx3bnSH3LZj0B4XmTob/7lwEh2Da vkrw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from; bh=Dry1UCwDPbvmTXUQkBFA09VPvsdAhKoeatPzJRgJBOI=; b=HDMSmDFAxHPXZr54CRESzDwWCmwoc8ohjhXYubQYorsBCQv2S1ExKAf0lOJZ7ztZuY jzHkxwq43q98bpEbvXBp9MvFGawkeCDS2pcoVXYWfyiOgpKcLB38lpLF9kML2YRaEiNM HMYn0evo2nB0z5vDQt0Qo3OzbvQIsV8CYoWOA3wV5yH1/yeTKRUNP0tplQYJUCcpXIfF p0Is++1PHkpJjCQB6dbDGbaDzC8vWjt0rANZwEHYVVopcWkcS2g3wsZPUcEu8lmZM1ca 7GvDkO/KlZdFfMNG4gVNnkoiRNCuqD6pnnCzHPlhzcMXyN6Kixfu7SriEbq4yYKukHwj YzfQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id f12si56550747pgd.68.2019.01.13.06.03.29; Sun, 13 Jan 2019 06:04:16 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726664AbfAMN75 (ORCPT + 99 others); Sun, 13 Jan 2019 08:59:57 -0500 Received: from mout.gmx.net ([212.227.15.18]:58127 "EHLO mout.gmx.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726592AbfAMN7z (ORCPT ); Sun, 13 Jan 2019 08:59:55 -0500 Received: from longitude ([109.90.232.48]) by mail.gmx.com (mrgmx002 [212.227.17.190]) with ESMTPSA (Nemesis) id 0LjqLx-1hFYHI1BpB-00brUh; Sun, 13 Jan 2019 14:59:53 +0100 From: =?UTF-8?q?Jonathan=20Neusch=C3=A4fer?= To: linux-gpio@vger.kernel.org Cc: Linus Walleij , Bartosz Golaszewski , linux-kernel@vger.kernel.org, =?UTF-8?q?Jonathan=20Neusch=C3=A4fer?= Subject: [PATCH 2/2] gpio: hlwd: Implement edge trigger emulation Date: Sun, 13 Jan 2019 14:58:44 +0100 Message-Id: <20190113135844.13109-3-j.neuschaefer@gmx.net> X-Mailer: git-send-email 2.20.1 In-Reply-To: <20190113135844.13109-1-j.neuschaefer@gmx.net> References: <20190113135844.13109-1-j.neuschaefer@gmx.net> MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit X-Provags-ID: V03:K1:1EhH58A94aX1Skvgwv/M6Q4NdPL3ZMkYIVE1ofZyM7LmYs378Ry XBb8O+/Vnirf0rkseaXW0tbeXjTB/RtQ6ExAk5IyM72nfFN9SqnnuJYONbI00Zin1f1Tg2K gRGPxZyJ93EFAgjgGWMJbQeipuJspMycVtsQlYlek0x+BSha3QpFYoHxv9fYtPdxnfRHIW9 MK/OKQlkFWL48g79zAJVg== X-Spam-Flag: NO X-UI-Out-Filterresults: notjunk:1;V03:K0:QcFX0cIxZUs=:VCe7UW4P1yCILk5AccNlKh xWua8MtIS5hkdaF6UYVt4rX6d7n1n2rfDQIry3k8m9SXOywe+sNyd7gAAeXeKziSpdD13HG7Y MDjrlc1QTudZnVFoKEfXgi2RpeoiYpWD/EyLETXdo5/f183q2lrwhUlNYGLDojJYIA+EsxjQu /L0xfuovJzQLYPD8vnb7vR5ayd1cS8qGQUQ0COFVFVFcYksrKviofvENuwBqVUHWR78TvfwOV X3IFrq3rC9elVE9Ljc9g8xPm7YnGV+dfFvMltG8Gk7Zops9ARs+mV4m2TULJmzYr+XyihAVb7 ev6dTB6L/LWAdFS+RxgyLMq/re8ZiuuPo7ruF6E7cQbXjZs8We9cOBSEVv9ujNh8rSPoSJW/f q+vTm/gRURaHul4uapglFc0UF7DpRa0cnYzdf0wO0EKN3rQdiwladvXpyWbRMJUPBe71ZSDRh l5qO9aIFdpernspHwSpX5dopzKOy+vPu4ZqhbuRTAMz4T7hHnVczdqnsxiNsMI98+Z50WSKZG nx0InafHVoX11uCbEHvn4KUSJMbF+kbeFptAVAwU1yUxNf6rMEf8OzkOdtKTHGiWtrAQ943n6 mDmRzJzRgiaJipewSqUsJta0y5qIPMEeXfs1oOjucXAmk6Gagoa+cAHFffcX1gN0oWPd8/giS H1YzAq1d/+C5DaS3hLCxiDyJaXyCeuVyKElSDUoguXgkNGXX9LGO27yKTuDFo5jKdt6PErLWX GhDMfGpL54bBbl64W7k3pq9tdonUgsepDRAeoyC9LBsvA706rKv1NkN8i8KnULz/O4ApErzM6 uC6pjpiC6gu5LGlQj67tytO9x+sz6AqOuAMYQeYpZii99fCd2Ra1OzHBjUXgnQldZ2uFEhOiI cSUlOID/cQujU2lwu6WDMyzQ1YiQZJXaQgxPSliUakhL5+bArN6b4snlhWbzhq004CR4Dcndi QUfebhVZZig== Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Like the Spreadtrum EIC driver[1], this driver needs to emulate edge triggered interrupts to support the generic gpio-keys driver. [1]: https://www.spinics.net/lists/kernel/msg2764576.html Signed-off-by: Jonathan Neuschäfer --- drivers/gpio/gpio-hlwd.c | 57 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 57 insertions(+) diff --git a/drivers/gpio/gpio-hlwd.c b/drivers/gpio/gpio-hlwd.c index 9a93546f3d11..3d85e6a71b35 100644 --- a/drivers/gpio/gpio-hlwd.c +++ b/drivers/gpio/gpio-hlwd.c @@ -51,6 +51,8 @@ struct hlwd_gpio { struct irq_chip irqc; void __iomem *regs; int irq; + u32 edge_emulation; + u32 rising_edge, falling_edge; }; static void hlwd_gpio_irqhandler(struct irq_desc *desc) @@ -61,10 +63,37 @@ static void hlwd_gpio_irqhandler(struct irq_desc *desc) unsigned long flags; unsigned long pending; int hwirq; + u32 emulated_pending; spin_lock_irqsave(&hlwd->gpioc.bgpio_lock, flags); pending = hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_INTFLAG); pending &= hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_INTMASK); + + /* Treat interrupts due to edge trigger emulation separately */ + emulated_pending = hlwd->edge_emulation & pending; + pending &= ~emulated_pending; + if (emulated_pending) { + u32 level, rising, falling; + + level = hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_INTLVL); + rising = level & emulated_pending; + falling = ~level & emulated_pending; + + /* Invert the levels */ + hlwd->gpioc.write_reg(hlwd->regs + HW_GPIOB_INTLVL, + level ^ emulated_pending); + + /* Ack all emulated-edge interrupts */ + hlwd->gpioc.write_reg(hlwd->regs + HW_GPIOB_INTFLAG, + emulated_pending); + + /* Signal interrupts only on the correct edge */ + rising &= hlwd->rising_edge; + falling &= hlwd->falling_edge; + + /* Mark emulated interrupts as pending */ + pending |= rising | falling; + } spin_unlock_irqrestore(&hlwd->gpioc.bgpio_lock, flags); chained_irq_enter(chip, desc); @@ -120,6 +149,27 @@ static void hlwd_gpio_irq_enable(struct irq_data *data) hlwd_gpio_irq_unmask(data); } +static void hlwd_gpio_irq_setup_emulation(struct hlwd_gpio *hlwd, int hwirq, + unsigned int flow_type) +{ + u32 level, state; + + /* Set the trigger level to the inactive level */ + level = hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_INTLVL); + state = hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_IN) & BIT(hwirq); + level &= ~BIT(hwirq); + level |= state ^ BIT(hwirq); + hlwd->gpioc.write_reg(hlwd->regs + HW_GPIOB_INTLVL, level); + + hlwd->edge_emulation |= BIT(hwirq); + hlwd->rising_edge &= ~BIT(hwirq); + hlwd->falling_edge &= ~BIT(hwirq); + if (flow_type & IRQ_TYPE_EDGE_RISING) + hlwd->rising_edge |= BIT(hwirq); + if (flow_type & IRQ_TYPE_EDGE_FALLING) + hlwd->falling_edge |= BIT(hwirq); +} + static int hlwd_gpio_irq_set_type(struct irq_data *data, unsigned int flow_type) { struct hlwd_gpio *hlwd = @@ -129,6 +179,8 @@ static int hlwd_gpio_irq_set_type(struct irq_data *data, unsigned int flow_type) spin_lock_irqsave(&hlwd->gpioc.bgpio_lock, flags); + hlwd->edge_emulation &= ~BIT(data->hwirq); + switch (flow_type) { case IRQ_TYPE_LEVEL_HIGH: level = hlwd->gpioc.read_reg(hlwd->regs + HW_GPIOB_INTLVL); @@ -140,6 +192,11 @@ static int hlwd_gpio_irq_set_type(struct irq_data *data, unsigned int flow_type) level &= ~BIT(data->hwirq); hlwd->gpioc.write_reg(hlwd->regs + HW_GPIOB_INTLVL, level); break; + case IRQ_TYPE_EDGE_RISING: + case IRQ_TYPE_EDGE_FALLING: + case IRQ_TYPE_EDGE_BOTH: + hlwd_gpio_irq_setup_emulation(hlwd, data->hwirq, flow_type); + break; default: spin_unlock_irqrestore(&hlwd->gpioc.bgpio_lock, flags); return -EINVAL; -- 2.20.1