intr.c revision 1.5 1 /* $NetBSD: intr.c,v 1.5 2012/01/21 19:44:30 nonaka Exp $ */
2
3 /*-
4 * Copyright (C) 2005 NONAKA Kimihiro <nonaka (at) netbsd.org>
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28 #include <sys/cdefs.h>
29 __KERNEL_RCSID(0, "$NetBSD: intr.c,v 1.5 2012/01/21 19:44:30 nonaka Exp $");
30
31 #include <sys/param.h>
32 #include <sys/systm.h>
33 #include <sys/kernel.h>
34 #include <sys/malloc.h>
35 #include <sys/device.h>
36
37 #include <sh3/exception.h>
38
39 #include <machine/intr.h>
40
41 #define _N_EXTINTR 8
42
43 #define LANDISK_INTEN 0xb0000005
44 #define INTEN_ALL_MASK 0x00
45
46 struct intrhand {
47 int (*ih_fun)(void *);
48 void *ih_arg;
49 struct intrhand *ih_next;
50 int ih_enable;
51 int ih_level;
52 int ih_irq;
53 struct evcnt ih_evcnt;
54 };
55
56 struct extintr_handler {
57 int (*eih_func)(void *eih_arg);
58 void *eih_arg;
59 struct intrhand *eih_ih;
60 int eih_nih;
61 };
62
63 static struct extintr_handler extintr_handler[_N_EXTINTR];
64
65 static const char *extintr_names[_N_EXTINTR] = {
66 "irq5", "irq6", "irq7", "irq8",
67 "irq9", "irq10", "irq11", "irq12"
68 };
69
70 static int fakeintr(void *arg);
71 static int extintr_intr_handler(void *arg);
72
73 void
74 intc_intr(int ssr, int spc, int ssp)
75 {
76 struct intc_intrhand *ih;
77 struct clockframe cf;
78 int evtcode;
79
80 curcpu()->ci_data.cpu_nintr++;
81
82 evtcode = _reg_read_4(SH4_INTEVT);
83 ih = EVTCODE_IH(evtcode);
84 KDASSERT(ih->ih_func);
85
86 switch (evtcode) {
87 #if 0
88 #define IRL(irq) (0x200 + ((irq) << 5))
89 case IRL(5): case IRL(6): case IRL(7): case IRL(8):
90 case IRL(9): case IRL(10): case IRL(11): case IRL(12):
91 {
92 int level;
93 uint8_t inten, bit;
94
95 bit = 1 << (EVTCODE_TO_MAP_INDEX(evtcode) - 5);
96 inten = _reg_read_1(LANDISK_INTEN);
97 _reg_write_1(LANDISK_INTEN, inten & ~bit);
98 level = (_IPL_NSOFT + 1) << 4; /* disable softintr */
99 ssr &= 0xf0;
100 if (level < ssr)
101 level = ssr;
102 (void)_cpu_intr_resume(level);
103 (*ih->ih_func)(ih->ih_arg);
104 _reg_write_1(LANDISK_INTEN, inten);
105 break;
106 }
107 #endif
108 default:
109 (void)_cpu_intr_resume(ih->ih_level);
110 (*ih->ih_func)(ih->ih_arg);
111 break;
112
113 case SH_INTEVT_TMU0_TUNI0:
114 (void)_cpu_intr_resume(ih->ih_level);
115 cf.spc = spc;
116 cf.ssr = ssr;
117 cf.ssp = ssp;
118 (*ih->ih_func)(&cf);
119 break;
120
121 case SH_INTEVT_NMI:
122 printf("NMI ignored.\n");
123 break;
124 }
125 }
126
127 void
128 intr_init(void)
129 {
130
131 _reg_write_1(LANDISK_INTEN, INTEN_ALL_MASK);
132 }
133
134 void *
135 extintr_establish(int irq, int level, int (*ih_fun)(void *), void *ih_arg)
136 {
137 static struct intrhand fakehand = {fakeintr};
138 struct extintr_handler *eih;
139 struct intrhand **p, *q, *ih;
140 const char *name;
141 int evtcode;
142 int s;
143
144 KDASSERT(irq >= 5 && irq <= 12);
145
146 ih = malloc(sizeof(*ih), M_DEVBUF, cold ? M_NOWAIT : M_WAITOK);
147 if (ih == NULL)
148 panic("intr_establish: can't malloc handler info");
149
150 s = _cpu_intr_suspend();
151
152 switch (level) {
153 default:
154 #if defined(DEBUG)
155 panic("extintr_establish: unknown level %d", level);
156 /*NOTREACHED*/
157 #endif
158 case IPL_VM:
159 break;
160 }
161
162 eih = &extintr_handler[irq - 5];
163 if (eih->eih_func == NULL) {
164 evtcode = 0x200 + (irq << 5);
165 eih->eih_func = intc_intr_establish(evtcode, IST_LEVEL, level,
166 extintr_intr_handler, eih);
167 }
168
169 /*
170 * Figure out where to put the handler.
171 * This is O(N^2), but we want to preserve the order, and N is
172 * generally small.
173 */
174 for (p = &eih->eih_ih; (q = *p) != NULL; p = &q->ih_next)
175 continue;
176
177 /*
178 * Actually install a fake handler momentarily, since we might be doing
179 * this with interrupts enabled and don't want the real routine called
180 * until masking is set up.
181 */
182 fakehand.ih_level = level;
183 *p = &fakehand;
184
185 /*
186 * Poke the real handler in now.
187 */
188 memset(ih, 0, sizeof(*ih));
189 ih->ih_fun = ih_fun;
190 ih->ih_arg = ih_arg;
191 ih->ih_next = NULL;
192 ih->ih_enable = 1;
193 ih->ih_level = level;
194 ih->ih_irq = irq - 5;
195 name = extintr_names[irq - 5];
196 evcnt_attach_dynamic(&ih->ih_evcnt, EVCNT_TYPE_INTR,
197 NULL, "ext", name);
198 *p = ih;
199
200 if (++eih->eih_nih == 1) {
201 /* Unmask interrupt */
202 _reg_bset_1(LANDISK_INTEN, (1 << (irq - 5)));
203 }
204
205 splx(s);
206
207 return (ih);
208 }
209
210 void
211 extintr_disestablish(void *aux)
212 {
213 struct intrhand *ih = aux;
214 struct intrhand **p, *q;
215 struct extintr_handler *eih;
216 int irq;
217 int s;
218
219 KDASSERT(ih != NULL);
220
221 s = _cpu_intr_suspend();
222
223 irq = ih->ih_irq;
224 eih = &extintr_handler[irq];
225
226 /*
227 * Remove the handler from the chain.
228 * This is O(n^2), too.
229 */
230 for (p = &eih->eih_ih; (q = *p) != NULL && q != ih; p = &q->ih_next)
231 continue;
232 if (q == NULL)
233 panic("extintr_disestablish: handler not registered");
234
235 *p = q->ih_next;
236
237 evcnt_detach(&ih->ih_evcnt);
238
239 free((void *)ih, M_DEVBUF);
240
241 if (--eih->eih_nih == 0) {
242 intc_intr_disestablish(eih->eih_func);
243
244 /* Mask interrupt */
245 _reg_bclr_1(LANDISK_INTEN, (1 << irq));
246 }
247
248 splx(s);
249 }
250
251 void
252 extintr_enable(void *aux)
253 {
254 struct intrhand *ih = aux;
255 struct intrhand *p, *q;
256 struct extintr_handler *eih;
257 int irq;
258 int cnt;
259 int s;
260
261 KDASSERT(ih != NULL);
262
263 s = _cpu_intr_suspend();
264
265 irq = ih->ih_irq;
266 KDASSERT(irq >= 0 && irq < 8);
267 eih = &extintr_handler[irq];
268 for (cnt = 0, p = eih->eih_ih, q = NULL; p != NULL; p = p->ih_next) {
269 if (p->ih_enable) {
270 cnt++;
271 }
272 if (p == ih) {
273 q = p;
274 p->ih_enable = 1;
275 }
276 }
277 KDASSERT(q != NULL);
278
279 if (cnt == 0) {
280 /* Unmask interrupt */
281 _reg_bset_1(LANDISK_INTEN, (1 << irq));
282 }
283
284 splx(s);
285 }
286
287 void
288 extintr_disable(void *aux)
289 {
290 struct intrhand *ih = aux;
291 struct intrhand *p, *q;
292 struct extintr_handler *eih;
293 int irq;
294 int cnt;
295 int s;
296
297 KDASSERT(ih != NULL);
298
299 s = _cpu_intr_suspend();
300
301 irq = ih->ih_irq;
302 KDASSERT(irq >= 0 && irq < 8);
303 eih = &extintr_handler[irq];
304 for (cnt = 0, p = eih->eih_ih, q = NULL; p != NULL; p = p->ih_next) {
305 if (p == ih) {
306 q = p;
307 p->ih_enable = 0;
308 }
309 if (!ih->ih_enable) {
310 cnt++;
311 }
312 }
313 KDASSERT(q != NULL);
314
315 if (cnt == 0) {
316 /* Mask interrupt */
317 _reg_bclr_1(LANDISK_INTEN, (1 << irq));
318 }
319
320 splx(s);
321 }
322
323 void
324 extintr_disable_by_num(int irq)
325 {
326 struct extintr_handler *eih;
327 struct intrhand *ih;
328 int s;
329
330 KDASSERT(irq >= 5 && irq <= 12);
331
332 s = _cpu_intr_suspend();
333 eih = &extintr_handler[irq - 5];
334 for (ih = eih->eih_ih; ih != NULL; ih = ih->ih_next) {
335 ih->ih_enable = 0;
336 }
337 /* Mask interrupt */
338 _reg_bclr_1(LANDISK_INTEN, (1 << irq));
339 splx(s);
340 }
341
342 static int
343 fakeintr(void *arg)
344 {
345
346 return 0;
347 }
348
349 static int
350 extintr_intr_handler(void *arg)
351 {
352 struct extintr_handler *eih = arg;
353 struct intrhand *ih;
354 int r;
355
356 if (__predict_true(eih != NULL)) {
357 for (ih = eih->eih_ih; ih != NULL; ih = ih->ih_next) {
358 if (__predict_true(ih->ih_enable)) {
359 r = (*ih->ih_fun)(ih->ih_arg);
360 if (__predict_true(r != 0)) {
361 ih->ih_evcnt.ev_count++;
362 }
363 }
364 }
365 return 1;
366 }
367 return 0;
368 }
369