/* $NetBSD: cy_pci.c,v 1.26 2018/12/09 11:14:02 jdolecek Exp $ */ /* * cy_pci.c * * Driver for Cyclades Cyclom-8/16/32 multiport serial cards * (currently not tested with Cyclom-32 cards) * * Timo Rossi, 1996 */ #include __KERNEL_RCSID(0, "$NetBSD: cy_pci.c,v 1.26 2018/12/09 11:14:02 jdolecek Exp $"); #include #include #include #include #include #include #include #include #include #include #include struct cy_pci_softc { struct cy_softc sc_cy; /* real cy softc */ bus_space_tag_t sc_iot; /* PLX runtime i/o tag */ bus_space_handle_t sc_ioh; /* PLX runtime i/o handle */ }; static const struct cy_pci_product { pci_product_id_t cp_product; /* product ID */ pcireg_t cp_memtype; /* memory type */ } cy_pci_products[] = { { PCI_PRODUCT_CYCLADES_CYCLOMY_1, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT_1M }, { PCI_PRODUCT_CYCLADES_CYCLOM4Y_1, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT_1M }, { PCI_PRODUCT_CYCLADES_CYCLOM8Y_1, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT_1M }, { PCI_PRODUCT_CYCLADES_CYCLOMY_2, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT }, { PCI_PRODUCT_CYCLADES_CYCLOM4Y_2, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT }, { PCI_PRODUCT_CYCLADES_CYCLOM8Y_2, PCI_MAPREG_TYPE_MEM|PCI_MAPREG_MEM_TYPE_32BIT }, { 0, 0 }, }; static const int cy_pci_nproducts = sizeof(cy_pci_products) / sizeof(cy_pci_products[0]); static const struct cy_pci_product * cy_pci_lookup(const struct pci_attach_args *pa) { const struct cy_pci_product *cp; int i; if (PCI_VENDOR(pa->pa_id) != PCI_VENDOR_CYCLADES) return (NULL); for (i = 0; i < cy_pci_nproducts; i++) { cp = &cy_pci_products[i]; if (PCI_PRODUCT(pa->pa_id) == cp->cp_product) return (cp); } return (NULL); } static int cy_pci_match(device_t parent, cfdata_t match, void *aux) { struct pci_attach_args *pa = aux; return (cy_pci_lookup(pa) != NULL); } static void cy_pci_attach(device_t parent, device_t self, void *aux) { struct cy_pci_softc *psc = device_private(self); struct cy_softc *sc = &psc->sc_cy; struct pci_attach_args *pa = aux; pci_intr_handle_t ih; const struct cy_pci_product *cp; const char *intrstr; int plx_ver; char intrbuf[PCI_INTRSTR_LEN]; sc->sc_dev = self; aprint_naive(": Multi-port serial controller\n"); sc->sc_bustype = CY_BUSTYPE_PCI; cp = cy_pci_lookup(pa); if (cp == NULL) panic("cy_pci_attach: impossible"); aprint_normal(": Cyclades-Y multiport serial\n"); if (pci_mapreg_map(pa, 0x14, PCI_MAPREG_TYPE_IO, 0, &psc->sc_iot, &psc->sc_ioh, NULL, NULL) != 0) { aprint_error_dev(sc->sc_dev, "unable to map PLX registers\n"); return; } if (pci_mapreg_map(pa, 0x18, cp->cp_memtype, 0, &sc->sc_memt, &sc->sc_bsh, NULL, NULL) != 0) { aprint_error_dev(sc->sc_dev, "unable to map device registers\n"); return; } if (cy_find(sc) == 0) { aprint_error_dev(sc->sc_dev, "unable to find CD1400s\n"); return; } /* * XXX Like the Cyclades-Z, we should really check the EEPROM to * determine the "poll or interrupt" setting. For now, we always * map the interrupt and enable it in the PLX. */ /* Map and establish the interrupt. */ if (pci_intr_map(pa, &ih) != 0) { aprint_error_dev(sc->sc_dev, "unable to map interrupt\n"); return; } intrstr = pci_intr_string(pa->pa_pc, ih, intrbuf, sizeof(intrbuf)); sc->sc_ih = pci_intr_establish_xname(pa->pa_pc, ih, IPL_TTY, cy_intr, sc, device_xname(self)); if (sc->sc_ih == NULL) { aprint_error_dev(sc->sc_dev, "unable to establish interrupt"); if (intrstr != NULL) aprint_error(" at %s", intrstr); aprint_error("\n"); return; } aprint_normal_dev(sc->sc_dev, "interrupting at %s\n", intrstr); cy_attach(sc); plx_ver = bus_space_read_1(sc->sc_memt, sc->sc_bsh, CY_PLX_VER) & 0x0f; /* Enable PCI card interrupts */ switch (plx_ver) { case CY_PLX_9050: bus_space_write_2(psc->sc_iot, psc->sc_ioh, CY_PCI_INTENA_9050, bus_space_read_2(psc->sc_iot, psc->sc_ioh, CY_PCI_INTENA_9050) | 0x40); break; case CY_PLX_9060: case CY_PLX_9080: default: bus_space_write_2(psc->sc_iot, psc->sc_ioh, CY_PCI_INTENA, bus_space_read_2(psc->sc_iot, psc->sc_ioh, CY_PCI_INTENA) | 0x900); } } CFATTACH_DECL_NEW(cy_pci, sizeof(struct cy_pci_softc), cy_pci_match, cy_pci_attach, NULL, NULL);