1bf973285SKuninori Morimoto // SPDX-License-Identifier: GPL-2.0
244358048SMagnus Damm /*
344358048SMagnus Damm  * Renesas INTC External IRQ Pin Driver
444358048SMagnus Damm  *
544358048SMagnus Damm  *  Copyright (C) 2013 Magnus Damm
644358048SMagnus Damm  */
744358048SMagnus Damm 
844358048SMagnus Damm #include <linux/init.h>
9894db164SGuennadi Liakhovetski #include <linux/of.h>
1044358048SMagnus Damm #include <linux/platform_device.h>
1144358048SMagnus Damm #include <linux/spinlock.h>
1244358048SMagnus Damm #include <linux/interrupt.h>
1344358048SMagnus Damm #include <linux/ioport.h>
1444358048SMagnus Damm #include <linux/io.h>
1544358048SMagnus Damm #include <linux/irq.h>
1644358048SMagnus Damm #include <linux/irqdomain.h>
1744358048SMagnus Damm #include <linux/err.h>
1844358048SMagnus Damm #include <linux/slab.h>
1944358048SMagnus Damm #include <linux/module.h>
20705bc96cSGeert Uytterhoeven #include <linux/pm_runtime.h>
2144358048SMagnus Damm 
2244358048SMagnus Damm #define INTC_IRQPIN_MAX 8 /* maximum 8 interrupts per driver instance */
2344358048SMagnus Damm 
2444358048SMagnus Damm #define INTC_IRQPIN_REG_SENSE 0 /* ICRn */
2544358048SMagnus Damm #define INTC_IRQPIN_REG_PRIO 1 /* INTPRInn */
2644358048SMagnus Damm #define INTC_IRQPIN_REG_SOURCE 2 /* INTREQnn */
2744358048SMagnus Damm #define INTC_IRQPIN_REG_MASK 3 /* INTMSKnn */
2844358048SMagnus Damm #define INTC_IRQPIN_REG_CLEAR 4 /* INTMSKCLRnn */
29e03f9088SMagnus Damm #define INTC_IRQPIN_REG_NR_MANDATORY 5
30e03f9088SMagnus Damm #define INTC_IRQPIN_REG_IRLM 5 /* ICR0 with IRLM bit (optional) */
31e03f9088SMagnus Damm #define INTC_IRQPIN_REG_NR 6
3244358048SMagnus Damm 
3344358048SMagnus Damm /* INTC external IRQ PIN hardware register access:
3444358048SMagnus Damm  *
3544358048SMagnus Damm  * SENSE is read-write 32-bit with 2-bits or 4-bits per IRQ (*)
3644358048SMagnus Damm  * PRIO is read-write 32-bit with 4-bits per IRQ (**)
3744358048SMagnus Damm  * SOURCE is read-only 32-bit or 8-bit with 1-bit per IRQ (***)
3844358048SMagnus Damm  * MASK is write-only 32-bit or 8-bit with 1-bit per IRQ (***)
3944358048SMagnus Damm  * CLEAR is write-only 32-bit or 8-bit with 1-bit per IRQ (***)
4044358048SMagnus Damm  *
4144358048SMagnus Damm  * (*) May be accessed by more than one driver instance - lock needed
4244358048SMagnus Damm  * (**) Read-modify-write access by one driver instance - lock needed
4344358048SMagnus Damm  * (***) Accessed by one driver instance only - no locking needed
4444358048SMagnus Damm  */
4544358048SMagnus Damm 
4644358048SMagnus Damm struct intc_irqpin_iomem {
4744358048SMagnus Damm 	void __iomem *iomem;
4844358048SMagnus Damm 	unsigned long (*read)(void __iomem *iomem);
4944358048SMagnus Damm 	void (*write)(void __iomem *iomem, unsigned long data);
5044358048SMagnus Damm 	int width;
5144358048SMagnus Damm };
5244358048SMagnus Damm 
5344358048SMagnus Damm struct intc_irqpin_irq {
5444358048SMagnus Damm 	int hw_irq;
5533f958f2SMagnus Damm 	int requested_irq;
5633f958f2SMagnus Damm 	int domain_irq;
5744358048SMagnus Damm 	struct intc_irqpin_priv *p;
5844358048SMagnus Damm };
5944358048SMagnus Damm 
6044358048SMagnus Damm struct intc_irqpin_priv {
6144358048SMagnus Damm 	struct intc_irqpin_iomem iomem[INTC_IRQPIN_REG_NR];
6244358048SMagnus Damm 	struct intc_irqpin_irq irq[INTC_IRQPIN_MAX];
63f9551a9cSGeert Uytterhoeven 	unsigned int sense_bitfield_width;
6444358048SMagnus Damm 	struct platform_device *pdev;
6544358048SMagnus Damm 	struct irq_chip irq_chip;
6644358048SMagnus Damm 	struct irq_domain *irq_domain;
6766bf8252SGeert Uytterhoeven 	atomic_t wakeup_path;
6886e57ca7SGeert Uytterhoeven 	unsigned shared_irqs:1;
69427cc720SBastian Hecht 	u8 shared_irq_mask;
7044358048SMagnus Damm };
7144358048SMagnus Damm 
7286e57ca7SGeert Uytterhoeven struct intc_irqpin_config {
73b388bdf2SGeert Uytterhoeven 	int irlm_bit;		/* -1 if non-existent */
74e03f9088SMagnus Damm };
75e03f9088SMagnus Damm 
intc_irqpin_read32(void __iomem * iomem)7644358048SMagnus Damm static unsigned long intc_irqpin_read32(void __iomem *iomem)
7744358048SMagnus Damm {
7844358048SMagnus Damm 	return ioread32(iomem);
7944358048SMagnus Damm }
8044358048SMagnus Damm 
intc_irqpin_read8(void __iomem * iomem)8144358048SMagnus Damm static unsigned long intc_irqpin_read8(void __iomem *iomem)
8244358048SMagnus Damm {
8344358048SMagnus Damm 	return ioread8(iomem);
8444358048SMagnus Damm }
8544358048SMagnus Damm 
intc_irqpin_write32(void __iomem * iomem,unsigned long data)8644358048SMagnus Damm static void intc_irqpin_write32(void __iomem *iomem, unsigned long data)
8744358048SMagnus Damm {
8844358048SMagnus Damm 	iowrite32(data, iomem);
8944358048SMagnus Damm }
9044358048SMagnus Damm 
intc_irqpin_write8(void __iomem * iomem,unsigned long data)9144358048SMagnus Damm static void intc_irqpin_write8(void __iomem *iomem, unsigned long data)
9244358048SMagnus Damm {
9344358048SMagnus Damm 	iowrite8(data, iomem);
9444358048SMagnus Damm }
9544358048SMagnus Damm 
intc_irqpin_read(struct intc_irqpin_priv * p,int reg)9644358048SMagnus Damm static inline unsigned long intc_irqpin_read(struct intc_irqpin_priv *p,
9744358048SMagnus Damm 					     int reg)
9844358048SMagnus Damm {
9944358048SMagnus Damm 	struct intc_irqpin_iomem *i = &p->iomem[reg];
100862d3098SMagnus Damm 
10144358048SMagnus Damm 	return i->read(i->iomem);
10244358048SMagnus Damm }
10344358048SMagnus Damm 
intc_irqpin_write(struct intc_irqpin_priv * p,int reg,unsigned long data)10444358048SMagnus Damm static inline void intc_irqpin_write(struct intc_irqpin_priv *p,
10544358048SMagnus Damm 				     int reg, unsigned long data)
10644358048SMagnus Damm {
10744358048SMagnus Damm 	struct intc_irqpin_iomem *i = &p->iomem[reg];
108862d3098SMagnus Damm 
10944358048SMagnus Damm 	i->write(i->iomem, data);
11044358048SMagnus Damm }
11144358048SMagnus Damm 
intc_irqpin_hwirq_mask(struct intc_irqpin_priv * p,int reg,int hw_irq)11244358048SMagnus Damm static inline unsigned long intc_irqpin_hwirq_mask(struct intc_irqpin_priv *p,
11344358048SMagnus Damm 						   int reg, int hw_irq)
11444358048SMagnus Damm {
11544358048SMagnus Damm 	return BIT((p->iomem[reg].width - 1) - hw_irq);
11644358048SMagnus Damm }
11744358048SMagnus Damm 
intc_irqpin_irq_write_hwirq(struct intc_irqpin_priv * p,int reg,int hw_irq)11844358048SMagnus Damm static inline void intc_irqpin_irq_write_hwirq(struct intc_irqpin_priv *p,
11944358048SMagnus Damm 					       int reg, int hw_irq)
12044358048SMagnus Damm {
12144358048SMagnus Damm 	intc_irqpin_write(p, reg, intc_irqpin_hwirq_mask(p, reg, hw_irq));
12244358048SMagnus Damm }
12344358048SMagnus Damm 
12444358048SMagnus Damm static DEFINE_RAW_SPINLOCK(intc_irqpin_lock); /* only used by slow path */
12544358048SMagnus Damm 
intc_irqpin_read_modify_write(struct intc_irqpin_priv * p,int reg,int shift,int width,int value)12644358048SMagnus Damm static void intc_irqpin_read_modify_write(struct intc_irqpin_priv *p,
12744358048SMagnus Damm 					  int reg, int shift,
12844358048SMagnus Damm 					  int width, int value)
12944358048SMagnus Damm {
13044358048SMagnus Damm 	unsigned long flags;
13144358048SMagnus Damm 	unsigned long tmp;
13244358048SMagnus Damm 
13344358048SMagnus Damm 	raw_spin_lock_irqsave(&intc_irqpin_lock, flags);
13444358048SMagnus Damm 
13544358048SMagnus Damm 	tmp = intc_irqpin_read(p, reg);
13644358048SMagnus Damm 	tmp &= ~(((1 << width) - 1) << shift);
13744358048SMagnus Damm 	tmp |= value << shift;
13844358048SMagnus Damm 	intc_irqpin_write(p, reg, tmp);
13944358048SMagnus Damm 
14044358048SMagnus Damm 	raw_spin_unlock_irqrestore(&intc_irqpin_lock, flags);
14144358048SMagnus Damm }
14244358048SMagnus Damm 
intc_irqpin_mask_unmask_prio(struct intc_irqpin_priv * p,int irq,int do_mask)14344358048SMagnus Damm static void intc_irqpin_mask_unmask_prio(struct intc_irqpin_priv *p,
14444358048SMagnus Damm 					 int irq, int do_mask)
14544358048SMagnus Damm {
146e55bc558SLaurent Pinchart 	/* The PRIO register is assumed to be 32-bit with fixed 4-bit fields. */
147e55bc558SLaurent Pinchart 	int bitfield_width = 4;
148e55bc558SLaurent Pinchart 	int shift = 32 - (irq + 1) * bitfield_width;
14944358048SMagnus Damm 
15044358048SMagnus Damm 	intc_irqpin_read_modify_write(p, INTC_IRQPIN_REG_PRIO,
15144358048SMagnus Damm 				      shift, bitfield_width,
15244358048SMagnus Damm 				      do_mask ? 0 : (1 << bitfield_width) - 1);
15344358048SMagnus Damm }
15444358048SMagnus Damm 
intc_irqpin_set_sense(struct intc_irqpin_priv * p,int irq,int value)15544358048SMagnus Damm static int intc_irqpin_set_sense(struct intc_irqpin_priv *p, int irq, int value)
15644358048SMagnus Damm {
157e55bc558SLaurent Pinchart 	/* The SENSE register is assumed to be 32-bit. */
158f9551a9cSGeert Uytterhoeven 	int bitfield_width = p->sense_bitfield_width;
159e55bc558SLaurent Pinchart 	int shift = 32 - (irq + 1) * bitfield_width;
16044358048SMagnus Damm 
16144358048SMagnus Damm 	dev_dbg(&p->pdev->dev, "sense irq = %d, mode = %d\n", irq, value);
16244358048SMagnus Damm 
16344358048SMagnus Damm 	if (value >= (1 << bitfield_width))
16444358048SMagnus Damm 		return -EINVAL;
16544358048SMagnus Damm 
16644358048SMagnus Damm 	intc_irqpin_read_modify_write(p, INTC_IRQPIN_REG_SENSE, shift,
16744358048SMagnus Damm 				      bitfield_width, value);
16844358048SMagnus Damm 	return 0;
16944358048SMagnus Damm }
17044358048SMagnus Damm 
intc_irqpin_dbg(struct intc_irqpin_irq * i,char * str)17144358048SMagnus Damm static void intc_irqpin_dbg(struct intc_irqpin_irq *i, char *str)
17244358048SMagnus Damm {
17344358048SMagnus Damm 	dev_dbg(&i->p->pdev->dev, "%s (%d:%d:%d)\n",
17433f958f2SMagnus Damm 		str, i->requested_irq, i->hw_irq, i->domain_irq);
17544358048SMagnus Damm }
17644358048SMagnus Damm 
intc_irqpin_irq_enable(struct irq_data * d)17744358048SMagnus Damm static void intc_irqpin_irq_enable(struct irq_data *d)
17844358048SMagnus Damm {
17944358048SMagnus Damm 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
18044358048SMagnus Damm 	int hw_irq = irqd_to_hwirq(d);
18144358048SMagnus Damm 
18244358048SMagnus Damm 	intc_irqpin_dbg(&p->irq[hw_irq], "enable");
18344358048SMagnus Damm 	intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_CLEAR, hw_irq);
18444358048SMagnus Damm }
18544358048SMagnus Damm 
intc_irqpin_irq_disable(struct irq_data * d)18644358048SMagnus Damm static void intc_irqpin_irq_disable(struct irq_data *d)
18744358048SMagnus Damm {
18844358048SMagnus Damm 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
18944358048SMagnus Damm 	int hw_irq = irqd_to_hwirq(d);
19044358048SMagnus Damm 
19144358048SMagnus Damm 	intc_irqpin_dbg(&p->irq[hw_irq], "disable");
19244358048SMagnus Damm 	intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_MASK, hw_irq);
19344358048SMagnus Damm }
19444358048SMagnus Damm 
intc_irqpin_shared_irq_enable(struct irq_data * d)195427cc720SBastian Hecht static void intc_irqpin_shared_irq_enable(struct irq_data *d)
196427cc720SBastian Hecht {
197427cc720SBastian Hecht 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
198427cc720SBastian Hecht 	int hw_irq = irqd_to_hwirq(d);
199427cc720SBastian Hecht 
200427cc720SBastian Hecht 	intc_irqpin_dbg(&p->irq[hw_irq], "shared enable");
201427cc720SBastian Hecht 	intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_CLEAR, hw_irq);
202427cc720SBastian Hecht 
203427cc720SBastian Hecht 	p->shared_irq_mask &= ~BIT(hw_irq);
204427cc720SBastian Hecht }
205427cc720SBastian Hecht 
intc_irqpin_shared_irq_disable(struct irq_data * d)206427cc720SBastian Hecht static void intc_irqpin_shared_irq_disable(struct irq_data *d)
207427cc720SBastian Hecht {
208427cc720SBastian Hecht 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
209427cc720SBastian Hecht 	int hw_irq = irqd_to_hwirq(d);
210427cc720SBastian Hecht 
211427cc720SBastian Hecht 	intc_irqpin_dbg(&p->irq[hw_irq], "shared disable");
212427cc720SBastian Hecht 	intc_irqpin_irq_write_hwirq(p, INTC_IRQPIN_REG_MASK, hw_irq);
213427cc720SBastian Hecht 
214427cc720SBastian Hecht 	p->shared_irq_mask |= BIT(hw_irq);
215427cc720SBastian Hecht }
216427cc720SBastian Hecht 
intc_irqpin_irq_enable_force(struct irq_data * d)21744358048SMagnus Damm static void intc_irqpin_irq_enable_force(struct irq_data *d)
21844358048SMagnus Damm {
21944358048SMagnus Damm 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
22033f958f2SMagnus Damm 	int irq = p->irq[irqd_to_hwirq(d)].requested_irq;
22144358048SMagnus Damm 
22244358048SMagnus Damm 	intc_irqpin_irq_enable(d);
223d1b6aecdSMagnus Damm 
224d1b6aecdSMagnus Damm 	/* enable interrupt through parent interrupt controller,
225d1b6aecdSMagnus Damm 	 * assumes non-shared interrupt with 1:1 mapping
226d1b6aecdSMagnus Damm 	 * needed for busted IRQs on some SoCs like sh73a0
227d1b6aecdSMagnus Damm 	 */
22844358048SMagnus Damm 	irq_get_chip(irq)->irq_unmask(irq_get_irq_data(irq));
22944358048SMagnus Damm }
23044358048SMagnus Damm 
intc_irqpin_irq_disable_force(struct irq_data * d)23144358048SMagnus Damm static void intc_irqpin_irq_disable_force(struct irq_data *d)
23244358048SMagnus Damm {
23344358048SMagnus Damm 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
23433f958f2SMagnus Damm 	int irq = p->irq[irqd_to_hwirq(d)].requested_irq;
23544358048SMagnus Damm 
236d1b6aecdSMagnus Damm 	/* disable interrupt through parent interrupt controller,
237d1b6aecdSMagnus Damm 	 * assumes non-shared interrupt with 1:1 mapping
238d1b6aecdSMagnus Damm 	 * needed for busted IRQs on some SoCs like sh73a0
239d1b6aecdSMagnus Damm 	 */
24044358048SMagnus Damm 	irq_get_chip(irq)->irq_mask(irq_get_irq_data(irq));
24144358048SMagnus Damm 	intc_irqpin_irq_disable(d);
24244358048SMagnus Damm }
24344358048SMagnus Damm 
24444358048SMagnus Damm #define INTC_IRQ_SENSE_VALID 0x10
24544358048SMagnus Damm #define INTC_IRQ_SENSE(x) (x + INTC_IRQ_SENSE_VALID)
24644358048SMagnus Damm 
24744358048SMagnus Damm static unsigned char intc_irqpin_sense[IRQ_TYPE_SENSE_MASK + 1] = {
24844358048SMagnus Damm 	[IRQ_TYPE_EDGE_FALLING] = INTC_IRQ_SENSE(0x00),
24944358048SMagnus Damm 	[IRQ_TYPE_EDGE_RISING] = INTC_IRQ_SENSE(0x01),
25044358048SMagnus Damm 	[IRQ_TYPE_LEVEL_LOW] = INTC_IRQ_SENSE(0x02),
25144358048SMagnus Damm 	[IRQ_TYPE_LEVEL_HIGH] = INTC_IRQ_SENSE(0x03),
25244358048SMagnus Damm 	[IRQ_TYPE_EDGE_BOTH] = INTC_IRQ_SENSE(0x04),
25344358048SMagnus Damm };
25444358048SMagnus Damm 
intc_irqpin_irq_set_type(struct irq_data * d,unsigned int type)25544358048SMagnus Damm static int intc_irqpin_irq_set_type(struct irq_data *d, unsigned int type)
25644358048SMagnus Damm {
25744358048SMagnus Damm 	unsigned char value = intc_irqpin_sense[type & IRQ_TYPE_SENSE_MASK];
25844358048SMagnus Damm 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
25944358048SMagnus Damm 
26044358048SMagnus Damm 	if (!(value & INTC_IRQ_SENSE_VALID))
26144358048SMagnus Damm 		return -EINVAL;
26244358048SMagnus Damm 
26344358048SMagnus Damm 	return intc_irqpin_set_sense(p, irqd_to_hwirq(d),
26444358048SMagnus Damm 				     value ^ INTC_IRQ_SENSE_VALID);
26544358048SMagnus Damm }
26644358048SMagnus Damm 
intc_irqpin_irq_set_wake(struct irq_data * d,unsigned int on)267705bc96cSGeert Uytterhoeven static int intc_irqpin_irq_set_wake(struct irq_data *d, unsigned int on)
268705bc96cSGeert Uytterhoeven {
269705bc96cSGeert Uytterhoeven 	struct intc_irqpin_priv *p = irq_data_get_irq_chip_data(d);
270f4e209cdSGeert Uytterhoeven 	int hw_irq = irqd_to_hwirq(d);
271f4e209cdSGeert Uytterhoeven 
272f4e209cdSGeert Uytterhoeven 	irq_set_irq_wake(p->irq[hw_irq].requested_irq, on);
273705bc96cSGeert Uytterhoeven 	if (on)
27466bf8252SGeert Uytterhoeven 		atomic_inc(&p->wakeup_path);
275705bc96cSGeert Uytterhoeven 	else
27666bf8252SGeert Uytterhoeven 		atomic_dec(&p->wakeup_path);
277705bc96cSGeert Uytterhoeven 
278705bc96cSGeert Uytterhoeven 	return 0;
279705bc96cSGeert Uytterhoeven }
280705bc96cSGeert Uytterhoeven 
intc_irqpin_irq_handler(int irq,void * dev_id)28144358048SMagnus Damm static irqreturn_t intc_irqpin_irq_handler(int irq, void *dev_id)
28244358048SMagnus Damm {
28344358048SMagnus Damm 	struct intc_irqpin_irq *i = dev_id;
28444358048SMagnus Damm 	struct intc_irqpin_priv *p = i->p;
28544358048SMagnus Damm 	unsigned long bit;
28644358048SMagnus Damm 
28744358048SMagnus Damm 	intc_irqpin_dbg(i, "demux1");
28844358048SMagnus Damm 	bit = intc_irqpin_hwirq_mask(p, INTC_IRQPIN_REG_SOURCE, i->hw_irq);
28944358048SMagnus Damm 
29044358048SMagnus Damm 	if (intc_irqpin_read(p, INTC_IRQPIN_REG_SOURCE) & bit) {
29144358048SMagnus Damm 		intc_irqpin_write(p, INTC_IRQPIN_REG_SOURCE, ~bit);
29244358048SMagnus Damm 		intc_irqpin_dbg(i, "demux2");
29333f958f2SMagnus Damm 		generic_handle_irq(i->domain_irq);
29444358048SMagnus Damm 		return IRQ_HANDLED;
29544358048SMagnus Damm 	}
29644358048SMagnus Damm 	return IRQ_NONE;
29744358048SMagnus Damm }
29844358048SMagnus Damm 
intc_irqpin_shared_irq_handler(int irq,void * dev_id)299427cc720SBastian Hecht static irqreturn_t intc_irqpin_shared_irq_handler(int irq, void *dev_id)
300427cc720SBastian Hecht {
301427cc720SBastian Hecht 	struct intc_irqpin_priv *p = dev_id;
302427cc720SBastian Hecht 	unsigned int reg_source = intc_irqpin_read(p, INTC_IRQPIN_REG_SOURCE);
303427cc720SBastian Hecht 	irqreturn_t status = IRQ_NONE;
304427cc720SBastian Hecht 	int k;
305427cc720SBastian Hecht 
306427cc720SBastian Hecht 	for (k = 0; k < 8; k++) {
307427cc720SBastian Hecht 		if (reg_source & BIT(7 - k)) {
308427cc720SBastian Hecht 			if (BIT(k) & p->shared_irq_mask)
309427cc720SBastian Hecht 				continue;
310427cc720SBastian Hecht 
311427cc720SBastian Hecht 			status |= intc_irqpin_irq_handler(irq, &p->irq[k]);
312427cc720SBastian Hecht 		}
313427cc720SBastian Hecht 	}
314427cc720SBastian Hecht 
315427cc720SBastian Hecht 	return status;
316427cc720SBastian Hecht }
317427cc720SBastian Hecht 
318769b5cf7SGeert Uytterhoeven /*
319769b5cf7SGeert Uytterhoeven  * This lock class tells lockdep that INTC External IRQ Pin irqs are in a
320769b5cf7SGeert Uytterhoeven  * different category than their parents, so it won't report false recursion.
321769b5cf7SGeert Uytterhoeven  */
322769b5cf7SGeert Uytterhoeven static struct lock_class_key intc_irqpin_irq_lock_class;
323769b5cf7SGeert Uytterhoeven 
32439c3fd58SAndrew Lunn /* And this is for the request mutex */
32539c3fd58SAndrew Lunn static struct lock_class_key intc_irqpin_irq_request_class;
32639c3fd58SAndrew Lunn 
intc_irqpin_irq_domain_map(struct irq_domain * h,unsigned int virq,irq_hw_number_t hw)32744358048SMagnus Damm static int intc_irqpin_irq_domain_map(struct irq_domain *h, unsigned int virq,
32844358048SMagnus Damm 				      irq_hw_number_t hw)
32944358048SMagnus Damm {
33044358048SMagnus Damm 	struct intc_irqpin_priv *p = h->host_data;
33144358048SMagnus Damm 
33233f958f2SMagnus Damm 	p->irq[hw].domain_irq = virq;
33333f958f2SMagnus Damm 	p->irq[hw].hw_irq = hw;
33433f958f2SMagnus Damm 
33544358048SMagnus Damm 	intc_irqpin_dbg(&p->irq[hw], "map");
33644358048SMagnus Damm 	irq_set_chip_data(virq, h->host_data);
33739c3fd58SAndrew Lunn 	irq_set_lockdep_class(virq, &intc_irqpin_irq_lock_class,
33839c3fd58SAndrew Lunn 			      &intc_irqpin_irq_request_class);
33944358048SMagnus Damm 	irq_set_chip_and_handler(virq, &p->irq_chip, handle_level_irq);
34044358048SMagnus Damm 	return 0;
34144358048SMagnus Damm }
34244358048SMagnus Damm 
34396009736SKrzysztof Kozlowski static const struct irq_domain_ops intc_irqpin_irq_domain_ops = {
34444358048SMagnus Damm 	.map	= intc_irqpin_irq_domain_map,
3459d833bbeSMagnus Damm 	.xlate  = irq_domain_xlate_twocell,
34644358048SMagnus Damm };
34744358048SMagnus Damm 
34886e57ca7SGeert Uytterhoeven static const struct intc_irqpin_config intc_irqpin_irlm_r8a777x = {
349e03f9088SMagnus Damm 	.irlm_bit = 23, /* ICR0.IRLM0 */
35086e57ca7SGeert Uytterhoeven };
35186e57ca7SGeert Uytterhoeven 
35286e57ca7SGeert Uytterhoeven static const struct intc_irqpin_config intc_irqpin_rmobile = {
353b388bdf2SGeert Uytterhoeven 	.irlm_bit = -1,
354e03f9088SMagnus Damm };
355e03f9088SMagnus Damm 
356e03f9088SMagnus Damm static const struct of_device_id intc_irqpin_dt_ids[] = {
357e03f9088SMagnus Damm 	{ .compatible = "renesas,intc-irqpin", },
35826c21dd9SUlrich Hecht 	{ .compatible = "renesas,intc-irqpin-r8a7778",
35926c21dd9SUlrich Hecht 	  .data = &intc_irqpin_irlm_r8a777x },
360e03f9088SMagnus Damm 	{ .compatible = "renesas,intc-irqpin-r8a7779",
36126c21dd9SUlrich Hecht 	  .data = &intc_irqpin_irlm_r8a777x },
36286e57ca7SGeert Uytterhoeven 	{ .compatible = "renesas,intc-irqpin-r8a7740",
36386e57ca7SGeert Uytterhoeven 	  .data = &intc_irqpin_rmobile },
36486e57ca7SGeert Uytterhoeven 	{ .compatible = "renesas,intc-irqpin-sh73a0",
36586e57ca7SGeert Uytterhoeven 	  .data = &intc_irqpin_rmobile },
366e03f9088SMagnus Damm 	{},
367e03f9088SMagnus Damm };
368e03f9088SMagnus Damm MODULE_DEVICE_TABLE(of, intc_irqpin_dt_ids);
369e03f9088SMagnus Damm 
intc_irqpin_probe(struct platform_device * pdev)37044358048SMagnus Damm static int intc_irqpin_probe(struct platform_device *pdev)
37144358048SMagnus Damm {
37242a5968cSGeert Uytterhoeven 	const struct intc_irqpin_config *config;
37336845f1bSGeert Uytterhoeven 	struct device *dev = &pdev->dev;
37444358048SMagnus Damm 	struct intc_irqpin_priv *p;
37544358048SMagnus Damm 	struct intc_irqpin_iomem *i;
37644358048SMagnus Damm 	struct resource *io[INTC_IRQPIN_REG_NR];
37744358048SMagnus Damm 	struct irq_chip *irq_chip;
37844358048SMagnus Damm 	void (*enable_fn)(struct irq_data *d);
37944358048SMagnus Damm 	void (*disable_fn)(struct irq_data *d);
38036845f1bSGeert Uytterhoeven 	const char *name = dev_name(dev);
381f9551a9cSGeert Uytterhoeven 	bool control_parent;
3821affe594SGeert Uytterhoeven 	unsigned int nirqs;
383427cc720SBastian Hecht 	int ref_irq;
38444358048SMagnus Damm 	int ret;
38544358048SMagnus Damm 	int k;
38644358048SMagnus Damm 
38736845f1bSGeert Uytterhoeven 	p = devm_kzalloc(dev, sizeof(*p), GFP_KERNEL);
38889626d4bSGeert Uytterhoeven 	if (!p)
389705bc96cSGeert Uytterhoeven 		return -ENOMEM;
39044358048SMagnus Damm 
39144358048SMagnus Damm 	/* deal with driver instance configuration */
39236845f1bSGeert Uytterhoeven 	of_property_read_u32(dev->of_node, "sense-bitfield-width",
393f9551a9cSGeert Uytterhoeven 			     &p->sense_bitfield_width);
394f9551a9cSGeert Uytterhoeven 	control_parent = of_property_read_bool(dev->of_node, "control-parent");
395f9551a9cSGeert Uytterhoeven 	if (!p->sense_bitfield_width)
396f9551a9cSGeert Uytterhoeven 		p->sense_bitfield_width = 4; /* default to 4 bits */
39744358048SMagnus Damm 
39844358048SMagnus Damm 	p->pdev = pdev;
39944358048SMagnus Damm 	platform_set_drvdata(pdev, p);
40044358048SMagnus Damm 
40142a5968cSGeert Uytterhoeven 	config = of_device_get_match_data(dev);
402705bc96cSGeert Uytterhoeven 
403705bc96cSGeert Uytterhoeven 	pm_runtime_enable(dev);
404705bc96cSGeert Uytterhoeven 	pm_runtime_get_sync(dev);
405705bc96cSGeert Uytterhoeven 
406e03f9088SMagnus Damm 	/* get hold of register banks */
407e03f9088SMagnus Damm 	memset(io, 0, sizeof(io));
40844358048SMagnus Damm 	for (k = 0; k < INTC_IRQPIN_REG_NR; k++) {
40944358048SMagnus Damm 		io[k] = platform_get_resource(pdev, IORESOURCE_MEM, k);
410e03f9088SMagnus Damm 		if (!io[k] && k < INTC_IRQPIN_REG_NR_MANDATORY) {
41136845f1bSGeert Uytterhoeven 			dev_err(dev, "not enough IOMEM resources\n");
41244358048SMagnus Damm 			ret = -EINVAL;
41308eba5baSMagnus Damm 			goto err0;
41444358048SMagnus Damm 		}
41544358048SMagnus Damm 	}
41644358048SMagnus Damm 
41744358048SMagnus Damm 	/* allow any number of IRQs between 1 and INTC_IRQPIN_MAX */
41844358048SMagnus Damm 	for (k = 0; k < INTC_IRQPIN_MAX; k++) {
41931bd548fSLad Prabhakar 		ret = platform_get_irq_optional(pdev, k);
42031bd548fSLad Prabhakar 		if (ret == -ENXIO)
42144358048SMagnus Damm 			break;
42231bd548fSLad Prabhakar 		if (ret < 0)
42331bd548fSLad Prabhakar 			goto err0;
42444358048SMagnus Damm 
42544358048SMagnus Damm 		p->irq[k].p = p;
42631bd548fSLad Prabhakar 		p->irq[k].requested_irq = ret;
42744358048SMagnus Damm 	}
42844358048SMagnus Damm 
4291affe594SGeert Uytterhoeven 	nirqs = k;
4301affe594SGeert Uytterhoeven 	if (nirqs < 1) {
43136845f1bSGeert Uytterhoeven 		dev_err(dev, "not enough IRQ resources\n");
43244358048SMagnus Damm 		ret = -EINVAL;
43308eba5baSMagnus Damm 		goto err0;
43444358048SMagnus Damm 	}
43544358048SMagnus Damm 
43644358048SMagnus Damm 	/* ioremap IOMEM and setup read/write callbacks */
43744358048SMagnus Damm 	for (k = 0; k < INTC_IRQPIN_REG_NR; k++) {
43844358048SMagnus Damm 		i = &p->iomem[k];
43944358048SMagnus Damm 
440e03f9088SMagnus Damm 		/* handle optional registers */
441e03f9088SMagnus Damm 		if (!io[k])
442e03f9088SMagnus Damm 			continue;
443e03f9088SMagnus Damm 
44444358048SMagnus Damm 		switch (resource_size(io[k])) {
44544358048SMagnus Damm 		case 1:
44644358048SMagnus Damm 			i->width = 8;
44744358048SMagnus Damm 			i->read = intc_irqpin_read8;
44844358048SMagnus Damm 			i->write = intc_irqpin_write8;
44944358048SMagnus Damm 			break;
45044358048SMagnus Damm 		case 4:
45144358048SMagnus Damm 			i->width = 32;
45244358048SMagnus Damm 			i->read = intc_irqpin_read32;
45344358048SMagnus Damm 			i->write = intc_irqpin_write32;
45444358048SMagnus Damm 			break;
45544358048SMagnus Damm 		default:
45636845f1bSGeert Uytterhoeven 			dev_err(dev, "IOMEM size mismatch\n");
45744358048SMagnus Damm 			ret = -EINVAL;
45808eba5baSMagnus Damm 			goto err0;
45944358048SMagnus Damm 		}
46044358048SMagnus Damm 
4614bdc0d67SChristoph Hellwig 		i->iomem = devm_ioremap(dev, io[k]->start,
46208eba5baSMagnus Damm 					resource_size(io[k]));
46344358048SMagnus Damm 		if (!i->iomem) {
46436845f1bSGeert Uytterhoeven 			dev_err(dev, "failed to remap IOMEM\n");
46544358048SMagnus Damm 			ret = -ENXIO;
46608eba5baSMagnus Damm 			goto err0;
46744358048SMagnus Damm 		}
46844358048SMagnus Damm 	}
46944358048SMagnus Damm 
470e03f9088SMagnus Damm 	/* configure "individual IRQ mode" where needed */
471b388bdf2SGeert Uytterhoeven 	if (config && config->irlm_bit >= 0) {
472e03f9088SMagnus Damm 		if (io[INTC_IRQPIN_REG_IRLM])
473e03f9088SMagnus Damm 			intc_irqpin_read_modify_write(p, INTC_IRQPIN_REG_IRLM,
47486e57ca7SGeert Uytterhoeven 						      config->irlm_bit, 1, 1);
475e03f9088SMagnus Damm 		else
476e03f9088SMagnus Damm 			dev_warn(dev, "unable to select IRLM mode\n");
477e03f9088SMagnus Damm 	}
478e03f9088SMagnus Damm 
47944358048SMagnus Damm 	/* mask all interrupts using priority */
4801affe594SGeert Uytterhoeven 	for (k = 0; k < nirqs; k++)
48144358048SMagnus Damm 		intc_irqpin_mask_unmask_prio(p, k, 1);
48244358048SMagnus Damm 
483427cc720SBastian Hecht 	/* clear all pending interrupts */
484427cc720SBastian Hecht 	intc_irqpin_write(p, INTC_IRQPIN_REG_SOURCE, 0x0);
485427cc720SBastian Hecht 
486427cc720SBastian Hecht 	/* scan for shared interrupt lines */
487427cc720SBastian Hecht 	ref_irq = p->irq[0].requested_irq;
48886e57ca7SGeert Uytterhoeven 	p->shared_irqs = 1;
4891affe594SGeert Uytterhoeven 	for (k = 1; k < nirqs; k++) {
490427cc720SBastian Hecht 		if (ref_irq != p->irq[k].requested_irq) {
49186e57ca7SGeert Uytterhoeven 			p->shared_irqs = 0;
492427cc720SBastian Hecht 			break;
493427cc720SBastian Hecht 		}
494427cc720SBastian Hecht 	}
495427cc720SBastian Hecht 
49644358048SMagnus Damm 	/* use more severe masking method if requested */
497f9551a9cSGeert Uytterhoeven 	if (control_parent) {
49844358048SMagnus Damm 		enable_fn = intc_irqpin_irq_enable_force;
49944358048SMagnus Damm 		disable_fn = intc_irqpin_irq_disable_force;
500427cc720SBastian Hecht 	} else if (!p->shared_irqs) {
50144358048SMagnus Damm 		enable_fn = intc_irqpin_irq_enable;
50244358048SMagnus Damm 		disable_fn = intc_irqpin_irq_disable;
503427cc720SBastian Hecht 	} else {
504427cc720SBastian Hecht 		enable_fn = intc_irqpin_shared_irq_enable;
505427cc720SBastian Hecht 		disable_fn = intc_irqpin_shared_irq_disable;
50644358048SMagnus Damm 	}
50744358048SMagnus Damm 
50844358048SMagnus Damm 	irq_chip = &p->irq_chip;
509ec93b94aSGeert Uytterhoeven 	irq_chip->name = "intc-irqpin";
51044358048SMagnus Damm 	irq_chip->irq_mask = disable_fn;
51144358048SMagnus Damm 	irq_chip->irq_unmask = enable_fn;
51244358048SMagnus Damm 	irq_chip->irq_set_type = intc_irqpin_irq_set_type;
513705bc96cSGeert Uytterhoeven 	irq_chip->irq_set_wake = intc_irqpin_irq_set_wake;
514705bc96cSGeert Uytterhoeven 	irq_chip->flags	= IRQCHIP_MASK_ON_SUSPEND;
51544358048SMagnus Damm 
5161affe594SGeert Uytterhoeven 	p->irq_domain = irq_domain_add_simple(dev->of_node, nirqs, 0,
5171affe594SGeert Uytterhoeven 					      &intc_irqpin_irq_domain_ops, p);
51844358048SMagnus Damm 	if (!p->irq_domain) {
51944358048SMagnus Damm 		ret = -ENXIO;
52036845f1bSGeert Uytterhoeven 		dev_err(dev, "cannot initialize irq domain\n");
52108eba5baSMagnus Damm 		goto err0;
52244358048SMagnus Damm 	}
52344358048SMagnus Damm 
524*c2ea6b9bSMarc Zyngier 	irq_domain_set_pm_device(p->irq_domain, dev);
525*c2ea6b9bSMarc Zyngier 
526427cc720SBastian Hecht 	if (p->shared_irqs) {
527427cc720SBastian Hecht 		/* request one shared interrupt */
52836845f1bSGeert Uytterhoeven 		if (devm_request_irq(dev, p->irq[0].requested_irq,
529427cc720SBastian Hecht 				intc_irqpin_shared_irq_handler,
530427cc720SBastian Hecht 				IRQF_SHARED, name, p)) {
53136845f1bSGeert Uytterhoeven 			dev_err(dev, "failed to request low IRQ\n");
53244358048SMagnus Damm 			ret = -ENOENT;
53308eba5baSMagnus Damm 			goto err1;
53444358048SMagnus Damm 		}
535427cc720SBastian Hecht 	} else {
536427cc720SBastian Hecht 		/* request interrupts one by one */
5371affe594SGeert Uytterhoeven 		for (k = 0; k < nirqs; k++) {
53836845f1bSGeert Uytterhoeven 			if (devm_request_irq(dev, p->irq[k].requested_irq,
53936845f1bSGeert Uytterhoeven 					     intc_irqpin_irq_handler, 0, name,
54036845f1bSGeert Uytterhoeven 					     &p->irq[k])) {
54136845f1bSGeert Uytterhoeven 				dev_err(dev, "failed to request low IRQ\n");
542427cc720SBastian Hecht 				ret = -ENOENT;
543427cc720SBastian Hecht 				goto err1;
54444358048SMagnus Damm 			}
545427cc720SBastian Hecht 		}
546427cc720SBastian Hecht 	}
547427cc720SBastian Hecht 
548427cc720SBastian Hecht 	/* unmask all interrupts on prio level */
5491affe594SGeert Uytterhoeven 	for (k = 0; k < nirqs; k++)
550427cc720SBastian Hecht 		intc_irqpin_mask_unmask_prio(p, k, 0);
55144358048SMagnus Damm 
5521affe594SGeert Uytterhoeven 	dev_info(dev, "driving %d irqs\n", nirqs);
55344358048SMagnus Damm 
55444358048SMagnus Damm 	return 0;
55544358048SMagnus Damm 
55644358048SMagnus Damm err1:
55708eba5baSMagnus Damm 	irq_domain_remove(p->irq_domain);
55844358048SMagnus Damm err0:
559705bc96cSGeert Uytterhoeven 	pm_runtime_put(dev);
560705bc96cSGeert Uytterhoeven 	pm_runtime_disable(dev);
56144358048SMagnus Damm 	return ret;
56244358048SMagnus Damm }
56344358048SMagnus Damm 
intc_irqpin_remove(struct platform_device * pdev)56444358048SMagnus Damm static int intc_irqpin_remove(struct platform_device *pdev)
56544358048SMagnus Damm {
56644358048SMagnus Damm 	struct intc_irqpin_priv *p = platform_get_drvdata(pdev);
56744358048SMagnus Damm 
56844358048SMagnus Damm 	irq_domain_remove(p->irq_domain);
569705bc96cSGeert Uytterhoeven 	pm_runtime_put(&pdev->dev);
570705bc96cSGeert Uytterhoeven 	pm_runtime_disable(&pdev->dev);
57144358048SMagnus Damm 	return 0;
57244358048SMagnus Damm }
57344358048SMagnus Damm 
intc_irqpin_suspend(struct device * dev)57466bf8252SGeert Uytterhoeven static int __maybe_unused intc_irqpin_suspend(struct device *dev)
57566bf8252SGeert Uytterhoeven {
57666bf8252SGeert Uytterhoeven 	struct intc_irqpin_priv *p = dev_get_drvdata(dev);
57766bf8252SGeert Uytterhoeven 
57866bf8252SGeert Uytterhoeven 	if (atomic_read(&p->wakeup_path))
57966bf8252SGeert Uytterhoeven 		device_set_wakeup_path(dev);
58066bf8252SGeert Uytterhoeven 
58166bf8252SGeert Uytterhoeven 	return 0;
58266bf8252SGeert Uytterhoeven }
58366bf8252SGeert Uytterhoeven 
58466bf8252SGeert Uytterhoeven static SIMPLE_DEV_PM_OPS(intc_irqpin_pm_ops, intc_irqpin_suspend, NULL);
58566bf8252SGeert Uytterhoeven 
58644358048SMagnus Damm static struct platform_driver intc_irqpin_device_driver = {
58744358048SMagnus Damm 	.probe		= intc_irqpin_probe,
58844358048SMagnus Damm 	.remove		= intc_irqpin_remove,
58944358048SMagnus Damm 	.driver		= {
59044358048SMagnus Damm 		.name	= "renesas_intc_irqpin",
5919d833bbeSMagnus Damm 		.of_match_table = intc_irqpin_dt_ids,
59266bf8252SGeert Uytterhoeven 		.pm	= &intc_irqpin_pm_ops,
59344358048SMagnus Damm 	}
59444358048SMagnus Damm };
59544358048SMagnus Damm 
intc_irqpin_init(void)59644358048SMagnus Damm static int __init intc_irqpin_init(void)
59744358048SMagnus Damm {
59844358048SMagnus Damm 	return platform_driver_register(&intc_irqpin_device_driver);
59944358048SMagnus Damm }
60044358048SMagnus Damm postcore_initcall(intc_irqpin_init);
60144358048SMagnus Damm 
intc_irqpin_exit(void)60244358048SMagnus Damm static void __exit intc_irqpin_exit(void)
60344358048SMagnus Damm {
60444358048SMagnus Damm 	platform_driver_unregister(&intc_irqpin_device_driver);
60544358048SMagnus Damm }
60644358048SMagnus Damm module_exit(intc_irqpin_exit);
60744358048SMagnus Damm 
60844358048SMagnus Damm MODULE_AUTHOR("Magnus Damm");
60944358048SMagnus Damm MODULE_DESCRIPTION("Renesas INTC External IRQ Pin Driver");
610