162306a36Sopenharmony_ci/*
262306a36Sopenharmony_ci * Allwinner A20/A31 SoCs NMI IRQ chip driver.
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * Carlo Caione <carlo.caione@gmail.com>
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci * This file is licensed under the terms of the GNU General Public
762306a36Sopenharmony_ci * License version 2.  This program is licensed "as is" without any
862306a36Sopenharmony_ci * warranty of any kind, whether express or implied.
962306a36Sopenharmony_ci */
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#define DRV_NAME	"sunxi-nmi"
1262306a36Sopenharmony_ci#define pr_fmt(fmt)	DRV_NAME ": " fmt
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci#include <linux/bitops.h>
1562306a36Sopenharmony_ci#include <linux/device.h>
1662306a36Sopenharmony_ci#include <linux/io.h>
1762306a36Sopenharmony_ci#include <linux/irq.h>
1862306a36Sopenharmony_ci#include <linux/interrupt.h>
1962306a36Sopenharmony_ci#include <linux/irqdomain.h>
2062306a36Sopenharmony_ci#include <linux/of_irq.h>
2162306a36Sopenharmony_ci#include <linux/of_address.h>
2262306a36Sopenharmony_ci#include <linux/irqchip.h>
2362306a36Sopenharmony_ci#include <linux/irqchip/chained_irq.h>
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci#define SUNXI_NMI_SRC_TYPE_MASK	0x00000003
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci#define SUNXI_NMI_IRQ_BIT	BIT(0)
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_ci/*
3062306a36Sopenharmony_ci * For deprecated sun6i-a31-sc-nmi compatible.
3162306a36Sopenharmony_ci */
3262306a36Sopenharmony_ci#define SUN6I_NMI_CTRL		0x00
3362306a36Sopenharmony_ci#define SUN6I_NMI_PENDING	0x04
3462306a36Sopenharmony_ci#define SUN6I_NMI_ENABLE	0x34
3562306a36Sopenharmony_ci
3662306a36Sopenharmony_ci#define SUN7I_NMI_CTRL		0x00
3762306a36Sopenharmony_ci#define SUN7I_NMI_PENDING	0x04
3862306a36Sopenharmony_ci#define SUN7I_NMI_ENABLE	0x08
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci#define SUN9I_NMI_CTRL		0x00
4162306a36Sopenharmony_ci#define SUN9I_NMI_ENABLE	0x04
4262306a36Sopenharmony_ci#define SUN9I_NMI_PENDING	0x08
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_cienum {
4562306a36Sopenharmony_ci	SUNXI_SRC_TYPE_LEVEL_LOW = 0,
4662306a36Sopenharmony_ci	SUNXI_SRC_TYPE_EDGE_FALLING,
4762306a36Sopenharmony_ci	SUNXI_SRC_TYPE_LEVEL_HIGH,
4862306a36Sopenharmony_ci	SUNXI_SRC_TYPE_EDGE_RISING,
4962306a36Sopenharmony_ci};
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_cistruct sunxi_sc_nmi_reg_offs {
5262306a36Sopenharmony_ci	u32 ctrl;
5362306a36Sopenharmony_ci	u32 pend;
5462306a36Sopenharmony_ci	u32 enable;
5562306a36Sopenharmony_ci};
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_cistatic const struct sunxi_sc_nmi_reg_offs sun6i_reg_offs __initconst = {
5862306a36Sopenharmony_ci	.ctrl	= SUN6I_NMI_CTRL,
5962306a36Sopenharmony_ci	.pend	= SUN6I_NMI_PENDING,
6062306a36Sopenharmony_ci	.enable	= SUN6I_NMI_ENABLE,
6162306a36Sopenharmony_ci};
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_cistatic const struct sunxi_sc_nmi_reg_offs sun7i_reg_offs __initconst = {
6462306a36Sopenharmony_ci	.ctrl	= SUN7I_NMI_CTRL,
6562306a36Sopenharmony_ci	.pend	= SUN7I_NMI_PENDING,
6662306a36Sopenharmony_ci	.enable	= SUN7I_NMI_ENABLE,
6762306a36Sopenharmony_ci};
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_cistatic const struct sunxi_sc_nmi_reg_offs sun9i_reg_offs __initconst = {
7062306a36Sopenharmony_ci	.ctrl	= SUN9I_NMI_CTRL,
7162306a36Sopenharmony_ci	.pend	= SUN9I_NMI_PENDING,
7262306a36Sopenharmony_ci	.enable	= SUN9I_NMI_ENABLE,
7362306a36Sopenharmony_ci};
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_cistatic inline void sunxi_sc_nmi_write(struct irq_chip_generic *gc, u32 off,
7662306a36Sopenharmony_ci				      u32 val)
7762306a36Sopenharmony_ci{
7862306a36Sopenharmony_ci	irq_reg_writel(gc, val, off);
7962306a36Sopenharmony_ci}
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_cistatic inline u32 sunxi_sc_nmi_read(struct irq_chip_generic *gc, u32 off)
8262306a36Sopenharmony_ci{
8362306a36Sopenharmony_ci	return irq_reg_readl(gc, off);
8462306a36Sopenharmony_ci}
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_cistatic void sunxi_sc_nmi_handle_irq(struct irq_desc *desc)
8762306a36Sopenharmony_ci{
8862306a36Sopenharmony_ci	struct irq_domain *domain = irq_desc_get_handler_data(desc);
8962306a36Sopenharmony_ci	struct irq_chip *chip = irq_desc_get_chip(desc);
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_ci	chained_irq_enter(chip, desc);
9262306a36Sopenharmony_ci	generic_handle_domain_irq(domain, 0);
9362306a36Sopenharmony_ci	chained_irq_exit(chip, desc);
9462306a36Sopenharmony_ci}
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_cistatic int sunxi_sc_nmi_set_type(struct irq_data *data, unsigned int flow_type)
9762306a36Sopenharmony_ci{
9862306a36Sopenharmony_ci	struct irq_chip_generic *gc = irq_data_get_irq_chip_data(data);
9962306a36Sopenharmony_ci	struct irq_chip_type *ct = gc->chip_types;
10062306a36Sopenharmony_ci	u32 src_type_reg;
10162306a36Sopenharmony_ci	u32 ctrl_off = ct->regs.type;
10262306a36Sopenharmony_ci	unsigned int src_type;
10362306a36Sopenharmony_ci	unsigned int i;
10462306a36Sopenharmony_ci
10562306a36Sopenharmony_ci	irq_gc_lock(gc);
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	switch (flow_type & IRQF_TRIGGER_MASK) {
10862306a36Sopenharmony_ci	case IRQ_TYPE_EDGE_FALLING:
10962306a36Sopenharmony_ci		src_type = SUNXI_SRC_TYPE_EDGE_FALLING;
11062306a36Sopenharmony_ci		break;
11162306a36Sopenharmony_ci	case IRQ_TYPE_EDGE_RISING:
11262306a36Sopenharmony_ci		src_type = SUNXI_SRC_TYPE_EDGE_RISING;
11362306a36Sopenharmony_ci		break;
11462306a36Sopenharmony_ci	case IRQ_TYPE_LEVEL_HIGH:
11562306a36Sopenharmony_ci		src_type = SUNXI_SRC_TYPE_LEVEL_HIGH;
11662306a36Sopenharmony_ci		break;
11762306a36Sopenharmony_ci	case IRQ_TYPE_NONE:
11862306a36Sopenharmony_ci	case IRQ_TYPE_LEVEL_LOW:
11962306a36Sopenharmony_ci		src_type = SUNXI_SRC_TYPE_LEVEL_LOW;
12062306a36Sopenharmony_ci		break;
12162306a36Sopenharmony_ci	default:
12262306a36Sopenharmony_ci		irq_gc_unlock(gc);
12362306a36Sopenharmony_ci		pr_err("Cannot assign multiple trigger modes to IRQ %d.\n",
12462306a36Sopenharmony_ci			data->irq);
12562306a36Sopenharmony_ci		return -EBADR;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	irqd_set_trigger_type(data, flow_type);
12962306a36Sopenharmony_ci	irq_setup_alt_chip(data, flow_type);
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	for (i = 0; i < gc->num_ct; i++, ct++)
13262306a36Sopenharmony_ci		if (ct->type & flow_type)
13362306a36Sopenharmony_ci			ctrl_off = ct->regs.type;
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci	src_type_reg = sunxi_sc_nmi_read(gc, ctrl_off);
13662306a36Sopenharmony_ci	src_type_reg &= ~SUNXI_NMI_SRC_TYPE_MASK;
13762306a36Sopenharmony_ci	src_type_reg |= src_type;
13862306a36Sopenharmony_ci	sunxi_sc_nmi_write(gc, ctrl_off, src_type_reg);
13962306a36Sopenharmony_ci
14062306a36Sopenharmony_ci	irq_gc_unlock(gc);
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_ci	return IRQ_SET_MASK_OK;
14362306a36Sopenharmony_ci}
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_cistatic int __init sunxi_sc_nmi_irq_init(struct device_node *node,
14662306a36Sopenharmony_ci					const struct sunxi_sc_nmi_reg_offs *reg_offs)
14762306a36Sopenharmony_ci{
14862306a36Sopenharmony_ci	struct irq_domain *domain;
14962306a36Sopenharmony_ci	struct irq_chip_generic *gc;
15062306a36Sopenharmony_ci	unsigned int irq;
15162306a36Sopenharmony_ci	unsigned int clr = IRQ_NOREQUEST | IRQ_NOPROBE | IRQ_NOAUTOEN;
15262306a36Sopenharmony_ci	int ret;
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_ci
15562306a36Sopenharmony_ci	domain = irq_domain_add_linear(node, 1, &irq_generic_chip_ops, NULL);
15662306a36Sopenharmony_ci	if (!domain) {
15762306a36Sopenharmony_ci		pr_err("Could not register interrupt domain.\n");
15862306a36Sopenharmony_ci		return -ENOMEM;
15962306a36Sopenharmony_ci	}
16062306a36Sopenharmony_ci
16162306a36Sopenharmony_ci	ret = irq_alloc_domain_generic_chips(domain, 1, 2, DRV_NAME,
16262306a36Sopenharmony_ci					     handle_fasteoi_irq, clr, 0,
16362306a36Sopenharmony_ci					     IRQ_GC_INIT_MASK_CACHE);
16462306a36Sopenharmony_ci	if (ret) {
16562306a36Sopenharmony_ci		pr_err("Could not allocate generic interrupt chip.\n");
16662306a36Sopenharmony_ci		goto fail_irqd_remove;
16762306a36Sopenharmony_ci	}
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ci	irq = irq_of_parse_and_map(node, 0);
17062306a36Sopenharmony_ci	if (irq <= 0) {
17162306a36Sopenharmony_ci		pr_err("unable to parse irq\n");
17262306a36Sopenharmony_ci		ret = -EINVAL;
17362306a36Sopenharmony_ci		goto fail_irqd_remove;
17462306a36Sopenharmony_ci	}
17562306a36Sopenharmony_ci
17662306a36Sopenharmony_ci	gc = irq_get_domain_generic_chip(domain, 0);
17762306a36Sopenharmony_ci	gc->reg_base = of_io_request_and_map(node, 0, of_node_full_name(node));
17862306a36Sopenharmony_ci	if (IS_ERR(gc->reg_base)) {
17962306a36Sopenharmony_ci		pr_err("unable to map resource\n");
18062306a36Sopenharmony_ci		ret = PTR_ERR(gc->reg_base);
18162306a36Sopenharmony_ci		goto fail_irqd_remove;
18262306a36Sopenharmony_ci	}
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci	gc->chip_types[0].type			= IRQ_TYPE_LEVEL_MASK;
18562306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_mask		= irq_gc_mask_clr_bit;
18662306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_unmask	= irq_gc_mask_set_bit;
18762306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_eoi		= irq_gc_ack_set_bit;
18862306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_set_type	= sunxi_sc_nmi_set_type;
18962306a36Sopenharmony_ci	gc->chip_types[0].chip.flags		= IRQCHIP_EOI_THREADED | IRQCHIP_EOI_IF_HANDLED;
19062306a36Sopenharmony_ci	gc->chip_types[0].regs.ack		= reg_offs->pend;
19162306a36Sopenharmony_ci	gc->chip_types[0].regs.mask		= reg_offs->enable;
19262306a36Sopenharmony_ci	gc->chip_types[0].regs.type		= reg_offs->ctrl;
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_ci	gc->chip_types[1].type			= IRQ_TYPE_EDGE_BOTH;
19562306a36Sopenharmony_ci	gc->chip_types[1].chip.name		= gc->chip_types[0].chip.name;
19662306a36Sopenharmony_ci	gc->chip_types[1].chip.irq_ack		= irq_gc_ack_set_bit;
19762306a36Sopenharmony_ci	gc->chip_types[1].chip.irq_mask		= irq_gc_mask_clr_bit;
19862306a36Sopenharmony_ci	gc->chip_types[1].chip.irq_unmask	= irq_gc_mask_set_bit;
19962306a36Sopenharmony_ci	gc->chip_types[1].chip.irq_set_type	= sunxi_sc_nmi_set_type;
20062306a36Sopenharmony_ci	gc->chip_types[1].regs.ack		= reg_offs->pend;
20162306a36Sopenharmony_ci	gc->chip_types[1].regs.mask		= reg_offs->enable;
20262306a36Sopenharmony_ci	gc->chip_types[1].regs.type		= reg_offs->ctrl;
20362306a36Sopenharmony_ci	gc->chip_types[1].handler		= handle_edge_irq;
20462306a36Sopenharmony_ci
20562306a36Sopenharmony_ci	/* Disable any active interrupts */
20662306a36Sopenharmony_ci	sunxi_sc_nmi_write(gc, reg_offs->enable, 0);
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_ci	/* Clear any pending NMI interrupts */
20962306a36Sopenharmony_ci	sunxi_sc_nmi_write(gc, reg_offs->pend, SUNXI_NMI_IRQ_BIT);
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci	irq_set_chained_handler_and_data(irq, sunxi_sc_nmi_handle_irq, domain);
21262306a36Sopenharmony_ci
21362306a36Sopenharmony_ci	return 0;
21462306a36Sopenharmony_ci
21562306a36Sopenharmony_cifail_irqd_remove:
21662306a36Sopenharmony_ci	irq_domain_remove(domain);
21762306a36Sopenharmony_ci
21862306a36Sopenharmony_ci	return ret;
21962306a36Sopenharmony_ci}
22062306a36Sopenharmony_ci
22162306a36Sopenharmony_cistatic int __init sun6i_sc_nmi_irq_init(struct device_node *node,
22262306a36Sopenharmony_ci					struct device_node *parent)
22362306a36Sopenharmony_ci{
22462306a36Sopenharmony_ci	return sunxi_sc_nmi_irq_init(node, &sun6i_reg_offs);
22562306a36Sopenharmony_ci}
22662306a36Sopenharmony_ciIRQCHIP_DECLARE(sun6i_sc_nmi, "allwinner,sun6i-a31-sc-nmi", sun6i_sc_nmi_irq_init);
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_cistatic int __init sun7i_sc_nmi_irq_init(struct device_node *node,
22962306a36Sopenharmony_ci					struct device_node *parent)
23062306a36Sopenharmony_ci{
23162306a36Sopenharmony_ci	return sunxi_sc_nmi_irq_init(node, &sun7i_reg_offs);
23262306a36Sopenharmony_ci}
23362306a36Sopenharmony_ciIRQCHIP_DECLARE(sun7i_sc_nmi, "allwinner,sun7i-a20-sc-nmi", sun7i_sc_nmi_irq_init);
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_cistatic int __init sun9i_nmi_irq_init(struct device_node *node,
23662306a36Sopenharmony_ci				     struct device_node *parent)
23762306a36Sopenharmony_ci{
23862306a36Sopenharmony_ci	return sunxi_sc_nmi_irq_init(node, &sun9i_reg_offs);
23962306a36Sopenharmony_ci}
24062306a36Sopenharmony_ciIRQCHIP_DECLARE(sun9i_nmi, "allwinner,sun9i-a80-nmi", sun9i_nmi_irq_init);
241