| .. | .. |
|---|
| 8 | 8 | #include <asm/irq.h> |
|---|
| 9 | 9 | #include <asm/fiq.h> |
|---|
| 10 | 10 | |
|---|
| 11 | | -static void iomd_ack_irq_a(struct irq_data *d) |
|---|
| 12 | | -{ |
|---|
| 13 | | - unsigned int val, mask; |
|---|
| 11 | +// 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 |
|---|
| 14 | 16 | |
|---|
| 15 | | - mask = 1 << d->irq; |
|---|
| 16 | | - val = iomd_readb(IOMD_IRQMASKA); |
|---|
| 17 | | - iomd_writeb(val & ~mask, IOMD_IRQMASKA); |
|---|
| 18 | | - iomd_writeb(mask, IOMD_IRQCLRA); |
|---|
| 17 | +static 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; |
|---|
| 19 | 22 | } |
|---|
| 20 | 23 | |
|---|
| 21 | | -static void iomd_mask_irq_a(struct irq_data *d) |
|---|
| 24 | +static void iomd_set_base_mask(unsigned int irq, void __iomem *base, u32 mask) |
|---|
| 22 | 25 | { |
|---|
| 23 | | - unsigned int val, mask; |
|---|
| 26 | + struct irq_data *d = irq_get_irq_data(irq); |
|---|
| 24 | 27 | |
|---|
| 25 | | - mask = 1 << d->irq; |
|---|
| 26 | | - val = iomd_readb(IOMD_IRQMASKA); |
|---|
| 27 | | - iomd_writeb(val & ~mask, IOMD_IRQMASKA); |
|---|
| 28 | + d->mask = mask; |
|---|
| 29 | + irq_set_chip_data(irq, (void *)(unsigned long)base); |
|---|
| 28 | 30 | } |
|---|
| 29 | 31 | |
|---|
| 30 | | -static void iomd_unmask_irq_a(struct irq_data *d) |
|---|
| 32 | +static void iomd_irq_mask_ack(struct irq_data *d) |
|---|
| 31 | 33 | { |
|---|
| 32 | | - unsigned int val, mask; |
|---|
| 34 | + void __iomem *base = iomd_get_base(d); |
|---|
| 35 | + unsigned int val, mask = d->mask; |
|---|
| 33 | 36 | |
|---|
| 34 | | - mask = 1 << d->irq; |
|---|
| 35 | | - val = iomd_readb(IOMD_IRQMASKA); |
|---|
| 36 | | - iomd_writeb(val | mask, IOMD_IRQMASKA); |
|---|
| 37 | + val = readb(base + MASK); |
|---|
| 38 | + writeb(val & ~mask, base + MASK); |
|---|
| 39 | + writeb(mask, base + CLR); |
|---|
| 37 | 40 | } |
|---|
| 38 | 41 | |
|---|
| 39 | | -static struct irq_chip iomd_a_chip = { |
|---|
| 40 | | - .irq_ack = iomd_ack_irq_a, |
|---|
| 41 | | - .irq_mask = iomd_mask_irq_a, |
|---|
| 42 | | - .irq_unmask = iomd_unmask_irq_a, |
|---|
| 42 | +static 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 | + |
|---|
| 51 | +static 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 | + |
|---|
| 60 | +static 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, |
|---|
| 43 | 64 | }; |
|---|
| 44 | 65 | |
|---|
| 45 | | -static void iomd_mask_irq_b(struct irq_data *d) |
|---|
| 46 | | -{ |
|---|
| 47 | | - unsigned int val, mask; |
|---|
| 48 | | - |
|---|
| 49 | | - mask = 1 << (d->irq & 7); |
|---|
| 50 | | - val = iomd_readb(IOMD_IRQMASKB); |
|---|
| 51 | | - iomd_writeb(val & ~mask, IOMD_IRQMASKB); |
|---|
| 52 | | -} |
|---|
| 53 | | - |
|---|
| 54 | | -static void iomd_unmask_irq_b(struct irq_data *d) |
|---|
| 55 | | -{ |
|---|
| 56 | | - unsigned int val, mask; |
|---|
| 57 | | - |
|---|
| 58 | | - mask = 1 << (d->irq & 7); |
|---|
| 59 | | - val = iomd_readb(IOMD_IRQMASKB); |
|---|
| 60 | | - iomd_writeb(val | mask, IOMD_IRQMASKB); |
|---|
| 61 | | -} |
|---|
| 62 | | - |
|---|
| 63 | | -static struct irq_chip iomd_b_chip = { |
|---|
| 64 | | - .irq_ack = iomd_mask_irq_b, |
|---|
| 65 | | - .irq_mask = iomd_mask_irq_b, |
|---|
| 66 | | - .irq_unmask = iomd_unmask_irq_b, |
|---|
| 67 | | -}; |
|---|
| 68 | | - |
|---|
| 69 | | -static void iomd_mask_irq_dma(struct irq_data *d) |
|---|
| 70 | | -{ |
|---|
| 71 | | - unsigned int val, mask; |
|---|
| 72 | | - |
|---|
| 73 | | - mask = 1 << (d->irq & 7); |
|---|
| 74 | | - val = iomd_readb(IOMD_DMAMASK); |
|---|
| 75 | | - iomd_writeb(val & ~mask, IOMD_DMAMASK); |
|---|
| 76 | | -} |
|---|
| 77 | | - |
|---|
| 78 | | -static void iomd_unmask_irq_dma(struct irq_data *d) |
|---|
| 79 | | -{ |
|---|
| 80 | | - unsigned int val, mask; |
|---|
| 81 | | - |
|---|
| 82 | | - mask = 1 << (d->irq & 7); |
|---|
| 83 | | - val = iomd_readb(IOMD_DMAMASK); |
|---|
| 84 | | - iomd_writeb(val | mask, IOMD_DMAMASK); |
|---|
| 85 | | -} |
|---|
| 86 | | - |
|---|
| 87 | | -static struct irq_chip iomd_dma_chip = { |
|---|
| 88 | | - .irq_ack = iomd_mask_irq_dma, |
|---|
| 89 | | - .irq_mask = iomd_mask_irq_dma, |
|---|
| 90 | | - .irq_unmask = iomd_unmask_irq_dma, |
|---|
| 91 | | -}; |
|---|
| 92 | | - |
|---|
| 93 | | -static void iomd_mask_irq_fiq(struct irq_data *d) |
|---|
| 94 | | -{ |
|---|
| 95 | | - unsigned int val, mask; |
|---|
| 96 | | - |
|---|
| 97 | | - mask = 1 << (d->irq & 7); |
|---|
| 98 | | - val = iomd_readb(IOMD_FIQMASK); |
|---|
| 99 | | - iomd_writeb(val & ~mask, IOMD_FIQMASK); |
|---|
| 100 | | -} |
|---|
| 101 | | - |
|---|
| 102 | | -static void iomd_unmask_irq_fiq(struct irq_data *d) |
|---|
| 103 | | -{ |
|---|
| 104 | | - unsigned int val, mask; |
|---|
| 105 | | - |
|---|
| 106 | | - mask = 1 << (d->irq & 7); |
|---|
| 107 | | - val = iomd_readb(IOMD_FIQMASK); |
|---|
| 108 | | - iomd_writeb(val | mask, IOMD_FIQMASK); |
|---|
| 109 | | -} |
|---|
| 110 | | - |
|---|
| 111 | | -static struct irq_chip iomd_fiq_chip = { |
|---|
| 112 | | - .irq_ack = iomd_mask_irq_fiq, |
|---|
| 113 | | - .irq_mask = iomd_mask_irq_fiq, |
|---|
| 114 | | - .irq_unmask = iomd_unmask_irq_fiq, |
|---|
| 66 | +static struct irq_chip iomd_chip_noclr = { |
|---|
| 67 | + .irq_mask = iomd_irq_mask, |
|---|
| 68 | + .irq_unmask = iomd_irq_unmask, |
|---|
| 115 | 69 | }; |
|---|
| 116 | 70 | |
|---|
| 117 | 71 | extern unsigned char rpc_default_fiq_start, rpc_default_fiq_end; |
|---|
| .. | .. |
|---|
| 141 | 95 | |
|---|
| 142 | 96 | switch (irq) { |
|---|
| 143 | 97 | case 0 ... 7: |
|---|
| 144 | | - irq_set_chip_and_handler(irq, &iomd_a_chip, |
|---|
| 98 | + irq_set_chip_and_handler(irq, &iomd_chip_clr, |
|---|
| 145 | 99 | handle_level_irq); |
|---|
| 146 | 100 | irq_modify_status(irq, clr, set); |
|---|
| 101 | + iomd_set_base_mask(irq, IOMD_BASE + IOMD_IRQSTATA, |
|---|
| 102 | + BIT(irq)); |
|---|
| 147 | 103 | break; |
|---|
| 148 | 104 | |
|---|
| 149 | 105 | case 8 ... 15: |
|---|
| 150 | | - irq_set_chip_and_handler(irq, &iomd_b_chip, |
|---|
| 106 | + irq_set_chip_and_handler(irq, &iomd_chip_noclr, |
|---|
| 151 | 107 | handle_level_irq); |
|---|
| 152 | 108 | irq_modify_status(irq, clr, set); |
|---|
| 109 | + iomd_set_base_mask(irq, IOMD_BASE + IOMD_IRQSTATB, |
|---|
| 110 | + BIT(irq - 8)); |
|---|
| 153 | 111 | break; |
|---|
| 154 | 112 | |
|---|
| 155 | 113 | case 16 ... 21: |
|---|
| 156 | | - irq_set_chip_and_handler(irq, &iomd_dma_chip, |
|---|
| 114 | + irq_set_chip_and_handler(irq, &iomd_chip_noclr, |
|---|
| 157 | 115 | handle_level_irq); |
|---|
| 158 | 116 | irq_modify_status(irq, clr, set); |
|---|
| 117 | + iomd_set_base_mask(irq, IOMD_BASE + IOMD_DMASTAT, |
|---|
| 118 | + BIT(irq - 16)); |
|---|
| 159 | 119 | break; |
|---|
| 160 | 120 | |
|---|
| 161 | 121 | case 64 ... 71: |
|---|
| 162 | | - irq_set_chip(irq, &iomd_fiq_chip); |
|---|
| 122 | + irq_set_chip(irq, &iomd_chip_noclr); |
|---|
| 163 | 123 | irq_modify_status(irq, clr, set); |
|---|
| 124 | + iomd_set_base_mask(irq, IOMD_BASE + IOMD_FIQSTAT, |
|---|
| 125 | + BIT(irq - 64)); |
|---|
| 164 | 126 | break; |
|---|
| 165 | 127 | } |
|---|
| 166 | 128 | } |
|---|
| 167 | 129 | |
|---|
| 168 | 130 | init_FIQ(FIQ_START); |
|---|
| 169 | 131 | } |
|---|
| 170 | | - |
|---|