/* $OpenBSD: pbcpcibus.c,v 1.5 1998/01/29 14:54:55 pefo Exp $ */ /* * Copyright (c) 1997 Per Fogelstrom * * 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. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed under OpenBSD by * Per Fogelstrom. * 4. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 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. * */ /* * ARC PCI BUS Bridge driver. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include extern vm_map_t phys_map; extern char eth_hw_addr[]; /* Hardware ethernet address stored elsewhere */ int pbcpcibrmatch __P((struct device *, void *, void *)); void pbcpcibrattach __P((struct device *, struct device *, void *)); void pbc_attach_hook __P((struct device *, struct device *, struct pcibus_attach_args *)); int pbc_bus_maxdevs __P((void *, int)); pcitag_t pbc_make_tag __P((void *, int, int, int)); void pbc_decompose_tag __P((void *, pcitag_t, int *, int *, int *)); pcireg_t pbc_conf_read __P((void *, pcitag_t, int)); void pbc_conf_write __P((void *, pcitag_t, int, pcireg_t)); int pbc_intr_map __P((void *, pcitag_t, int, int, pci_intr_handle_t *)); const char *pbc_intr_string __P((void *, pci_intr_handle_t)); void *pbc_intr_establish __P((void *, pci_intr_handle_t, int, int (*func)(void *), void *, char *)); void pbc_intr_disestablish __P((void *, void *)); int pbc_ether_hw_addr __P((u_int8_t *)); struct cfattach pbcpcibr_ca = { sizeof(struct pcibr_softc), pbcpcibrmatch, pbcpcibrattach, }; struct cfdriver pbcpcibr_cd = { NULL, "pbcpcibr", DV_DULL, }; static int pbcpcibrprint __P((void *, const char *pnp)); struct pcibr_config pbc_config; int pbcpcibrmatch(parent, match, aux) struct device *parent; void *match, *aux; { struct confargs *ca = aux; /* Make sure that we're looking for a PCI bridge. */ if (strcmp(ca->ca_name, pbcpcibr_cd.cd_name) != 0) return (0); return (1); } void pbcpcibrattach(parent, self, aux) struct device *parent, *self; void *aux; { struct pcibr_softc *sc = (struct pcibr_softc *)self; struct pcibr_config *lcp; struct pcibus_attach_args pba; switch(system_type) { case ALGOR_P4032: V96X_PCI_BASE0 = V96X_PCI_BASE0 & 0xffff0000; lcp = sc->sc_pcibr = &pbc_config; sc->sc_bus_space.bus_base = V96X_PCI_MEM_SPACE; sc->sc_bus_space.bus_sparse1 = 0; sc->sc_bus_space.bus_sparse2 = 0; sc->sc_bus_space.bus_sparse4 = 0; sc->sc_bus_space.bus_sparse8 = 0; lcp->lc_pc.pc_conf_v = lcp; lcp->lc_pc.pc_attach_hook = pbc_attach_hook; lcp->lc_pc.pc_bus_maxdevs = pbc_bus_maxdevs; lcp->lc_pc.pc_make_tag = pbc_make_tag; lcp->lc_pc.pc_decompose_tag = pbc_decompose_tag; lcp->lc_pc.pc_conf_read = pbc_conf_read; lcp->lc_pc.pc_conf_write = pbc_conf_write; lcp->lc_pc.pc_ether_hw_addr = pbc_ether_hw_addr; lcp->lc_pc.pc_intr_v = lcp; lcp->lc_pc.pc_intr_map = pbc_intr_map; lcp->lc_pc.pc_intr_string = pbc_intr_string; lcp->lc_pc.pc_intr_establish = pbc_intr_establish; lcp->lc_pc.pc_intr_disestablish = pbc_intr_disestablish; printf(": V3 V962, Revision %x.\n", V96X_PCI_CC_REV); break; } pba.pba_busname = "pci"; pba.pba_iot = &sc->sc_bus_space; pba.pba_memt = &sc->sc_bus_space; pba.pba_pc = &lcp->lc_pc; pba.pba_bus = 0; config_found(self, &pba, pbcpcibrprint); } static int pbcpcibrprint(aux, pnp) void *aux; const char *pnp; { struct pcibus_attach_args *pba = aux; if(pnp) printf("%s at %s", pba->pba_busname, pnp); printf(" bus %d", pba->pba_bus); return(UNCONF); } /* * Get PCI physical address from given viritual address. */ vm_offset_t vtophys(p) void *p; { vm_offset_t pa; vm_offset_t va; va = (vm_offset_t)p; if(va >= UADDR) { /* Stupid driver have buf on stack!! */ va = (vm_offset_t)curproc->p_addr + (va & ~UADDR); } if((vm_offset_t)va < VM_MIN_KERNEL_ADDRESS) { pa = CACHED_TO_PHYS(va); } else { pa = pmap_extract(vm_map_pmap(phys_map), va); } return(pa); } void pbc_attach_hook(parent, self, pba) struct device *parent, *self; struct pcibus_attach_args *pba; { } int pbc_bus_maxdevs(cpv, busno) void *cpv; int busno; { return(16); } pcitag_t pbc_make_tag(cpv, bus, dev, fnc) void *cpv; int bus, dev, fnc; { return (bus << 16) | (dev << 11) | (fnc << 8); } void pbc_decompose_tag(cpv, tag, busp, devp, fncp) void *cpv; pcitag_t tag; int *busp, *devp, *fncp; { if (busp != NULL) *busp = (tag >> 16) & 0xff; if (devp != NULL) *devp = (tag >> 11) & 0x1f; if (fncp != NULL) *fncp = (tag >> 8) & 0x7; } pcireg_t pbc_conf_read(cpv, tag, offset) void *cpv; pcitag_t tag; int offset; { pcireg_t data; u_int32_t addr; int device; int s; if((tag >> 16) != 0) return(~0); if(offset & 3 || offset < 0 || offset >= 0x100) { printf ("pci_conf_read: bad reg %x\n", offset); return(~0); } device = (tag >> 11) & 0x1f; addr = (0x800 << device) | (tag & 0x380) | offset; s = splhigh(); /* clear aborts */ V96X_PCI_STAT |= V96X_PCI_STAT_M_ABORT | V96X_PCI_STAT_T_ABORT; /* high 12 bits of address go in map register, and set for conf space */ V96X_LB_MAP0 = ((addr >> 16) & V96X_LB_MAPx_MAP_ADR) | V96X_LB_TYPE_CONF; wbflush(); /* low 20 bits of address are in the actual address */ data = *(volatile pcireg_t *) (V96X_PCI_CONF_SPACE + (addr&0xfffff)); if (V96X_PCI_STAT & V96X_PCI_STAT_M_ABORT) { V96X_PCI_STAT |= V96X_PCI_STAT_M_ABORT; printf ("device %d: master abort\n", device); return(~0); } if (V96X_PCI_STAT & V96X_PCI_STAT_T_ABORT) { V96X_PCI_STAT |= V96X_PCI_STAT_T_ABORT; printf ("PCI slot %d: target abort!\n", device); return(~0); } splx(s); return(data); } void pbc_conf_write(cpv, tag, offset, data) void *cpv; pcitag_t tag; int offset; pcireg_t data; { u_int32_t addr; int device; int s; device = (tag >> 11) & 0x1f; addr = (0x800 << device) | (tag & 0x380) | offset; s = splhigh(); /* clear aborts */ V96X_PCI_STAT |= V96X_PCI_STAT_M_ABORT | V96X_PCI_STAT_T_ABORT; /* high 12 bits of address go in map register, and set for conf space */ V96X_LB_MAP0 = ((addr >> 16) & V96X_LB_MAPx_MAP_ADR) | V96X_LB_TYPE_CONF; wbflush(); /* low 20 bits of address are in the actual address */ *(volatile pcireg_t *) (V96X_PCI_CONF_SPACE + (addr&0xfffff)) = data; /* wait for write FIFO to empty */ do { } while (V96X_FIFO_STAT & V96X_FIFO_STAT_L2P_WR); if (V96X_PCI_STAT & V96X_PCI_STAT_M_ABORT) { V96X_PCI_STAT |= V96X_PCI_STAT_M_ABORT; printf ("PCI slot %d: conf_write: master abort\n", device); } if (V96X_PCI_STAT & V96X_PCI_STAT_T_ABORT) { V96X_PCI_STAT |= V96X_PCI_STAT_T_ABORT; printf ("PCI slot %d: conf_write: target abort!\n", device); } splx(s); } /* * Hook to get ethernet hardware address when not in dev rom */ int pbc_ether_hw_addr(cp) u_int8_t *cp; { if(system_type == ALGOR_P4032) { bcopy(eth_hw_addr, cp, 6); return(0); } return(-1); } int pbc_intr_map(lcv, bustag, buspin, line, ihp) void *lcv; pcitag_t bustag; int buspin, line; pci_intr_handle_t *ihp; { struct pcibr_config *lcp = lcv; pci_chipset_tag_t pc = &lcp->lc_pc; int device, pirq; if (buspin == 0) { /* No IRQ used. */ *ihp = -1; return 1; } if (buspin > 4) { printf("pbc_intr_map: bad interrupt pin %d\n", buspin); *ihp = -1; return 1; } pci_decompose_tag(pc, bustag, NULL, &device, NULL); pirq = buspin - 1; switch(device) { case 5: /* DC21041 */ pirq = 1; break; case 8: /* NCR SCSI */ pirq = 0; break; default: switch (buspin) { case PCI_INTERRUPT_PIN_A: pirq = 0; break; case PCI_INTERRUPT_PIN_B: pirq = 1; break; case PCI_INTERRUPT_PIN_C: pirq = 2; break; case PCI_INTERRUPT_PIN_D: pirq = 3; break; } } *ihp = pirq; return 0; } const char * pbc_intr_string(lcv, ih) void *lcv; pci_intr_handle_t ih; { static char str[16]; sprintf(str, "pciirq%d", ih); return(str); } void * pbc_intr_establish(lcv, ih, level, func, arg, name) void *lcv; pci_intr_handle_t ih; int level; int (*func) __P((void *)); void *arg; char *name; { return algor_pci_intr_establish(ih, level, func, arg, name); } void pbc_intr_disestablish(lcv, cookie) void *lcv, *cookie; { algor_pci_intr_disestablish(cookie); }