1 /* $NetBSD: pcib.c,v 1.12 2006/05/12 10:58:12 tsutsui Exp $ */
4 * Copyright 2002 Wasabi Systems, Inc.
7 * Written by Simon Burge for Wasabi Systems, Inc.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 * must display the following acknowledgement:
19 * This product includes software developed for the NetBSD Project by
20 * Wasabi Systems, Inc.
21 * 4. The name of Wasabi Systems, Inc. may not be used to endorse
22 * or promote products derived from this software without specific prior
25 * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
27 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
28 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
29 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
30 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
31 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
32 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
33 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
34 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
35 * POSSIBILITY OF SUCH DAMAGE.
38 #include <sys/cdefs.h>
39 __KERNEL_RCSID(0, "$NetBSD: pcib.c,v 1.12 2006/05/12 10:58:12 tsutsui Exp $");
41 #include <sys/param.h>
42 #include <sys/systm.h>
43 #include <sys/kernel.h>
44 #include <sys/device.h>
45 #include <sys/malloc.h>
47 #include <machine/bus.h>
48 #include <evbmips/malta/maltareg.h>
49 #include <evbmips/malta/maltavar.h>
50 #include <evbmips/malta/dev/gtreg.h>
51 #include <evbmips/malta/pci/pcibvar.h>
53 #include <dev/isa/isareg.h>
54 #include <dev/isa/isavar.h>
56 #include <dev/pci/pcireg.h>
57 #include <dev/pci/pcivar.h>
58 #include <dev/pci/pcidevs.h>
60 #include <dev/ic/i8259reg.h>
63 #define ICU_LEN 16 /* number of ISA IRQs */
65 const char *isa_intrnames
[ICU_LEN
] = {
68 "reserved", /* by South Bridge (for cascading) */
76 "pci A,B", /* PCI slots 1..4, ethernet */
77 "pci C,D", /* PCI slots 1..4, audio, usb */
81 "ide secondary", /* and compact flash connector */
84 struct pcib_intrhead
{
85 LIST_HEAD(, evbmips_intrhand
) intr_q
;
86 struct evcnt intr_count
;
93 bus_space_tag_t sc_iot
;
94 bus_space_handle_t sc_ioh_icu1
;
95 bus_space_handle_t sc_ioh_icu2
;
96 bus_space_handle_t sc_ioh_elcr
;
98 struct mips_isa_chipset sc_ic
;
100 struct pcib_intrhead sc_intrtab
[ICU_LEN
];
105 u_int16_t sc_reserved
;
112 * There is only one pci-isa bridge, and all external interrupts
113 * are routed through it, so we need to remember the softc when
114 * called from other interrupt handling code.
116 static struct pcib_softc
*my_sc
;
117 struct mips_isa_chipset
*pcib_ic
;
119 static int pcib_match(struct device
*, struct cfdata
*, void *);
120 static void pcib_attach(struct device
*, struct device
*, void *);
121 static int pcib_intr(void *v
);
122 static void pcib_bridge_callback(struct device
*);
123 static void pcib_set_icus(struct pcib_softc
*sc
);
124 static void pcib_cleanup(void *arg
);
126 static const struct evcnt
*
127 pcib_isa_intr_evcnt(void *, int);
128 static void *pcib_isa_intr_establish(void *, int, int, int,
129 int (*)(void *), void *);
130 static void pcib_isa_intr_disestablish(void *, void *);
131 static void pcib_isa_attach_hook(struct device
*, struct device
*,
132 struct isabus_attach_args
*);
133 static void pcib_isa_detach_hook(isa_chipset_tag_t
, device_t
);
134 static int pcib_isa_intr_alloc(void *, int, int, int *);
136 pcib_isa_intr_string(void *, int);
138 CFATTACH_DECL(pcib
, sizeof(struct pcib_softc
),
139 pcib_match
, pcib_attach
, NULL
, NULL
);
142 pcib_match(struct device
*parent
, struct cfdata
*match
, void *aux
)
144 struct pci_attach_args
*pa
= aux
;
146 if (PCI_CLASS(pa
->pa_class
) == PCI_CLASS_BRIDGE
&&
147 PCI_SUBCLASS(pa
->pa_class
) == PCI_SUBCLASS_BRIDGE_ISA
)
154 pcib_attach(struct device
*parent
, struct device
*self
, void *aux
)
156 struct pci_attach_args
*pa
= aux
;
163 panic("pcib_attach: already attached!");
164 my_sc
= (void *)self
;
167 * Just print out a description and defer configuration
168 * until all PCI devices have been attached.
170 pci_devinfo(pa
->pa_id
, pa
->pa_class
, 0, devinfo
, sizeof(devinfo
));
171 printf("%s: %s, (rev . 0x%02x)\n", self
->dv_xname
, devinfo
,
172 PCI_REVISION(pa
->pa_class
));
174 my_sc
->sc_iot
= pa
->pa_iot
;
177 * Map the PIC/ELCR registers.
179 if (bus_space_map(my_sc
->sc_iot
, 0x4d0, 2, 0, &my_sc
->sc_ioh_elcr
) != 0)
180 printf("%s: unable to map ELCR registers\n",
181 my_sc
->sc_dev
.dv_xname
);
182 if (bus_space_map(my_sc
->sc_iot
, IO_ICU1
, 2, 0, &my_sc
->sc_ioh_icu1
) != 0)
183 printf("%s: unable to map ICU1 registers\n",
184 my_sc
->sc_dev
.dv_xname
);
185 if (bus_space_map(my_sc
->sc_iot
, IO_ICU2
, 2, 0, &my_sc
->sc_ioh_icu2
) != 0)
186 printf("%s: unable to map ICU2 registers\n",
187 my_sc
->sc_dev
.dv_xname
);
189 /* All interrupts default to "masked off". */
190 my_sc
->sc_imask
= 0xffff;
192 /* All interrupts default to edge-triggered. */
196 * Initialize the 8259s.
198 /* reset, program device, 4 bytes */
199 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_ICW1
,
200 ICW1_SELECT
| ICW1_IC4
);
201 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_ICW2
,
202 ICW2_VECTOR(0)/*XXX*/);
203 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_ICW3
,
205 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_ICW4
,
208 /* mask all interrupts */
209 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_OCW1
,
210 my_sc
->sc_imask
& 0xff);
212 /* enable special mask mode */
213 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_OCW3
,
214 OCW3_SELECT
| OCW3_SSMM
| OCW3_SMM
);
216 /* read IRR by default */
217 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu1
, PIC_OCW3
,
218 OCW3_SELECT
| OCW3_RR
);
220 /* reset, program device, 4 bytes */
221 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_ICW1
,
222 ICW1_SELECT
| ICW1_IC4
);
223 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_ICW2
,
224 ICW2_VECTOR(0)/*XXX*/);
225 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_ICW3
,
227 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_ICW4
,
230 /* mask all interrupts */
231 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_OCW1
,
232 my_sc
->sc_imask
& 0xff);
234 /* enable special mask mode */
235 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_OCW3
,
236 OCW3_SELECT
| OCW3_SSMM
| OCW3_SMM
);
238 /* read IRR by default */
239 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_icu2
, PIC_OCW3
,
240 OCW3_SELECT
| OCW3_RR
);
243 * Default all interrupts to edge-triggered.
245 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_elcr
, 0,
246 my_sc
->sc_elcr
& 0xff);
247 bus_space_write_1(my_sc
->sc_iot
, my_sc
->sc_ioh_elcr
, 1,
248 (my_sc
->sc_elcr
>> 8) & 0xff);
251 * Some ISA interrupts are reserved for devices that
252 * we know are hard-wired to certain IRQs.
255 (1U << 0) | /* timer */
256 (1U << 1) | /* keyboard controller (keyboard) */
257 (1U << 2) | /* PIC cascade */
258 (1U << 3) | /* COM 2 */
259 (1U << 4) | /* COM 1 */
260 (1U << 6) | /* floppy */
261 (1U << 7) | /* centronics */
262 (1U << 8) | /* RTC */
263 (1U << 9) | /* I2C */
264 (1U << 12) | /* keyboard controller (mouse) */
265 (1U << 14) | /* IDE primary */
266 (1U << 15); /* IDE secondary */
268 /* Set up our ISA chipset. */
269 my_sc
->sc_ic
.ic_v
= my_sc
;
270 my_sc
->sc_ic
.ic_intr_evcnt
= pcib_isa_intr_evcnt
;
271 my_sc
->sc_ic
.ic_intr_establish
= pcib_isa_intr_establish
;
272 my_sc
->sc_ic
.ic_intr_disestablish
= pcib_isa_intr_disestablish
;
273 my_sc
->sc_ic
.ic_intr_alloc
= pcib_isa_intr_alloc
;
274 my_sc
->sc_ic
.ic_intr_string
= pcib_isa_intr_string
;
276 pcib_ic
= &my_sc
->sc_ic
; /* XXX for external use */
278 /* Initialize our interrupt table. */
279 for (i
= 0; i
< ICU_LEN
; i
++) {
281 char irqstr
[8]; /* 4 + 2 + NULL + sanity */
283 sprintf(irqstr
, "irq %d", i
);
284 evcnt_attach_dynamic(&my_sc
->sc_intrtab
[i
].intr_count
,
285 EVCNT_TYPE_INTR
, NULL
, "pcib", irqstr
);
287 evcnt_attach_dynamic(&my_sc
->sc_intrtab
[i
].intr_count
,
288 EVCNT_TYPE_INTR
, NULL
, "pcib", isa_intrnames
[i
]);
290 LIST_INIT(&my_sc
->sc_intrtab
[i
].intr_q
);
291 my_sc
->sc_intrtab
[i
].intr_type
= IST_NONE
;
294 /* Hook up our interrupt handler. */
295 my_sc
->sc_ih
= evbmips_intr_establish(MALTA_SOUTHBRIDGE_INTR
, pcib_intr
, my_sc
);
296 if (my_sc
->sc_ih
== NULL
)
297 printf("%s: WARNING: unable to register interrupt handler\n",
298 my_sc
->sc_dev
.dv_xname
);
302 * Disable ISA interrupts before returning to YAMON.
304 if (shutdownhook_establish(pcib_cleanup
, my_sc
) == NULL
)
305 panic("pcib_attach: could not establish shutdown hook");
307 config_defer(self
, pcib_bridge_callback
);
311 pcib_bridge_callback(struct device
*self
)
313 struct pcib_softc
*sc
= (void *)self
;
314 struct malta_config
*mcp
= &malta_configuration
;
315 struct isabus_attach_args iba
;
318 * Attach the ISA bus behind this bridge.
320 memset(&iba
, 0, sizeof(iba
));
322 iba
.iba_iot
= &mcp
->mc_iot
;
323 iba
.iba_memt
= &mcp
->mc_memt
;
324 iba
.iba_dmat
= &mcp
->mc_isa_dmat
;
326 iba
.iba_ic
= &sc
->sc_ic
;
327 iba
.iba_ic
->ic_attach_hook
= pcib_isa_attach_hook
;
328 iba
.iba_ic
->ic_detach_hook
= pcib_isa_detach_hook
;
330 config_found_ia(&sc
->sc_dev
, "isabus", &iba
, isabusprint
);
334 pcib_isa_attach_hook(struct device
*parent
, struct device
*self
,
335 struct isabus_attach_args
*iba
)
342 pcib_isa_detach_hook(isa_chipset_tag_t ic
, device_t self
)
349 pcib_set_icus(struct pcib_softc
*sc
)
352 /* Enable the cascade IRQ (2) if 8-15 is enabled. */
353 if ((sc
->sc_imask
& 0xff00) != 0xff00)
354 sc
->sc_imask
&= ~(1U << 2);
356 sc
->sc_imask
|= (1U << 2);
358 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu1
, PIC_OCW1
,
359 sc
->sc_imask
& 0xff);
360 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu2
, PIC_OCW1
,
361 (sc
->sc_imask
>> 8) & 0xff);
363 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_elcr
, 0,
365 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_elcr
, 1,
366 (sc
->sc_elcr
>> 8) & 0xff);
372 struct pcib_softc
*sc
= v
;
373 struct evbmips_intrhand
*ih
;
378 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu1
, PIC_OCW3
,
379 OCW3_SELECT
| OCW3_POLL
);
380 irq
= bus_space_read_1(sc
->sc_iot
, sc
->sc_ioh_icu1
, PIC_OCW3
);
381 if ((irq
& OCW3_POLL_PENDING
) == 0)
384 irq
= OCW3_POLL_IRQ(irq
);
387 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu2
,
388 PIC_OCW3
, OCW3_SELECT
| OCW3_POLL
);
389 irq
= bus_space_read_1(sc
->sc_iot
, sc
->sc_ioh_icu2
,
391 if (irq
& OCW3_POLL_PENDING
)
392 irq
= OCW3_POLL_IRQ(irq
) + 8;
397 /* XXX - should be a function call to gt.c? */
398 irq
= GT_REGVAL(GT_PCI0_INTR_ACK
) & 0xff;
401 * From YAMON source code:
403 * IRQ7 is used to detect spurious interrupts.
404 * The interrupt acknowledge cycle returns IRQ7, if no
405 * interrupts is requested.
406 * We can differentiate between this situation and a
407 * "Normal" IRQ7 by reading the ISR.
413 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu1
, PIC_OCW3
,
414 OCW3_SELECT
| OCW3_RR
| OCW3_RIS
);
415 reg
= bus_space_read_1(sc
->sc_iot
, sc
->sc_ioh_icu1
,
417 if (!(reg
& (1 << 7)))
418 break; /* spurious interrupt */
422 sc
->sc_intrtab
[irq
].intr_count
.ev_count
++;
423 LIST_FOREACH(ih
, &sc
->sc_intrtab
[irq
].intr_q
, ih_q
)
424 (*ih
->ih_func
)(ih
->ih_arg
);
426 /* Send a specific EOI to the 8259. */
428 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu2
,
429 PIC_OCW2
, OCW2_SELECT
| OCW2_EOI
| OCW2_SL
|
434 bus_space_write_1(sc
->sc_iot
, sc
->sc_ioh_icu1
, PIC_OCW2
,
435 OCW2_SELECT
| OCW2_EOI
| OCW2_SL
| OCW2_ILS(irq
));
440 pcib_isa_intr_string(void *v
, int irq
)
442 static char irqstr
[12]; /* 8 + 2 + NULL + sanity */
444 if (irq
== 0 || irq
>= ICU_LEN
|| irq
== 2)
445 panic("pcib_isa_intr_string: bogus isa irq 0x%x", irq
);
447 sprintf(irqstr
, "isa irq %d", irq
);
452 pcib_isa_intr_evcnt(void *v
, int irq
)
455 if (irq
== 0 || irq
>= ICU_LEN
|| irq
== 2)
456 panic("pcib_isa_intr_evcnt: bogus isa irq 0x%x", irq
);
458 return (&my_sc
->sc_intrtab
[irq
].intr_count
);
462 pcib_isa_intr_establish(void *v
, int irq
, int type
, int level
,
463 int (*func
)(void *), void *arg
)
465 struct evbmips_intrhand
*ih
;
468 if (irq
>= ICU_LEN
|| irq
== 2 || type
== IST_NONE
)
469 panic("pcib_isa_intr_establish: bad irq or type");
471 switch (my_sc
->sc_intrtab
[irq
].intr_type
) {
473 my_sc
->sc_intrtab
[irq
].intr_type
= type
;
478 if (type
== my_sc
->sc_intrtab
[irq
].intr_type
)
483 * We can't share interrupts in this case.
488 ih
= malloc(sizeof(*ih
), M_DEVBUF
, M_NOWAIT
);
498 /* Insert the handler into the table. */
499 LIST_INSERT_HEAD(&my_sc
->sc_intrtab
[irq
].intr_q
, ih
, ih_q
);
500 my_sc
->sc_intrtab
[irq
].intr_type
= type
;
502 /* Enable it, set trigger mode. */
503 my_sc
->sc_imask
&= ~(1 << irq
);
504 if (my_sc
->sc_intrtab
[irq
].intr_type
== IST_LEVEL
)
505 my_sc
->sc_elcr
|= (1 << irq
);
507 my_sc
->sc_elcr
&= ~(1 << irq
);
509 pcib_set_icus(my_sc
);
517 pcib_isa_intr_disestablish(void *v
, void *arg
)
519 struct evbmips_intrhand
*ih
= arg
;
524 LIST_REMOVE(ih
, ih_q
);
526 /* If there are no more handlers on this IRQ, disable it. */
527 if (LIST_FIRST(&my_sc
->sc_intrtab
[ih
->ih_irq
].intr_q
) == NULL
) {
528 my_sc
->sc_imask
|= (1 << ih
->ih_irq
);
529 pcib_set_icus(my_sc
);
538 pcib_isa_intr_alloc(void *v
, int mask
, int type
, int *irq
)
540 int i
, tmp
, bestirq
, count
;
541 struct evbmips_intrhand
*ih
;
543 if (type
== IST_NONE
)
544 panic("pcib_intr_alloc: bogus type");
549 mask
&= ~my_sc
->sc_reserved
;
551 for (i
= 0; i
< ICU_LEN
; i
++) {
552 if ((mask
& (1 << i
)) == 0)
555 switch (my_sc
->sc_intrtab
[i
].intr_type
) {
558 * If nothing's using the IRQ, just return it.
565 if (type
!= my_sc
->sc_intrtab
[i
].intr_type
)
568 * If the IRQ is sharable, count the number of
569 * other handlers, and if it's smaller than the
570 * last IRQ like this, remember it.
573 for (ih
= LIST_FIRST(&my_sc
->sc_intrtab
[i
].intr_q
);
574 ih
!= NULL
; ih
= LIST_NEXT(ih
, ih_q
))
576 if (bestirq
== -1 || count
> tmp
) {
583 /* This just isn't sharable. */
596 pcib_cleanup(void *arg
)
599 my_sc
->sc_imask
= 0xffff;
600 pcib_set_icus(my_sc
);