aboutsummaryrefslogtreecommitdiffstats
path: root/arch/m68knommu
diff options
context:
space:
mode:
authorGreg Ungerer <gerg@uclinux.org>2011-03-11 07:25:44 -0500
committerGreg Ungerer <gerg@uclinux.org>2011-03-15 07:01:57 -0400
commit47e0c7e128afb85cf4fb7792e6e7fcb91e2a5cc4 (patch)
treed3722b4ede246db0ba353eaa70c30b79c0e6cc3e /arch/m68knommu
parent57b481436f2a5580054784af8f044d2e3f602b53 (diff)
m68knommu: external interrupt support to ColdFire intc-simr controller
The EDGE Port module of some ColdFire parts using the intc-simr interrupt controller provides support for 7 external interrupts. These interrupts go off-chip (that is they are not for internal peripherals). They need some special handling and have some extra setup registers. Add code to support them. Signed-off-by: Greg Ungerer <gerg@uclinux.org>
Diffstat (limited to 'arch/m68knommu')
-rw-r--r--arch/m68knommu/platform/coldfire/intc-simr.c114
1 files changed, 110 insertions, 4 deletions
diff --git a/arch/m68knommu/platform/coldfire/intc-simr.c b/arch/m68knommu/platform/coldfire/intc-simr.c
index 8876f434c125..e642b24ab729 100644
--- a/arch/m68knommu/platform/coldfire/intc-simr.c
+++ b/arch/m68knommu/platform/coldfire/intc-simr.c
@@ -3,7 +3,7 @@
3 * 3 *
4 * Interrupt controller code for the ColdFire 5208, 5207 & 532x parts. 4 * Interrupt controller code for the ColdFire 5208, 5207 & 532x parts.
5 * 5 *
6 * (C) Copyright 2009, Greg Ungerer <gerg@snapgear.com> 6 * (C) Copyright 2009-2011, Greg Ungerer <gerg@snapgear.com>
7 * 7 *
8 * This file is subject to the terms and conditions of the GNU General Public 8 * This file is subject to the terms and conditions of the GNU General Public
9 * License. See the file COPYING in the main directory of this archive 9 * License. See the file COPYING in the main directory of this archive
@@ -21,6 +21,44 @@
21#include <asm/traps.h> 21#include <asm/traps.h>
22 22
23/* 23/*
24 * The EDGE Port interrupts are the fixed 7 external interrupts.
25 * They need some special treatment, for example they need to be acked.
26 */
27#ifdef CONFIG_M520x
28/*
29 * The 520x parts only support a limited range of these external
30 * interrupts, only 1, 4 and 7 (as interrupts 65, 66 and 67).
31 */
32#define EINT0 64 /* Is not actually used, but spot reserved for it */
33#define EINT1 65 /* EDGE Port interrupt 1 */
34#define EINT4 66 /* EDGE Port interrupt 4 */
35#define EINT7 67 /* EDGE Port interrupt 7 */
36
37static unsigned int irqebitmap[] = { 0, 1, 4, 7 };
38static unsigned int inline irq2ebit(unsigned int irq)
39{
40 return irqebitmap[irq - EINT0];
41}
42
43#else
44
45/*
46 * Most of the ColdFire parts with the EDGE Port module just have
47 * a strait direct mapping of the 7 external interrupts. Although
48 * there is a bit reserved for 0, it is not used.
49 */
50#define EINT0 64 /* Is not actually used, but spot reserved for it */
51#define EINT1 65 /* EDGE Port interrupt 1 */
52#define EINT7 71 /* EDGE Port interrupt 7 */
53
54static unsigned int inline irq2ebit(unsigned int irq)
55{
56 return irq - EINT0;
57}
58
59#endif
60
61/*
24 * There maybe one or two interrupt control units, each has 64 62 * There maybe one or two interrupt control units, each has 64
25 * interrupts. If there is no second unit then MCFINTC1_* defines 63 * interrupts. If there is no second unit then MCFINTC1_* defines
26 * will be 0 (and code for them optimized away). 64 * will be 0 (and code for them optimized away).
@@ -46,21 +84,86 @@ static void intc_irq_unmask(struct irq_data *d)
46 __raw_writeb(irq, MCFINTC0_CIMR); 84 __raw_writeb(irq, MCFINTC0_CIMR);
47} 85}
48 86
49static int intc_irq_set_type(struct irq_data *d, unsigned int type) 87static void intc_irq_ack(struct irq_data *d)
50{ 88{
51 unsigned int irq = d->irq - MCFINT_VECBASE; 89 unsigned int ebit = irq2ebit(d->irq);
52 90
91 __raw_writeb(0x1 << ebit, MCFEPORT_EPFR);
92}
93
94static unsigned int intc_irq_startup(struct irq_data *d)
95{
96 unsigned int irq = d->irq;
97
98 if ((irq >= EINT1) && (irq <= EINT7)) {
99 unsigned int ebit = irq2ebit(irq);
100 u8 v;
101
102 /* Set EPORT line as input */
103 v = __raw_readb(MCFEPORT_EPDDR);
104 __raw_writeb(v & ~(0x1 << ebit), MCFEPORT_EPDDR);
105
106 /* Set EPORT line as interrupt source */
107 v = __raw_readb(MCFEPORT_EPIER);
108 __raw_writeb(v | (0x1 << ebit), MCFEPORT_EPIER);
109 }
110
111 irq -= MCFINT_VECBASE;
53 if (MCFINTC1_ICR0 && (irq > 64)) 112 if (MCFINTC1_ICR0 && (irq > 64))
54 __raw_writeb(5, MCFINTC1_ICR0 + irq - 64); 113 __raw_writeb(5, MCFINTC1_ICR0 + irq - 64);
55 else 114 else
56 __raw_writeb(5, MCFINTC0_ICR0 + irq); 115 __raw_writeb(5, MCFINTC0_ICR0 + irq);
116
117
118 intc_irq_unmask(d);
119 return 0;
120}
121
122static int intc_irq_set_type(struct irq_data *d, unsigned int type)
123{
124 unsigned int ebit, irq = d->irq;
125 u16 pa, tb;
126
127 switch (type) {
128 case IRQ_TYPE_EDGE_RISING:
129 tb = 0x1;
130 break;
131 case IRQ_TYPE_EDGE_FALLING:
132 tb = 0x2;
133 break;
134 case IRQ_TYPE_EDGE_BOTH:
135 tb = 0x3;
136 break;
137 default:
138 /* Level triggered */
139 tb = 0;
140 break;
141 }
142
143 if (tb)
144 set_irq_handler(irq, handle_edge_irq);
145
146 ebit = irq2ebit(irq) * 2;
147 pa = __raw_readw(MCFEPORT_EPPAR);
148 pa = (pa & ~(0x3 << ebit)) | (tb << ebit);
149 __raw_writew(pa, MCFEPORT_EPPAR);
150
57 return 0; 151 return 0;
58} 152}
59 153
60static struct irq_chip intc_irq_chip = { 154static struct irq_chip intc_irq_chip = {
61 .name = "CF-INTC", 155 .name = "CF-INTC",
156 .irq_startup = intc_irq_startup,
157 .irq_mask = intc_irq_mask,
158 .irq_unmask = intc_irq_unmask,
159};
160
161static struct irq_chip intc_irq_chip_edge_port = {
162 .name = "CF-INTC-EP",
163 .irq_startup = intc_irq_startup,
62 .irq_mask = intc_irq_mask, 164 .irq_mask = intc_irq_mask,
63 .irq_unmask = intc_irq_unmask, 165 .irq_unmask = intc_irq_unmask,
166 .irq_ack = intc_irq_ack,
64 .irq_set_type = intc_irq_set_type, 167 .irq_set_type = intc_irq_set_type,
65}; 168};
66 169
@@ -77,7 +180,10 @@ void __init init_IRQ(void)
77 180
78 eirq = MCFINT_VECBASE + 64 + (MCFINTC1_ICR0 ? 64 : 0); 181 eirq = MCFINT_VECBASE + 64 + (MCFINTC1_ICR0 ? 64 : 0);
79 for (irq = MCFINT_VECBASE; (irq < eirq); irq++) { 182 for (irq = MCFINT_VECBASE; (irq < eirq); irq++) {
80 set_irq_chip(irq, &intc_irq_chip); 183 if ((irq >= EINT1) && (irq <= EINT7))
184 set_irq_chip(irq, &intc_irq_chip_edge_port);
185 else
186 set_irq_chip(irq, &intc_irq_chip);
81 set_irq_type(irq, IRQ_TYPE_LEVEL_HIGH); 187 set_irq_type(irq, IRQ_TYPE_LEVEL_HIGH);
82 set_irq_handler(irq, handle_level_irq); 188 set_irq_handler(irq, handle_level_irq);
83 } 189 }