/* $NetBSD: pcib.c,v 1.2 2000/03/31 14:51:55 soren Exp $ */ /* * Copyright (c) 2000 Soren S. Jorvang. All rights reserved. * * 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 AUTHOR 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 AUTHOR 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 #include #include #include #include #include #include #include #include #include #include #include #include static int pcib_match(struct device *, struct cfdata *, void *); static void pcib_attach(struct device *, struct device *, void *); static int icu_intr(void *); struct cfattach pcib_ca = { sizeof(struct device), pcib_match, pcib_attach }; static struct { int (*func)(void *); void *arg; } icu[IO_ICUSIZE]; static int pcib_match(parent, match, aux) struct device *parent; struct cfdata *match; void *aux; { struct pci_attach_args *pa = aux; if ((PCI_VENDOR(pa->pa_id) == PCI_VENDOR_VIATECH) && (PCI_PRODUCT(pa->pa_id) == PCI_PRODUCT_VIATECH_VT82C586_ISA)) return 1; return 0; } static void pcib_attach(parent, self, aux) struct device *parent; struct device *self; void *aux; { struct pci_attach_args *pa = aux; char devinfo[256]; pci_devinfo(pa->pa_id, pa->pa_class, 0, devinfo); printf("\n%s: %s, rev %d\n", self->dv_xname, devinfo, PCI_REVISION(pa->pa_class)); /* * Initialize ICU. Since we block all these interrupts with * splbio(), we can just enable all of them all the time here. */ *(volatile u_int8_t *)MIPS_PHYS_TO_KSEG1(0x10000000 + IO_ICU1) = 0x10; *(volatile u_int8_t *)MIPS_PHYS_TO_KSEG1(0x10000000 + IO_ICU1+1) = 0xff; *(volatile u_int8_t *)MIPS_PHYS_TO_KSEG1(0x10000000 + IO_ICU2) = 0x10; *(volatile u_int8_t *)MIPS_PHYS_TO_KSEG1(0x10000000 + IO_ICU2+1) = 0xff; wbflush(); cpu_intr_establish(4, IPL_NONE, icu_intr, NULL); } void * icu_intr_establish(irq, type, level, func, arg) int irq; int type; int level; int (*func)(void *); void *arg; { int i; for (i = 0; i <= IO_ICUSIZE; i++) { if (i == IO_ICUSIZE) panic("too many IRQs"); if (icu[i].func != NULL) continue; icu[i].func = func; icu[i].arg = arg; break; } return (void *)-1; } int icu_intr(arg) void *arg; { int i; for (i = 0; i < IO_ICUSIZE; i++) { if (icu[i].func == NULL) return 0; (*icu[i].func)(icu[i].arg); } return 0; }