454af1c0e8
There are still about 1600 left, but they have ',' or /* ... */ in the actual variable definitions - which my awk script doesn't handle. There are also many that need () -> (void). (The script does handle misordered arguments.)
565 lines
14 KiB
C
565 lines
14 KiB
C
/* $NetBSD: pcib.c,v 1.20 2009/03/14 15:35:59 dsl Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2000, 2001 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Jason R. Thorpe.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h> /* RCS ID & Copyright macro defns */
|
|
|
|
__KERNEL_RCSID(0, "$NetBSD: pcib.c,v 1.20 2009/03/14 15:35:59 dsl Exp $");
|
|
|
|
#include "opt_algor_p5064.h"
|
|
#include "opt_algor_p6032.h"
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/device.h>
|
|
#include <sys/malloc.h>
|
|
|
|
#include <machine/intr.h>
|
|
#include <machine/bus.h>
|
|
|
|
#include <dev/isa/isareg.h>
|
|
#include <dev/isa/isavar.h>
|
|
|
|
#include <dev/pci/pcireg.h>
|
|
#include <dev/pci/pcivar.h>
|
|
#include <dev/pci/pcidevs.h>
|
|
|
|
#include <dev/ic/i8259reg.h>
|
|
|
|
#ifdef ALGOR_P5064
|
|
#include <algor/algor/algor_p5064var.h>
|
|
#endif
|
|
|
|
#ifdef ALGOR_P6032
|
|
#include <algor/algor/algor_p6032var.h>
|
|
#endif
|
|
|
|
const char *pcib_intrnames[16] = {
|
|
"irq 0",
|
|
"irq 1",
|
|
"irq 2",
|
|
"irq 3",
|
|
"irq 4",
|
|
"irq 5",
|
|
"irq 6",
|
|
"irq 7",
|
|
"irq 8",
|
|
"irq 9",
|
|
"irq 10",
|
|
"irq 11",
|
|
"irq 12",
|
|
"irq 13",
|
|
"irq 14",
|
|
"irq 15",
|
|
};
|
|
|
|
struct pcib_intrhead {
|
|
LIST_HEAD(, algor_intrhand) intr_q;
|
|
struct evcnt intr_count;
|
|
int intr_type;
|
|
};
|
|
|
|
struct pcib_softc {
|
|
struct device sc_dev;
|
|
|
|
bus_space_tag_t sc_iot;
|
|
bus_space_handle_t sc_ioh_icu1;
|
|
bus_space_handle_t sc_ioh_icu2;
|
|
bus_space_handle_t sc_ioh_elcr;
|
|
|
|
struct algor_isa_chipset sc_ic;
|
|
|
|
struct pcib_intrhead sc_intrtab[16];
|
|
|
|
u_int16_t sc_imask;
|
|
u_int16_t sc_elcr;
|
|
|
|
#if defined(ALGOR_P5064)
|
|
isa_chipset_tag_t sc_parent_ic;
|
|
#endif
|
|
|
|
u_int16_t sc_reserved;
|
|
|
|
void *sc_ih;
|
|
};
|
|
|
|
int pcib_match(struct device *, struct cfdata *, void *);
|
|
void pcib_attach(struct device *, struct device *, void *);
|
|
|
|
CFATTACH_DECL(pcib, sizeof(struct pcib_softc),
|
|
pcib_match, pcib_attach, NULL, NULL);
|
|
|
|
void pcib_isa_attach_hook(struct device *, struct device *,
|
|
struct isabus_attach_args *);
|
|
|
|
int pcib_intr(void *);
|
|
|
|
void pcib_bridge_callback(struct device *);
|
|
|
|
const struct evcnt *pcib_isa_intr_evcnt(void *, int);
|
|
void *pcib_isa_intr_establish(void *, int, int, int,
|
|
int (*)(void *), void *);
|
|
void pcib_isa_intr_disestablish(void *, void *);
|
|
int pcib_isa_intr_alloc(void *, int, int, int *);
|
|
|
|
void pcib_set_icus(struct pcib_softc *);
|
|
|
|
int
|
|
pcib_match(struct device *parent, struct cfdata *match, void *aux)
|
|
{
|
|
struct pci_attach_args *pa = aux;
|
|
|
|
if (PCI_CLASS(pa->pa_class) == PCI_CLASS_BRIDGE &&
|
|
PCI_SUBCLASS(pa->pa_class) == PCI_SUBCLASS_BRIDGE_ISA)
|
|
return (1);
|
|
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
pcib_attach(struct device *parent, struct device *self, void *aux)
|
|
{
|
|
struct pcib_softc *sc = (void *) self;
|
|
struct pci_attach_args *pa = aux;
|
|
char devinfo[256];
|
|
int i;
|
|
|
|
pci_devinfo(pa->pa_id, pa->pa_class, 0, devinfo, sizeof(devinfo));
|
|
printf(": %s (rev. 0x%02x)\n", devinfo,
|
|
PCI_REVISION(pa->pa_class));
|
|
|
|
sc->sc_iot = pa->pa_iot;
|
|
|
|
/*
|
|
* Map the PIC/ELCR registers.
|
|
*/
|
|
if (bus_space_map(sc->sc_iot, 0x4d0, 2, 0, &sc->sc_ioh_elcr) != 0)
|
|
printf("%s: unable to map ELCR registers\n",
|
|
sc->sc_dev.dv_xname);
|
|
if (bus_space_map(sc->sc_iot, IO_ICU1, 2, 0, &sc->sc_ioh_icu1) != 0)
|
|
printf("%s: unable to map ICU1 registers\n",
|
|
sc->sc_dev.dv_xname);
|
|
if (bus_space_map(sc->sc_iot, IO_ICU2, 2, 0, &sc->sc_ioh_icu2) != 0)
|
|
printf("%s: unable to map ICU2 registers\n",
|
|
sc->sc_dev.dv_xname);
|
|
|
|
/* All interrupts default to "masked off". */
|
|
sc->sc_imask = 0xffff;
|
|
|
|
/* All interrupts default to edge-triggered. */
|
|
sc->sc_elcr = 0;
|
|
|
|
/*
|
|
* Initialize the 8259s.
|
|
*/
|
|
|
|
/* reset, program device, 4 bytes */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_ICW1,
|
|
ICW1_SELECT | ICW1_IC4);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_ICW2,
|
|
ICW2_VECTOR(0)/*XXX*/);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_ICW3,
|
|
ICW3_CASCADE(2));
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_ICW4,
|
|
ICW4_8086);
|
|
|
|
/* mask all interrupts */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW1,
|
|
sc->sc_imask & 0xff);
|
|
|
|
/* enable special mask mode */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW3,
|
|
OCW3_SELECT | OCW3_SSMM | OCW3_SMM);
|
|
|
|
/* read IRR by default */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW3,
|
|
OCW3_SELECT | OCW3_RR);
|
|
|
|
/* reset; program device, 4 bytes */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_ICW1,
|
|
ICW1_SELECT | ICW1_IC4);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_ICW2,
|
|
ICW2_VECTOR(0)/*XXX*/);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_ICW3,
|
|
ICW3_SIC(2));
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_ICW4,
|
|
ICW4_8086);
|
|
|
|
/* mask all interrupts */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_OCW1,
|
|
(sc->sc_imask >> 8) & 0xff);
|
|
|
|
/* enable special mask mode */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_OCW3,
|
|
OCW3_SELECT | OCW3_SSMM | OCW3_SMM);
|
|
|
|
/* read IRR by default */
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_OCW3,
|
|
OCW3_SELECT | OCW3_RR);
|
|
|
|
/*
|
|
* Default all interrupts to edge-triggered.
|
|
*/
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_elcr, 0,
|
|
sc->sc_elcr & 0xff);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_elcr, 1,
|
|
(sc->sc_elcr >> 8) & 0xff);
|
|
|
|
/*
|
|
* Some ISA interrupts are reserved for devices that
|
|
* we know are hard-wired to certain IRQs.
|
|
*/
|
|
sc->sc_reserved =
|
|
(1U << 0) | /* timer */
|
|
(1U << 1) | /* keyboard controller */
|
|
(1U << 2) | /* PIC cascade */
|
|
(1U << 3) | /* COM 2 */
|
|
(1U << 4) | /* COM 1 */
|
|
(1U << 6) | /* floppy */
|
|
(1U << 7) | /* centronics */
|
|
(1U << 8) | /* RTC */
|
|
(1U << 12) | /* keyboard controller */
|
|
(1U << 14) | /* IDE 0 */
|
|
(1U << 15); /* IDE 1 */
|
|
|
|
#if defined(ALGOR_P5064)
|
|
/*
|
|
* Some "ISA" interrupts are a little wacky, wired up directly
|
|
* to the P-5064 interrupt controller.
|
|
*/
|
|
sc->sc_parent_ic = &p5064_configuration.ac_ic;
|
|
#endif /* ALGOR_P5064 */
|
|
|
|
/* Set up our ISA chipset. */
|
|
sc->sc_ic.ic_v = sc;
|
|
sc->sc_ic.ic_intr_evcnt = pcib_isa_intr_evcnt;
|
|
sc->sc_ic.ic_intr_establish = pcib_isa_intr_establish;
|
|
sc->sc_ic.ic_intr_disestablish = pcib_isa_intr_disestablish;
|
|
sc->sc_ic.ic_intr_alloc = pcib_isa_intr_alloc;
|
|
|
|
/* Initialize our interrupt table. */
|
|
for (i = 0; i < 16; i++) {
|
|
LIST_INIT(&sc->sc_intrtab[i].intr_q);
|
|
evcnt_attach_dynamic(&sc->sc_intrtab[i].intr_count,
|
|
EVCNT_TYPE_INTR, NULL, "pcib", pcib_intrnames[i]);
|
|
sc->sc_intrtab[i].intr_type = IST_NONE;
|
|
}
|
|
|
|
/* Hook up our interrupt handler. */
|
|
#if defined(ALGOR_P5064)
|
|
sc->sc_ih = (*algor_intr_establish)(P5064_IRQ_ISABRIDGE,
|
|
pcib_intr, sc);
|
|
#elif defined(ALGOR_P6032)
|
|
sc->sc_ih = (*algor_intr_establish)(P6032_IRQ_ISABRIDGE,
|
|
pcib_intr, sc);
|
|
#endif
|
|
if (sc->sc_ih == NULL)
|
|
printf("%s: WARNING: unable to register interrupt handler\n",
|
|
sc->sc_dev.dv_xname);
|
|
|
|
config_defer(self, pcib_bridge_callback);
|
|
}
|
|
|
|
void
|
|
pcib_bridge_callback(struct device *self)
|
|
{
|
|
struct pcib_softc *sc = (struct pcib_softc *)self;
|
|
struct isabus_attach_args iba;
|
|
|
|
memset(&iba, 0, sizeof(iba));
|
|
|
|
#if defined(ALGOR_P5064)
|
|
{
|
|
struct p5064_config *acp = &p5064_configuration;
|
|
|
|
iba.iba_iot = &acp->ac_iot;
|
|
iba.iba_memt = &acp->ac_memt;
|
|
iba.iba_dmat = &acp->ac_isa_dmat;
|
|
}
|
|
#elif defined(ALGOR_P6032)
|
|
{
|
|
struct p6032_config *acp = &p6032_configuration;
|
|
|
|
iba.iba_iot = &acp->ac_iot;
|
|
iba.iba_memt = &acp->ac_memt;
|
|
iba.iba_dmat = &acp->ac_isa_dmat;
|
|
}
|
|
#endif
|
|
|
|
iba.iba_ic = &sc->sc_ic;
|
|
iba.iba_ic->ic_attach_hook = pcib_isa_attach_hook;
|
|
|
|
(void) config_found_ia(&sc->sc_dev, "isabus", &iba, isabusprint);
|
|
}
|
|
|
|
void
|
|
pcib_isa_attach_hook(struct device *parent, struct device *self,
|
|
struct isabus_attach_args *iba)
|
|
{
|
|
|
|
/* Nothing to do. */
|
|
}
|
|
|
|
void
|
|
pcib_set_icus(struct pcib_softc *sc)
|
|
{
|
|
|
|
/* Enable the cascade IRQ (2) if 8-15 is enabled. */
|
|
if ((sc->sc_imask & 0xff00) != 0xff00)
|
|
sc->sc_imask &= ~(1U << 2);
|
|
else
|
|
sc->sc_imask |= (1U << 2);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW1,
|
|
sc->sc_imask & 0xff);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2, PIC_OCW1,
|
|
(sc->sc_imask >> 8) & 0xff);
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_elcr, 0,
|
|
sc->sc_elcr & 0xff);
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_elcr, 1,
|
|
(sc->sc_elcr >> 8) & 0xff);
|
|
}
|
|
|
|
int
|
|
pcib_intr(void *v)
|
|
{
|
|
struct pcib_softc *sc = v;
|
|
struct algor_intrhand *ih;
|
|
int irq;
|
|
|
|
for (;;) {
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW3,
|
|
OCW3_SELECT | OCW3_POLL);
|
|
irq = bus_space_read_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW3);
|
|
if ((irq & OCW3_POLL_PENDING) == 0)
|
|
return (1);
|
|
|
|
irq = OCW3_POLL_IRQ(irq);
|
|
|
|
if (irq == 2) {
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2,
|
|
PIC_OCW3, OCW3_SELECT | OCW3_POLL);
|
|
irq = bus_space_read_1(sc->sc_iot, sc->sc_ioh_icu2,
|
|
PIC_OCW3);
|
|
if (irq & OCW3_POLL_PENDING)
|
|
irq = OCW3_POLL_IRQ(irq) + 8;
|
|
else
|
|
irq = 2;
|
|
}
|
|
|
|
sc->sc_intrtab[irq].intr_count.ev_count++;
|
|
for (ih = LIST_FIRST(&sc->sc_intrtab[irq].intr_q);
|
|
ih != NULL; ih = LIST_NEXT(ih, ih_q)) {
|
|
(*ih->ih_func)(ih->ih_arg);
|
|
}
|
|
|
|
/* Send a specific EOI to the 8259. */
|
|
if (irq > 7) {
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu2,
|
|
PIC_OCW2, OCW2_SELECT | OCW2_EOI | OCW2_SL |
|
|
OCW2_ILS(irq & 7));
|
|
irq = 2;
|
|
}
|
|
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh_icu1, PIC_OCW2,
|
|
OCW2_SELECT | OCW2_EOI | OCW2_SL | OCW2_ILS(irq));
|
|
}
|
|
}
|
|
|
|
const struct evcnt *
|
|
pcib_isa_intr_evcnt(void *v, int irq)
|
|
{
|
|
struct pcib_softc *sc = v;
|
|
|
|
#if defined(ALGOR_P5064)
|
|
if (p5064_isa_to_irqmap[irq] != -1)
|
|
return (isa_intr_evcnt(sc->sc_parent_ic, irq));
|
|
#endif
|
|
|
|
return (&sc->sc_intrtab[irq].intr_count);
|
|
}
|
|
|
|
void *
|
|
pcib_isa_intr_establish(void *v, int irq, int type, int level,
|
|
int (*func)(void *), void *arg)
|
|
{
|
|
struct pcib_softc *sc = v;
|
|
struct algor_intrhand *ih;
|
|
int s;
|
|
|
|
if (irq > 15 || irq == 2 || type == IST_NONE)
|
|
panic("pcib_isa_intr_establish: bad irq or type");
|
|
|
|
#if defined(ALGOR_P5064)
|
|
if (p5064_isa_to_irqmap[irq] != -1)
|
|
return (isa_intr_establish(sc->sc_parent_ic, irq, type,
|
|
level, func, arg));
|
|
#endif
|
|
|
|
switch (sc->sc_intrtab[irq].intr_type) {
|
|
case IST_NONE:
|
|
sc->sc_intrtab[irq].intr_type = type;
|
|
break;
|
|
|
|
case IST_EDGE:
|
|
case IST_LEVEL:
|
|
if (type == sc->sc_intrtab[irq].intr_type)
|
|
break;
|
|
/* FALLTHROUGH */
|
|
case IST_PULSE:
|
|
/*
|
|
* We can't share interrupts in this case.
|
|
*/
|
|
return (NULL);
|
|
}
|
|
|
|
ih = malloc(sizeof(*ih), M_DEVBUF, M_NOWAIT);
|
|
if (ih == NULL)
|
|
return (NULL);
|
|
|
|
ih->ih_func = func;
|
|
ih->ih_arg = arg;
|
|
ih->ih_irq = irq;
|
|
ih->ih_irqmap = NULL;
|
|
|
|
s = splhigh();
|
|
|
|
/* Insert the handler into the table. */
|
|
LIST_INSERT_HEAD(&sc->sc_intrtab[irq].intr_q, ih, ih_q);
|
|
sc->sc_intrtab[irq].intr_type = type;
|
|
|
|
/* Enable it, set trigger mode. */
|
|
sc->sc_imask &= ~(1 << irq);
|
|
if (sc->sc_intrtab[irq].intr_type == IST_LEVEL)
|
|
sc->sc_elcr |= (1 << irq);
|
|
else
|
|
sc->sc_elcr &= ~(1 << irq);
|
|
|
|
pcib_set_icus(sc);
|
|
|
|
splx(s);
|
|
|
|
return (ih);
|
|
}
|
|
|
|
void
|
|
pcib_isa_intr_disestablish(void *v, void *arg)
|
|
{
|
|
struct pcib_softc *sc = v;
|
|
struct algor_intrhand *ih = arg;
|
|
int s;
|
|
|
|
#if defined(ALGOR_P5064)
|
|
if (p5064_isa_to_irqmap[ih->ih_irq] != -1) {
|
|
isa_intr_disestablish(sc->sc_parent_ic, ih);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
s = splhigh();
|
|
|
|
LIST_REMOVE(ih, ih_q);
|
|
|
|
/* If there are no more handlers on this IRQ, disable it. */
|
|
if (LIST_FIRST(&sc->sc_intrtab[ih->ih_irq].intr_q) == NULL) {
|
|
sc->sc_imask |= (1 << ih->ih_irq);
|
|
pcib_set_icus(sc);
|
|
}
|
|
|
|
splx(s);
|
|
|
|
free(ih, M_DEVBUF);
|
|
}
|
|
|
|
int
|
|
pcib_isa_intr_alloc(void *v, int mask, int type, int *irq)
|
|
{
|
|
struct pcib_softc *sc = v;
|
|
int i, tmp, bestirq, count;
|
|
struct algor_intrhand *ih;
|
|
|
|
if (type == IST_NONE)
|
|
panic("pcib_intr_alloc: bogus type");
|
|
|
|
bestirq = -1;
|
|
count = -1;
|
|
|
|
mask &= ~sc->sc_reserved;
|
|
|
|
#if 0
|
|
printf("pcib_intr_alloc: mask = 0x%04x\n", mask);
|
|
#endif
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
if ((mask & (1 << i)) == 0)
|
|
continue;
|
|
|
|
switch (sc->sc_intrtab[i].intr_type) {
|
|
case IST_NONE:
|
|
/*
|
|
* If nothing's using the IRQ, just return it.
|
|
*/
|
|
*irq = i;
|
|
return (0);
|
|
|
|
case IST_EDGE:
|
|
case IST_LEVEL:
|
|
if (type != sc->sc_intrtab[i].intr_type)
|
|
continue;
|
|
/*
|
|
* If the IRQ is sharable, count the number of
|
|
* other handlers, and if it's smaller than the
|
|
* last IRQ like this, remember it.
|
|
*/
|
|
tmp = 0;
|
|
for (ih = LIST_FIRST(&sc->sc_intrtab[i].intr_q);
|
|
ih != NULL; ih = LIST_NEXT(ih, ih_q))
|
|
tmp++;
|
|
if (bestirq == -1 || count > tmp) {
|
|
bestirq = i;
|
|
count = tmp;
|
|
}
|
|
break;
|
|
|
|
case IST_PULSE:
|
|
/* This just isn't sharable. */
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (bestirq == -1)
|
|
return (1);
|
|
|
|
*irq = bestirq;
|
|
return (0);
|
|
}
|