blob: 803aeb126f0e9f9d6eaa061e26d7b21df6cc51b6 [file] [log] [blame]
Andrew Scullb4b6d4a2019-01-02 15:54:55 +00001// SPDX-License-Identifier: GPL-2.0
2#include <linux/init.h>
3#include <linux/list.h>
4#include <linux/io.h>
5
6#include <asm/mach/irq.h>
7#include <asm/hardware/iomd.h>
8#include <asm/irq.h>
9#include <asm/fiq.h>
10
David Brazdil0f672f62019-12-10 10:32:29 +000011// These are offsets from the stat register for each IRQ bank
12#define STAT 0x00
13#define REQ 0x04
14#define CLR 0x04
15#define MASK 0x08
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000016
David Brazdil0f672f62019-12-10 10:32:29 +000017static void __iomem *iomd_get_base(struct irq_data *d)
18{
19 void *cd = irq_data_get_irq_chip_data(d);
20
21 return (void __iomem *)(unsigned long)cd;
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000022}
23
David Brazdil0f672f62019-12-10 10:32:29 +000024static void iomd_set_base_mask(unsigned int irq, void __iomem *base, u32 mask)
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000025{
David Brazdil0f672f62019-12-10 10:32:29 +000026 struct irq_data *d = irq_get_irq_data(irq);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000027
David Brazdil0f672f62019-12-10 10:32:29 +000028 d->mask = mask;
29 irq_set_chip_data(irq, (void *)(unsigned long)base);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000030}
31
David Brazdil0f672f62019-12-10 10:32:29 +000032static void iomd_irq_mask_ack(struct irq_data *d)
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000033{
David Brazdil0f672f62019-12-10 10:32:29 +000034 void __iomem *base = iomd_get_base(d);
35 unsigned int val, mask = d->mask;
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000036
David Brazdil0f672f62019-12-10 10:32:29 +000037 val = readb(base + MASK);
38 writeb(val & ~mask, base + MASK);
39 writeb(mask, base + CLR);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000040}
41
David Brazdil0f672f62019-12-10 10:32:29 +000042static void iomd_irq_mask(struct irq_data *d)
43{
44 void __iomem *base = iomd_get_base(d);
45 unsigned int val, mask = d->mask;
46
47 val = readb(base + MASK);
48 writeb(val & ~mask, base + MASK);
49}
50
51static void iomd_irq_unmask(struct irq_data *d)
52{
53 void __iomem *base = iomd_get_base(d);
54 unsigned int val, mask = d->mask;
55
56 val = readb(base + MASK);
57 writeb(val | mask, base + MASK);
58}
59
60static struct irq_chip iomd_chip_clr = {
61 .irq_mask_ack = iomd_irq_mask_ack,
62 .irq_mask = iomd_irq_mask,
63 .irq_unmask = iomd_irq_unmask,
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000064};
65
David Brazdil0f672f62019-12-10 10:32:29 +000066static struct irq_chip iomd_chip_noclr = {
67 .irq_mask = iomd_irq_mask,
68 .irq_unmask = iomd_irq_unmask,
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000069};
70
71extern unsigned char rpc_default_fiq_start, rpc_default_fiq_end;
72
73void __init rpc_init_irq(void)
74{
David Brazdil0f672f62019-12-10 10:32:29 +000075 unsigned int irq, clr, set;
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000076
77 iomd_writeb(0, IOMD_IRQMASKA);
78 iomd_writeb(0, IOMD_IRQMASKB);
79 iomd_writeb(0, IOMD_FIQMASK);
80 iomd_writeb(0, IOMD_DMAMASK);
81
82 set_fiq_handler(&rpc_default_fiq_start,
83 &rpc_default_fiq_end - &rpc_default_fiq_start);
84
85 for (irq = 0; irq < NR_IRQS; irq++) {
86 clr = IRQ_NOREQUEST;
David Brazdil0f672f62019-12-10 10:32:29 +000087 set = 0;
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000088
89 if (irq <= 6 || (irq >= 9 && irq <= 15))
90 clr |= IRQ_NOPROBE;
91
92 if (irq == 21 || (irq >= 16 && irq <= 19) ||
93 irq == IRQ_KEYBOARDTX)
94 set |= IRQ_NOAUTOEN;
95
96 switch (irq) {
97 case 0 ... 7:
David Brazdil0f672f62019-12-10 10:32:29 +000098 irq_set_chip_and_handler(irq, &iomd_chip_clr,
Andrew Scullb4b6d4a2019-01-02 15:54:55 +000099 handle_level_irq);
100 irq_modify_status(irq, clr, set);
David Brazdil0f672f62019-12-10 10:32:29 +0000101 iomd_set_base_mask(irq, IOMD_BASE + IOMD_IRQSTATA,
102 BIT(irq));
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000103 break;
104
105 case 8 ... 15:
David Brazdil0f672f62019-12-10 10:32:29 +0000106 irq_set_chip_and_handler(irq, &iomd_chip_noclr,
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000107 handle_level_irq);
108 irq_modify_status(irq, clr, set);
David Brazdil0f672f62019-12-10 10:32:29 +0000109 iomd_set_base_mask(irq, IOMD_BASE + IOMD_IRQSTATB,
110 BIT(irq - 8));
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000111 break;
112
113 case 16 ... 21:
David Brazdil0f672f62019-12-10 10:32:29 +0000114 irq_set_chip_and_handler(irq, &iomd_chip_noclr,
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000115 handle_level_irq);
116 irq_modify_status(irq, clr, set);
David Brazdil0f672f62019-12-10 10:32:29 +0000117 iomd_set_base_mask(irq, IOMD_BASE + IOMD_DMASTAT,
118 BIT(irq - 16));
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000119 break;
120
121 case 64 ... 71:
David Brazdil0f672f62019-12-10 10:32:29 +0000122 irq_set_chip(irq, &iomd_chip_noclr);
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000123 irq_modify_status(irq, clr, set);
David Brazdil0f672f62019-12-10 10:32:29 +0000124 iomd_set_base_mask(irq, IOMD_BASE + IOMD_FIQSTAT,
125 BIT(irq - 64));
Andrew Scullb4b6d4a2019-01-02 15:54:55 +0000126 break;
127 }
128 }
129
130 init_FIQ(FIQ_START);
131}