diff options
-rw-r--r-- | sys/dev/pci/if_cas.c | 1906 | ||||
-rw-r--r-- | sys/dev/pci/if_casreg.h | 555 | ||||
-rw-r--r-- | sys/dev/pci/if_casvar.h | 263 |
3 files changed, 2724 insertions, 0 deletions
diff --git a/sys/dev/pci/if_cas.c b/sys/dev/pci/if_cas.c new file mode 100644 index 00000000000..366762ab1c9 --- /dev/null +++ b/sys/dev/pci/if_cas.c @@ -0,0 +1,1906 @@ +/* $OpenBSD: if_cas.c,v 1.1 2007/02/24 20:13:34 kettenis Exp $ */ + +/* + * + * Copyright (C) 2001 Eduardo Horvath. + * 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 ``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. + * + */ + +/* + * Driver for Sun Cassini ethernet controllers. + */ + +#include "bpfilter.h" + +#include <sys/param.h> +#include <sys/systm.h> +#include <sys/timeout.h> +#include <sys/mbuf.h> +#include <sys/syslog.h> +#include <sys/malloc.h> +#include <sys/kernel.h> +#include <sys/socket.h> +#include <sys/ioctl.h> +#include <sys/errno.h> +#include <sys/device.h> + +#include <machine/endian.h> + +#include <net/if.h> +#include <net/if_dl.h> +#include <net/if_media.h> + +#ifdef INET +#include <netinet/in.h> +#include <netinet/if_ether.h> +#endif + +#if NBPFILTER > 0 +#include <net/bpf.h> +#endif + +#include <machine/bus.h> +#include <machine/intr.h> + +#include <dev/mii/mii.h> +#include <dev/mii/miivar.h> +#include <dev/mii/mii_bitbang.h> + +#include <dev/pci/if_casreg.h> +#include <dev/pci/if_casvar.h> + +#include <dev/pci/pcivar.h> +#include <dev/pci/pcireg.h> +#include <dev/pci/pcidevs.h> + +#ifdef __sparc64__ +#include <dev/ofw/openfirm.h> +#endif + +#define TRIES 10000 + +struct cfdriver cas_cd = { + NULL, "cas", DV_IFNET +}; + +int cas_match(struct device *, void *, void *); +void cas_attach(struct device *, struct device *, void *); +int cas_pci_enaddr(struct cas_softc *, struct pci_attach_args *); + +struct cfattach cas_ca = { + sizeof(struct cas_softc), cas_match, cas_attach +}; + +void cas_start(struct ifnet *); +void cas_stop(struct ifnet *, int); +int cas_ioctl(struct ifnet *, u_long, caddr_t); +void cas_tick(void *); +void cas_watchdog(struct ifnet *); +void cas_shutdown(void *); +int cas_init(struct ifnet *); +void cas_init_regs(struct cas_softc *); +int cas_ringsize(int); +int cas_meminit(struct cas_softc *); +void cas_mifinit(struct cas_softc *); +int cas_bitwait(struct cas_softc *, bus_space_handle_t, int, + u_int32_t, u_int32_t); +void cas_reset(struct cas_softc *); +int cas_reset_rx(struct cas_softc *); +int cas_reset_tx(struct cas_softc *); +int cas_disable_rx(struct cas_softc *); +int cas_disable_tx(struct cas_softc *); +void cas_rxdrain(struct cas_softc *); +int cas_add_rxbuf(struct cas_softc *, int idx); +void cas_setladrf(struct cas_softc *); +int cas_encap(struct cas_softc *, struct mbuf *, u_int32_t *); + +/* MII methods & callbacks */ +int cas_mii_readreg(struct device *, int, int); +void cas_mii_writereg(struct device *, int, int, int); +void cas_mii_statchg(struct device *); +int cas_pcs_readreg(struct device *, int, int); +void cas_pcs_writereg(struct device *, int, int, int); + +int cas_mediachange(struct ifnet *); +void cas_mediastatus(struct ifnet *, struct ifmediareq *); + +struct mbuf *cas_get(struct cas_softc *, int, int); +int cas_eint(struct cas_softc *, u_int); +int cas_rint(struct cas_softc *); +int cas_tint(struct cas_softc *, u_int32_t); +int cas_pint(struct cas_softc *); + +#ifdef CAS_DEBUG +#define DPRINTF(sc, x) if ((sc)->sc_arpcom.ac_if.if_flags & IFF_DEBUG) \ + printf x +#else +#define DPRINTF(sc, x) /* nothing */ +#endif + +const struct pci_matchid cas_pci_devices[] = { + { PCI_VENDOR_SUN, PCI_PRODUCT_SUN_CASSINI } +}; + +int +cas_match(struct device *parent, void *cf, void *aux) +{ + return (pci_matchbyid((struct pci_attach_args *)aux, cas_pci_devices, + sizeof(cas_pci_devices)/sizeof(cas_pci_devices[0]))); +} + +#define PROMHDR_PTR_DATA 0x18 +#define PROMDATA_PTR_VPD 0x08 +#define PROMDATA_DATA2 0x0a + +static const u_int8_t cas_promhdr[] = { 0x55, 0xaa }; +static const u_int8_t cas_promdat[] = { + 'P', 'C', 'I', 'R', + PCI_VENDOR_SUN & 0xff, PCI_VENDOR_SUN >> 8, + PCI_PRODUCT_SUN_CASSINI & 0xff, PCI_PRODUCT_SUN_CASSINI >> 8 +}; + +static const u_int8_t cas_promdat2[] = { + 0x18, 0x00, /* structure length */ + 0x00, /* structure revision */ + 0x00, /* interface revision */ + PCI_SUBCLASS_NETWORK_ETHERNET, /* subclass code */ + PCI_CLASS_NETWORK /* class code */ +}; + +int +cas_pci_enaddr(struct cas_softc *sc, struct pci_attach_args *pa) +{ + struct pci_vpd *vpd; + bus_space_handle_t romh; + bus_space_tag_t romt; + bus_size_t romsize; + u_int8_t buf[32]; + pcireg_t address, mask; + int dataoff, vpdoff; + int rv = -1; + + address = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_ROM_REG); + pci_conf_write(pa->pa_pc, pa->pa_tag, PCI_ROM_REG, 0xfffffffe); + mask = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_ROM_REG); + address |= PCI_ROM_ENABLE; + pci_conf_write(pa->pa_pc, pa->pa_tag, PCI_ROM_REG, address); + + romt = pa->pa_memt; + romsize = PCI_ROM_SIZE(mask); + if (bus_space_map(romt, PCI_ROM_ADDR(address), romsize, 0, &romh)) { + romsize = 0; + goto fail; + } + + bus_space_read_region_1(romt, romh, 0, buf, sizeof(buf)); + if (bcmp(buf, cas_promhdr, sizeof(cas_promhdr))) + goto fail; + + dataoff = buf[PROMHDR_PTR_DATA] | (buf[PROMHDR_PTR_DATA + 1] << 8); + if (dataoff < 0x1c) + goto fail; + + bus_space_read_region_1(romt, romh, dataoff, buf, sizeof(buf)); + if (bcmp(buf, cas_promdat, sizeof(cas_promdat)) || + bcmp(buf + PROMDATA_DATA2, cas_promdat2, sizeof(cas_promdat2))) + goto fail; + + vpdoff = buf[PROMDATA_PTR_VPD] | (buf[PROMDATA_PTR_VPD + 1] << 8); + if (vpdoff < 0x1c) + goto fail; + + bus_space_read_region_1(romt, romh, vpdoff, buf, sizeof(buf)); + + /* + * The VPD is not in PCI 2.2 standard format. The length in + * the resource header is big endian. + */ + vpd = (struct pci_vpd *)(buf + 3); + if (!PCI_VPDRES_ISLARGE(buf[0]) || + PCI_VPDRES_LARGE_NAME(buf[0]) != PCI_VPDRES_TYPE_VPD) + goto fail; + if (vpd->vpd_key0 != 'N' || vpd->vpd_key1 != 'A') + goto fail; + + bcopy(buf + 6, sc->sc_arpcom.ac_enaddr, ETHER_ADDR_LEN); + rv = 0; + + fail: + if (romsize != 0) + bus_space_unmap(romt, romh, romsize); + + address = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_ROM_REG); + address &= ~PCI_ROM_ENABLE; + pci_conf_write(pa->pa_pc, pa->pa_tag, PCI_ROM_REG, address); + + return (rv); +} + +void +cas_attach(struct device *parent, struct device *self, void *aux) +{ + struct pci_attach_args *pa = aux; + struct cas_softc *sc = (void *)self; + pci_intr_handle_t ih; +#ifdef __sparc64__ + /* XXX the following declarations should be elsewhere */ + extern void myetheraddr(u_char *); +#endif + const char *intrstr = NULL; + bus_size_t size; + int gotenaddr = 0; + + sc->sc_dmatag = pa->pa_dmat; + +#define PCI_CAS_BASEADDR 0x10 + if (pci_mapreg_map(pa, PCI_CAS_BASEADDR, PCI_MAPREG_TYPE_MEM, 0, + &sc->sc_memt, &sc->sc_memh, NULL, &size, 0) != 0) { + printf(": could not map registers\n"); + return; + } + + if (cas_pci_enaddr(sc, pa) == 0) + gotenaddr = 1; + +#ifdef __sparc64__ + if (!gotenaddr) { + if (OF_getprop(PCITAG_NODE(pa->pa_tag), "local-mac-address", + sc->sc_arpcom.ac_enaddr, ETHER_ADDR_LEN) <= 0) + myetheraddr(sc->sc_arpcom.ac_enaddr); + gotenaddr = 1; + } +#endif +#ifdef __powerpc__ + if (!gotenaddr) { + pci_ether_hw_addr(pa->pa_pc, sc->sc_arpcom.ac_enaddr); + gotenaddr = 1; + } +#endif + + sc->sc_burst = 16; /* XXX */ + + if (pci_intr_map(pa, &ih) != 0) { + printf(": couldn't map interrupt\n"); + bus_space_unmap(sc->sc_memt, sc->sc_memh, size); + return; + } + intrstr = pci_intr_string(pa->pa_pc, ih); + sc->sc_ih = pci_intr_establish(pa->pa_pc, + ih, IPL_NET, cas_intr, sc, self->dv_xname); + if (sc->sc_ih == NULL) { + printf(": couldn't establish interrupt"); + if (intrstr != NULL) + printf(" at %s", intrstr); + printf("\n"); + bus_space_unmap(sc->sc_memt, sc->sc_memh, size); + return; + } + + printf(": %s", intrstr); + + /* + * call the main configure + */ + cas_config(sc); +} + +/* + * cas_config: + * + * Attach a Cassini interface to the system. + */ +void +cas_config(struct cas_softc *sc) +{ + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + struct mii_data *mii = &sc->sc_mii; + struct mii_softc *child; + int i, error; + + /* Make sure the chip is stopped. */ + ifp->if_softc = sc; + cas_reset(sc); + + /* + * Allocate the control data structures, and create and load the + * DMA map for it. + */ + if ((error = bus_dmamem_alloc(sc->sc_dmatag, + sizeof(struct cas_control_data), PAGE_SIZE, 0, &sc->sc_cdseg, + 1, &sc->sc_cdnseg, 0)) != 0) { + printf("\n%s: unable to allocate control data, error = %d\n", + sc->sc_dev.dv_xname, error); + goto fail_0; + } + + /* XXX should map this in with correct endianness */ + if ((error = bus_dmamem_map(sc->sc_dmatag, &sc->sc_cdseg, sc->sc_cdnseg, + sizeof(struct cas_control_data), (caddr_t *)&sc->sc_control_data, + BUS_DMA_COHERENT)) != 0) { + printf("\n%s: unable to map control data, error = %d\n", + sc->sc_dev.dv_xname, error); + goto fail_1; + } + + if ((error = bus_dmamap_create(sc->sc_dmatag, + sizeof(struct cas_control_data), 1, + sizeof(struct cas_control_data), 0, 0, &sc->sc_cddmamap)) != 0) { + printf("\n%s: unable to create control data DMA map, " + "error = %d\n", sc->sc_dev.dv_xname, error); + goto fail_2; + } + + if ((error = bus_dmamap_load(sc->sc_dmatag, sc->sc_cddmamap, + sc->sc_control_data, sizeof(struct cas_control_data), NULL, + 0)) != 0) { + printf("\n%s: unable to load control data DMA map, error = %d\n", + sc->sc_dev.dv_xname, error); + goto fail_3; + } + + /* + * Create the receive buffer DMA maps. + */ + for (i = 0; i < CAS_NRXDESC; i++) { + if ((error = bus_dmamap_create(sc->sc_dmatag, MCLBYTES, 1, + MCLBYTES, 0, 0, &sc->sc_rxsoft[i].rxs_dmamap)) != 0) { + printf("\n%s: unable to create rx DMA map %d, " + "error = %d\n", sc->sc_dev.dv_xname, i, error); + goto fail_5; + } + sc->sc_rxsoft[i].rxs_mbuf = NULL; + } + /* + * Create the transmit buffer DMA maps. + */ + for (i = 0; i < CAS_NTXDESC; i++) { + if ((error = bus_dmamap_create(sc->sc_dmatag, MCLBYTES, + CAS_NTXSEGS, MCLBYTES, 0, BUS_DMA_NOWAIT, + &sc->sc_txd[i].sd_map)) != 0) { + printf("\n%s: unable to create tx DMA map %d, " + "error = %d\n", sc->sc_dev.dv_xname, i, error); + goto fail_6; + } + sc->sc_txd[i].sd_mbuf = NULL; + } + + /* + * From this point forward, the attachment cannot fail. A failure + * before this point releases all resources that may have been + * allocated. + */ + + /* Announce ourselves. */ + printf(", address %s\n", ether_sprintf(sc->sc_arpcom.ac_enaddr)); + + /* Get RX FIFO size */ + sc->sc_rxfifosize = 64 * + bus_space_read_4(sc->sc_memt, sc->sc_memh, CAS_RX_FIFO_SIZE); + + /* Initialize ifnet structure. */ + strlcpy(ifp->if_xname, sc->sc_dev.dv_xname, sizeof ifp->if_xname); + ifp->if_softc = sc; + ifp->if_flags = + IFF_BROADCAST | IFF_SIMPLEX | IFF_NOTRAILERS | IFF_MULTICAST; + ifp->if_start = cas_start; + ifp->if_ioctl = cas_ioctl; + ifp->if_watchdog = cas_watchdog; + IFQ_SET_MAXLEN(&ifp->if_snd, CAS_NTXDESC - 1); + IFQ_SET_READY(&ifp->if_snd); + + ifp->if_capabilities = IFCAP_VLAN_MTU; + + /* Initialize ifmedia structures and MII info */ + mii->mii_ifp = ifp; + mii->mii_readreg = cas_mii_readreg; + mii->mii_writereg = cas_mii_writereg; + mii->mii_statchg = cas_mii_statchg; + + ifmedia_init(&mii->mii_media, 0, cas_mediachange, cas_mediastatus); + + bus_space_write_4(sc->sc_memt, sc->sc_memh, CAS_MII_DATAPATH_MODE, 0); + + cas_mifinit(sc); + + if (sc->sc_mif_config & CAS_MIF_CONFIG_MDI1) { + sc->sc_mif_config |= CAS_MIF_CONFIG_PHY_SEL; + bus_space_write_4(sc->sc_memt, sc->sc_memh, + CAS_MIF_CONFIG, sc->sc_mif_config); + } + + mii_attach(&sc->sc_dev, mii, 0xffffffff, MII_PHY_ANY, + MII_OFFSET_ANY, 0); + + child = LIST_FIRST(&mii->mii_phys); + if (child == NULL && + sc->sc_mif_config & (CAS_MIF_CONFIG_MDI0|CAS_MIF_CONFIG_MDI1)) { + /* + * Try the external PCS SERDES if we didn't find any + * MII devices. + */ + bus_space_write_4(sc->sc_memt, sc->sc_memh, + CAS_MII_DATAPATH_MODE, CAS_MII_DATAPATH_SERDES); + + bus_space_write_4(sc->sc_memt, sc->sc_memh, + CAS_MII_SLINK_CONTROL, + CAS_MII_SLINK_LOOPBACK|CAS_MII_SLINK_EN_SYNC_D); + + bus_space_write_4(sc->sc_memt, sc->sc_memh, + CAS_MII_CONFIG, CAS_MII_CONFIG_ENABLE); + + mii->mii_readreg = cas_pcs_readreg; + mii->mii_writereg = cas_pcs_writereg; + + mii_attach(&sc->sc_dev, mii, 0xffffffff, MII_PHY_ANY, + MII_OFFSET_ANY, MIIF_NOISOLATE); + } + + child = LIST_FIRST(&mii->mii_phys); + if (child == NULL) { + /* No PHY attached */ + ifmedia_add(&sc->sc_media, IFM_ETHER|IFM_MANUAL, 0, NULL); + ifmedia_set(&sc->sc_media, IFM_ETHER|IFM_MANUAL); + } else { + /* + * Walk along the list of attached MII devices and + * establish an `MII instance' to `phy number' + * mapping. We'll use this mapping in media change + * requests to determine which phy to use to program + * the MIF configuration register. + */ + for (; child != NULL; child = LIST_NEXT(child, mii_list)) { + /* + * Note: we support just two PHYs: the built-in + * internal device and an external on the MII + * connector. + */ + if (child->mii_phy > 1 || child->mii_inst > 1) { + printf("%s: cannot accommodate MII device %s" + " at phy %d, instance %d\n", + sc->sc_dev.dv_xname, + child->mii_dev.dv_xname, + child->mii_phy, child->mii_inst); + continue; + } + + sc->sc_phys[child->mii_inst] = child->mii_phy; + } + +#if 0 + /* + * Now select and activate the PHY we will use. + * + * The order of preference is External (MDI1), + * Internal (MDI0), Serial Link (no MII). + */ + if (sc->sc_phys[1]) { +#ifdef CAS_DEBUG + printf("using external phy\n"); +#endif + sc->sc_mif_config |= CAS_MIF_CONFIG_PHY_SEL; + } else { +#ifdef CAS_DEBUG + printf("using internal phy\n"); +#endif + sc->sc_mif_config &= ~CAS_MIF_CONFIG_PHY_SEL; + } + bus_space_write_4(sc->sc_memt, sc->sc_memh, CAS_MIF_CONFIG, + sc->sc_mif_config); +#endif + + /* + * XXX - we can really do the following ONLY if the + * phy indeed has the auto negotiation capability!! + */ + ifmedia_set(&sc->sc_media, IFM_ETHER|IFM_AUTO); + } + + /* Attach the interface. */ + if_attach(ifp); + ether_ifattach(ifp); + + sc->sc_sh = shutdownhook_establish(cas_shutdown, sc); + if (sc->sc_sh == NULL) + panic("cas_config: can't establish shutdownhook"); + + timeout_set(&sc->sc_tick_ch, cas_tick, sc); + return; + + /* + * Free any resources we've allocated during the failed attach + * attempt. Do this in reverse order and fall through. + */ + fail_6: + for (i = 0; i < CAS_NTXDESC; i++) { + if (sc->sc_txd[i].sd_map != NULL) + bus_dmamap_destroy(sc->sc_dmatag, + sc->sc_txd[i].sd_map); + } + fail_5: + for (i = 0; i < CAS_NRXDESC; i++) { + if (sc->sc_rxsoft[i].rxs_dmamap != NULL) + bus_dmamap_destroy(sc->sc_dmatag, + sc->sc_rxsoft[i].rxs_dmamap); + } + bus_dmamap_unload(sc->sc_dmatag, sc->sc_cddmamap); + fail_3: + bus_dmamap_destroy(sc->sc_dmatag, sc->sc_cddmamap); + fail_2: + bus_dmamem_unmap(sc->sc_dmatag, (caddr_t)sc->sc_control_data, + sizeof(struct cas_control_data)); + fail_1: + bus_dmamem_free(sc->sc_dmatag, &sc->sc_cdseg, sc->sc_cdnseg); + fail_0: + return; +} + + +void +cas_tick(void *arg) +{ + struct cas_softc *sc = arg; + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t mac = sc->sc_memh; + int s; + + /* unload collisions counters */ + ifp->if_collisions += + bus_space_read_4(t, mac, CAS_MAC_NORM_COLL_CNT) + + bus_space_read_4(t, mac, CAS_MAC_FIRST_COLL_CNT) + + bus_space_read_4(t, mac, CAS_MAC_EXCESS_COLL_CNT) + + bus_space_read_4(t, mac, CAS_MAC_LATE_COLL_CNT); + + /* clear the hardware counters */ + bus_space_write_4(t, mac, CAS_MAC_NORM_COLL_CNT, 0); + bus_space_write_4(t, mac, CAS_MAC_FIRST_COLL_CNT, 0); + bus_space_write_4(t, mac, CAS_MAC_EXCESS_COLL_CNT, 0); + bus_space_write_4(t, mac, CAS_MAC_LATE_COLL_CNT, 0); + + s = splnet(); + mii_tick(&sc->sc_mii); + splx(s); + + timeout_add(&sc->sc_tick_ch, hz); +} + +int +cas_bitwait(struct cas_softc *sc, bus_space_handle_t h, int r, + u_int32_t clr, u_int32_t set) +{ + int i; + u_int32_t reg; + + for (i = TRIES; i--; DELAY(100)) { + reg = bus_space_read_4(sc->sc_memt, h, r); + if ((reg & clr) == 0 && (reg & set) == set) + return (1); + } + + return (0); +} + +void +cas_reset(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + int s; + + s = splnet(); + DPRINTF(sc, ("%s: cas_reset\n", sc->sc_dev.dv_xname)); + cas_reset_rx(sc); + cas_reset_tx(sc); + + /* Do a full reset */ + bus_space_write_4(t, h, CAS_RESET, CAS_RESET_RX|CAS_RESET_TX); + if (!cas_bitwait(sc, h, CAS_RESET, CAS_RESET_RX | CAS_RESET_TX, 0)) + printf("%s: cannot reset device\n", sc->sc_dev.dv_xname); + splx(s); +} + + +/* + * cas_rxdrain: + * + * Drain the receive queue. + */ +void +cas_rxdrain(struct cas_softc *sc) +{ + struct cas_rxsoft *rxs; + int i; + + for (i = 0; i < CAS_NRXDESC; i++) { + rxs = &sc->sc_rxsoft[i]; + if (rxs->rxs_mbuf != NULL) { + bus_dmamap_sync(sc->sc_dmatag, rxs->rxs_dmamap, 0, + rxs->rxs_dmamap->dm_mapsize, BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(sc->sc_dmatag, rxs->rxs_dmamap); + m_freem(rxs->rxs_mbuf); + rxs->rxs_mbuf = NULL; + } + } +} + +/* + * Reset the whole thing. + */ +void +cas_stop(struct ifnet *ifp, int disable) +{ + struct cas_softc *sc = (struct cas_softc *)ifp->if_softc; + struct cas_sxd *sd; + u_int32_t i; + + DPRINTF(sc, ("%s: cas_stop\n", sc->sc_dev.dv_xname)); + + timeout_del(&sc->sc_tick_ch); + + /* + * Mark the interface down and cancel the watchdog timer. + */ + ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); + ifp->if_timer = 0; + + mii_down(&sc->sc_mii); + + cas_reset_rx(sc); + cas_reset_tx(sc); + + /* + * Release any queued transmit buffers. + */ + for (i = 0; i < CAS_NTXDESC; i++) { + sd = &sc->sc_txd[i]; + if (sd->sd_mbuf != NULL) { + bus_dmamap_sync(sc->sc_dmatag, sd->sd_map, 0, + sd->sd_map->dm_mapsize, BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(sc->sc_dmatag, sd->sd_map); + m_freem(sd->sd_mbuf); + sd->sd_mbuf = NULL; + } + } + sc->sc_tx_cnt = sc->sc_tx_prod = sc->sc_tx_cons = 0; + + if (disable) + cas_rxdrain(sc); +} + + +/* + * Reset the receiver + */ +int +cas_reset_rx(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + + /* + * Resetting while DMA is in progress can cause a bus hang, so we + * disable DMA first. + */ + cas_disable_rx(sc); + bus_space_write_4(t, h, CAS_RX_CONFIG, 0); + /* Wait till it finishes */ + if (!cas_bitwait(sc, h, CAS_RX_CONFIG, 1, 0)) + printf("%s: cannot disable rx dma\n", sc->sc_dev.dv_xname); + /* Wait 5ms extra. */ + delay(5000); + + /* Finally, reset the ERX */ + bus_space_write_4(t, h, CAS_RESET, CAS_RESET_RX); + /* Wait till it finishes */ + if (!cas_bitwait(sc, h, CAS_RESET, CAS_RESET_RX, 0)) { + printf("%s: cannot reset receiver\n", sc->sc_dev.dv_xname); + return (1); + } + return (0); +} + + +/* + * Reset the transmitter + */ +int +cas_reset_tx(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + + /* + * Resetting while DMA is in progress can cause a bus hang, so we + * disable DMA first. + */ + cas_disable_tx(sc); + bus_space_write_4(t, h, CAS_TX_CONFIG, 0); + /* Wait till it finishes */ + if (!cas_bitwait(sc, h, CAS_TX_CONFIG, 1, 0)) + printf("%s: cannot disable tx dma\n", sc->sc_dev.dv_xname); + /* Wait 5ms extra. */ + delay(5000); + + /* Finally, reset the ETX */ + bus_space_write_4(t, h, CAS_RESET, CAS_RESET_TX); + /* Wait till it finishes */ + if (!cas_bitwait(sc, h, CAS_RESET, CAS_RESET_TX, 0)) { + printf("%s: cannot reset transmitter\n", + sc->sc_dev.dv_xname); + return (1); + } + return (0); +} + +/* + * disable receiver. + */ +int +cas_disable_rx(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + u_int32_t cfg; + + /* Flip the enable bit */ + cfg = bus_space_read_4(t, h, CAS_MAC_RX_CONFIG); + cfg &= ~CAS_MAC_RX_ENABLE; + bus_space_write_4(t, h, CAS_MAC_RX_CONFIG, cfg); + + /* Wait for it to finish */ + return (cas_bitwait(sc, h, CAS_MAC_RX_CONFIG, CAS_MAC_RX_ENABLE, 0)); +} + +/* + * disable transmitter. + */ +int +cas_disable_tx(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + u_int32_t cfg; + + /* Flip the enable bit */ + cfg = bus_space_read_4(t, h, CAS_MAC_TX_CONFIG); + cfg &= ~CAS_MAC_TX_ENABLE; + bus_space_write_4(t, h, CAS_MAC_TX_CONFIG, cfg); + + /* Wait for it to finish */ + return (cas_bitwait(sc, h, CAS_MAC_TX_CONFIG, CAS_MAC_TX_ENABLE, 0)); +} + +/* + * Initialize interface. + */ +int +cas_meminit(struct cas_softc *sc) +{ + struct cas_rxsoft *rxs; + int i, error; + + /* + * Initialize the transmit descriptor ring. + */ + for (i = 0; i < CAS_NTXDESC; i++) { + sc->sc_txdescs[i].gd_flags = 0; + sc->sc_txdescs[i].gd_addr = 0; + } + CAS_CDTXSYNC(sc, 0, CAS_NTXDESC, + BUS_DMASYNC_PREREAD|BUS_DMASYNC_PREWRITE); + + /* + * Initialize the receive descriptor and receive job + * descriptor rings. + */ + for (i = 0; i < CAS_NRXDESC; i++) { + rxs = &sc->sc_rxsoft[i]; + if (rxs->rxs_mbuf == NULL) { + if ((error = cas_add_rxbuf(sc, i)) != 0) { + printf("%s: unable to allocate or map rx " + "buffer %d, error = %d\n", + sc->sc_dev.dv_xname, i, error); + /* + * XXX Should attempt to run with fewer receive + * XXX buffers instead of just failing. + */ + cas_rxdrain(sc); + return (1); + } + } else + CAS_INIT_RXDESC(sc, i); + } + sc->sc_rxptr = 0; + + return (0); +} + +int +cas_ringsize(int sz) +{ + switch (sz) { + case 32: + return CAS_RING_SZ_32; + case 64: + return CAS_RING_SZ_64; + case 128: + return CAS_RING_SZ_128; + case 256: + return CAS_RING_SZ_256; + case 512: + return CAS_RING_SZ_512; + case 1024: + return CAS_RING_SZ_1024; + case 2048: + return CAS_RING_SZ_2048; + case 4096: + return CAS_RING_SZ_4096; + case 8192: + return CAS_RING_SZ_8192; + default: + printf("cas: invalid Receive Descriptor ring size %d\n", sz); + return CAS_RING_SZ_32; + } +} + +/* + * Initialization of interface; set up initialization block + * and transmit/receive descriptor rings. + */ +int +cas_init(struct ifnet *ifp) +{ + + struct cas_softc *sc = (struct cas_softc *)ifp->if_softc; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + int s; + u_int max_frame_size; + u_int32_t v; + + s = splnet(); + + DPRINTF(sc, ("%s: cas_init: calling stop\n", sc->sc_dev.dv_xname)); + /* + * Initialization sequence. The numbered steps below correspond + * to the sequence outlined in section 6.3.5.1 in the Ethernet + * Channel Engine manual (part of the PCIO manual). + * See also the STP2002-STQ document from Sun Microsystems. + */ + + /* step 1 & 2. Reset the Ethernet Channel */ + cas_stop(ifp, 0); + cas_reset(sc); + DPRINTF(sc, ("%s: cas_init: restarting\n", sc->sc_dev.dv_xname)); + + /* Re-initialize the MIF */ + cas_mifinit(sc); + + /* step 3. Setup data structures in host memory */ + cas_meminit(sc); + + /* step 4. TX MAC registers & counters */ + cas_init_regs(sc); + max_frame_size = ETHER_MAX_LEN + ETHER_VLAN_ENCAP_LEN; + v = (max_frame_size) | (0x2000 << 16) /* Burst size */; + bus_space_write_4(t, h, CAS_MAC_MAC_MAX_FRAME, v); + + /* step 5. RX MAC registers & counters */ + cas_setladrf(sc); + + /* step 6 & 7. Program Descriptor Ring Base Addresses */ + bus_space_write_4(t, h, CAS_TX_RING_PTR_HI, + (((uint64_t)CAS_CDTXADDR(sc,0)) >> 32)); + bus_space_write_4(t, h, CAS_TX_RING_PTR_LO, CAS_CDTXADDR(sc, 0)); + + bus_space_write_4(t, h, CAS_RX_RING_PTR_HI, + (((uint64_t)CAS_CDRXADDR(sc,0)) >> 32)); + bus_space_write_4(t, h, CAS_RX_RING_PTR_LO, CAS_CDRXADDR(sc, 0)); + + /* step 8. Global Configuration & Interrupt Mask */ + bus_space_write_4(t, h, CAS_INTMASK, + ~(CAS_INTR_TX_INTME| + CAS_INTR_TX_EMPTY| + CAS_INTR_RX_DONE|CAS_INTR_RX_NOBUF| + CAS_INTR_RX_TAG_ERR|CAS_INTR_PCS| + CAS_INTR_MAC_CONTROL|CAS_INTR_MIF| + CAS_INTR_BERR)); + bus_space_write_4(t, h, CAS_MAC_RX_MASK, + CAS_MAC_RX_DONE|CAS_MAC_RX_FRAME_CNT); + bus_space_write_4(t, h, CAS_MAC_TX_MASK, 0 /*CAS_MAC_TX_XMIT_DONE*/); + bus_space_write_4(t, h, CAS_MAC_CONTROL_MASK, 0); /* XXXX */ + + /* step 9. ETX Configuration: use mostly default values */ + + /* Enable DMA */ + v = cas_ringsize(CAS_NTXDESC /*XXX*/) << 13; + bus_space_write_4(t, h, CAS_TX_CONFIG, + v|CAS_TX_CONFIG_TXDMA_EN|(1<<24)|(1<<29)); + bus_space_write_4(t, h, CAS_TX_KICK, 0); + + /* step 10. ERX Configuration */ + + /* Encode Receive Descriptor ring size: four possible values */ + v = cas_ringsize(CAS_NRXDESC /*XXX*/) << 1; + + /* Enable DMA */ + bus_space_write_4(t, h, CAS_RX_CONFIG, + v|(CAS_THRSH_1024<<CAS_RX_CONFIG_FIFO_THRS_SHIFT)| + (2<<CAS_RX_CONFIG_FBOFF_SHFT)|CAS_RX_CONFIG_RXDMA_EN| + (0<<CAS_RX_CONFIG_CXM_START_SHFT)); + /* + * The following value is for an OFF Threshold of about 3/4 full + * and an ON Threshold of 1/4 full. + */ + bus_space_write_4(t, h, CAS_RX_PAUSE_THRESH, + (3 * sc->sc_rxfifosize / 256) | + ( (sc->sc_rxfifosize / 256) << 12)); + bus_space_write_4(t, h, CAS_RX_BLANKING, (6<<12)|6); + + /* step 11. Configure Media */ + mii_mediachg(&sc->sc_mii); + + /* step 12. RX_MAC Configuration Register */ + v = bus_space_read_4(t, h, CAS_MAC_RX_CONFIG); + v |= CAS_MAC_RX_ENABLE | CAS_MAC_RX_STRIP_CRC; + bus_space_write_4(t, h, CAS_MAC_RX_CONFIG, v); + + /* step 14. Issue Transmit Pending command */ + + /* step 15. Give the receiver a swift kick */ + bus_space_write_4(t, h, CAS_RX_KICK, CAS_NRXDESC-4); + + /* Start the one second timer. */ + timeout_add(&sc->sc_tick_ch, hz); + + ifp->if_flags |= IFF_RUNNING; + ifp->if_flags &= ~IFF_OACTIVE; + ifp->if_timer = 0; + splx(s); + + return (0); +} + +void +cas_init_regs(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + u_int32_t v, r; + + /* These regs are not cleared on reset */ + sc->sc_inited = 0; + if (!sc->sc_inited) { + + /* Wooo. Magic values. */ + bus_space_write_4(t, h, CAS_MAC_IPG0, 0); + bus_space_write_4(t, h, CAS_MAC_IPG1, 8); + bus_space_write_4(t, h, CAS_MAC_IPG2, 4); + + bus_space_write_4(t, h, CAS_MAC_MAC_MIN_FRAME, ETHER_MIN_LEN); + /* Max frame and max burst size */ + v = ETHER_MAX_LEN | (0x2000 << 16) /* Burst size */; + bus_space_write_4(t, h, CAS_MAC_MAC_MAX_FRAME, v); + + bus_space_write_4(t, h, CAS_MAC_PREAMBLE_LEN, 0x7); + bus_space_write_4(t, h, CAS_MAC_JAM_SIZE, 0x4); + bus_space_write_4(t, h, CAS_MAC_ATTEMPT_LIMIT, 0x10); + /* Dunno.... */ + bus_space_write_4(t, h, CAS_MAC_CONTROL_TYPE, 0x8088); + bus_space_write_4(t, h, CAS_MAC_RANDOM_SEED, + ((sc->sc_arpcom.ac_enaddr[5]<<8)|sc->sc_arpcom.ac_enaddr[4])&0x3ff); + + /* Secondary MAC addresses set to 0:0:0:0:0:0 */ + for (r = CAS_MAC_ADDR3; r < CAS_MAC_ADDR42; r += 4) + bus_space_write_4(t, h, r, 0); + + /* MAC control addr set to 0:1:c2:0:1:80 */ + bus_space_write_4(t, h, CAS_MAC_ADDR42, 0x0001); + bus_space_write_4(t, h, CAS_MAC_ADDR43, 0xc200); + bus_space_write_4(t, h, CAS_MAC_ADDR44, 0x0180); + + /* MAC filter addr set to 0:0:0:0:0:0 */ + bus_space_write_4(t, h, CAS_MAC_ADDR_FILTER0, 0); + bus_space_write_4(t, h, CAS_MAC_ADDR_FILTER1, 0); + bus_space_write_4(t, h, CAS_MAC_ADDR_FILTER2, 0); + + bus_space_write_4(t, h, CAS_MAC_ADR_FLT_MASK1_2, 0); + bus_space_write_4(t, h, CAS_MAC_ADR_FLT_MASK0, 0); + + /* Hash table initialized to 0 */ + for (r = CAS_MAC_HASH0; r <= CAS_MAC_HASH15; r += 4) + bus_space_write_4(t, h, r, 0); + + sc->sc_inited = 1; + } + + /* Counters need to be zeroed */ + bus_space_write_4(t, h, CAS_MAC_NORM_COLL_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_FIRST_COLL_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_EXCESS_COLL_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_LATE_COLL_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_DEFER_TMR_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_PEAK_ATTEMPTS, 0); + bus_space_write_4(t, h, CAS_MAC_RX_FRAME_COUNT, 0); + bus_space_write_4(t, h, CAS_MAC_RX_LEN_ERR_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_RX_ALIGN_ERR, 0); + bus_space_write_4(t, h, CAS_MAC_RX_CRC_ERR_CNT, 0); + bus_space_write_4(t, h, CAS_MAC_RX_CODE_VIOL, 0); + + /* Un-pause stuff */ + bus_space_write_4(t, h, CAS_MAC_SEND_PAUSE_CMD, 0); + + /* + * Set the station address. + */ + bus_space_write_4(t, h, CAS_MAC_ADDR0, + (sc->sc_arpcom.ac_enaddr[4]<<8) | sc->sc_arpcom.ac_enaddr[5]); + bus_space_write_4(t, h, CAS_MAC_ADDR1, + (sc->sc_arpcom.ac_enaddr[2]<<8) | sc->sc_arpcom.ac_enaddr[3]); + bus_space_write_4(t, h, CAS_MAC_ADDR2, + (sc->sc_arpcom.ac_enaddr[0]<<8) | sc->sc_arpcom.ac_enaddr[1]); +} + +/* + * Receive interrupt. + */ +int +cas_rint(struct cas_softc *sc) +{ + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + struct ether_header *eh; + struct cas_rxsoft *rxs; + struct mbuf *m; + u_int64_t rxstat; + int i, len; + + for (i = sc->sc_rxptr;; i = CAS_NEXTRX(i)) { + rxs = &sc->sc_rxsoft[i]; + + CAS_CDRXSYNC(sc, i, + BUS_DMASYNC_POSTREAD|BUS_DMASYNC_POSTWRITE); + + rxstat = CAS_DMA_READ(sc, sc->sc_rxdescs[i].gd_flags); + + if (rxstat & CAS_RD_OWN) { + /* + * We have processed all of the receive buffers. + */ + break; + } + + if (rxstat & CAS_RD_BAD_CRC) { +#ifdef CAS_DEBUG + printf("%s: receive error: CRC error\n", + sc->sc_dev.dv_xname); +#endif + CAS_INIT_RXDESC(sc, i); + continue; + } + + bus_dmamap_sync(sc->sc_dmatag, rxs->rxs_dmamap, 0, + rxs->rxs_dmamap->dm_mapsize, BUS_DMASYNC_POSTREAD); +#ifdef CAS_DEBUG + if (ifp->if_flags & IFF_DEBUG) { + printf(" rxsoft %p descriptor %d: ", rxs, i); + printf("gd_flags: 0x%016llx\t", (long long) + CAS_DMA_READ(sc, sc->sc_rxdescs[i].gd_flags)); + printf("gd_addr: 0x%016llx\n", (long long) + CAS_DMA_READ(sc, sc->sc_rxdescs[i].gd_addr)); + } +#endif + + /* No errors; receive the packet. */ + len = CAS_RD_BUFLEN(rxstat); + + /* + * Allocate a new mbuf cluster. If that fails, we are + * out of memory, and must drop the packet and recycle + * the buffer that's already attached to this descriptor. + */ + m = rxs->rxs_mbuf; + if (cas_add_rxbuf(sc, i) != 0) { + ifp->if_ierrors++; + CAS_INIT_RXDESC(sc, i); + bus_dmamap_sync(sc->sc_dmatag, rxs->rxs_dmamap, 0, + rxs->rxs_dmamap->dm_mapsize, BUS_DMASYNC_PREREAD); + continue; + } + m->m_data += 2; /* We're already off by two */ + + ifp->if_ipackets++; + eh = mtod(m, struct ether_header *); + m->m_pkthdr.rcvif = ifp; + m->m_pkthdr.len = m->m_len = len; + +#if NBPFILTER > 0 + /* + * Pass this up to any BPF listeners, but only + * pass it up the stack if its for us. + */ + if (ifp->if_bpf) + bpf_mtap(ifp->if_bpf, m, BPF_DIRECTION_IN); +#endif /* NPBFILTER > 0 */ + + /* Pass it on. */ + ether_input_mbuf(ifp, m); + } + + /* Update the receive pointer. */ + sc->sc_rxptr = i; + bus_space_write_4(t, h, CAS_RX_KICK, i); + + DPRINTF(sc, ("cas_rint: done sc->rxptr %d, complete %d\n", + sc->sc_rxptr, bus_space_read_4(t, h, CAS_RX_COMPLETION))); + + return (1); +} + + +/* + * cas_add_rxbuf: + * + * Add a receive buffer to the indicated descriptor. + */ +int +cas_add_rxbuf(struct cas_softc *sc, int idx) +{ + struct cas_rxsoft *rxs = &sc->sc_rxsoft[idx]; + struct mbuf *m; + int error; + + MGETHDR(m, M_DONTWAIT, MT_DATA); + if (m == NULL) + return (ENOBUFS); + + MCLGET(m, M_DONTWAIT); + if ((m->m_flags & M_EXT) == 0) { + m_freem(m); + return (ENOBUFS); + } + +#ifdef CAS_DEBUG +/* bzero the packet to check dma */ + memset(m->m_ext.ext_buf, 0, m->m_ext.ext_size); +#endif + + if (rxs->rxs_mbuf != NULL) + bus_dmamap_unload(sc->sc_dmatag, rxs->rxs_dmamap); + + rxs->rxs_mbuf = m; + + error = bus_dmamap_load(sc->sc_dmatag, rxs->rxs_dmamap, + m->m_ext.ext_buf, m->m_ext.ext_size, NULL, + BUS_DMA_READ|BUS_DMA_NOWAIT); + if (error) { + printf("%s: can't load rx DMA map %d, error = %d\n", + sc->sc_dev.dv_xname, idx, error); + panic("cas_add_rxbuf"); /* XXX */ + } + + bus_dmamap_sync(sc->sc_dmatag, rxs->rxs_dmamap, 0, + rxs->rxs_dmamap->dm_mapsize, BUS_DMASYNC_PREREAD); + + CAS_INIT_RXDESC(sc, idx); + + return (0); +} + + +int +cas_eint(struct cas_softc *sc, u_int status) +{ + if ((status & CAS_INTR_MIF) != 0) { +#ifdef CAS_DEBUG + printf("%s: link status changed\n", sc->sc_dev.dv_xname); +#endif + return (1); + } + + printf("%s: status=%b\n", sc->sc_dev.dv_xname, status, CAS_INTR_BITS); + return (1); +} + +int +cas_pint(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t seb = sc->sc_memh; + u_int32_t status; + + status = bus_space_read_4(t, seb, CAS_MII_INTERRUP_STATUS); + status |= bus_space_read_4(t, seb, CAS_MII_INTERRUP_STATUS); +#ifdef CAS_DEBUG + if (status) + printf("%s: link status changed\n", sc->sc_dev.dv_xname); +#endif + return (1); +} + +int +cas_intr(void *v) +{ + struct cas_softc *sc = (struct cas_softc *)v; + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t seb = sc->sc_memh; + u_int32_t status; + int r = 0; + + status = bus_space_read_4(t, seb, CAS_STATUS); + DPRINTF(sc, ("%s: cas_intr: cplt %xstatus %b\n", + sc->sc_dev.dv_xname, (status>>19), status, CAS_INTR_BITS)); + + if ((status & CAS_INTR_PCS) != 0) + r |= cas_pint(sc); + + if ((status & (CAS_INTR_RX_TAG_ERR | CAS_INTR_BERR)) != 0) + r |= cas_eint(sc, status); + + if ((status & (CAS_INTR_TX_EMPTY | CAS_INTR_TX_INTME)) != 0) + r |= cas_tint(sc, status); + + if ((status & (CAS_INTR_RX_DONE | CAS_INTR_RX_NOBUF)) != 0) + r |= cas_rint(sc); + + /* We should eventually do more than just print out error stats. */ + if (status & CAS_INTR_TX_MAC) { + int txstat = bus_space_read_4(t, seb, CAS_MAC_TX_STATUS); +#ifdef CAS_DEBUG + if (txstat & ~CAS_MAC_TX_XMIT_DONE) + printf("%s: MAC tx fault, status %x\n", + sc->sc_dev.dv_xname, txstat); +#endif + if (txstat & (CAS_MAC_TX_UNDERRUN | CAS_MAC_TX_PKT_TOO_LONG)) + cas_init(ifp); + } + if (status & CAS_INTR_RX_MAC) { + int rxstat = bus_space_read_4(t, seb, CAS_MAC_RX_STATUS); +#ifdef CAS_DEBUG + if (rxstat & ~CAS_MAC_RX_DONE) + printf("%s: MAC rx fault, status %x\n", + sc->sc_dev.dv_xname, rxstat); +#endif + /* + * On some chip revisions CAS_MAC_RX_OVERFLOW happen often + * due to a silicon bug so handle them silently. + */ + if (rxstat & CAS_MAC_RX_OVERFLOW) { + ifp->if_ierrors++; + cas_init(ifp); + } +#ifdef CAS_DEBUG + else if (rxstat & ~(CAS_MAC_RX_DONE | CAS_MAC_RX_FRAME_CNT)) + printf("%s: MAC rx fault, status %x\n", + sc->sc_dev.dv_xname, rxstat); +#endif + } + return (r); +} + + +void +cas_watchdog(struct ifnet *ifp) +{ + struct cas_softc *sc = ifp->if_softc; + + DPRINTF(sc, ("cas_watchdog: CAS_RX_CONFIG %x CAS_MAC_RX_STATUS %x " + "CAS_MAC_RX_CONFIG %x\n", + bus_space_read_4(sc->sc_memt, sc->sc_memh, CAS_RX_CONFIG), + bus_space_read_4(sc->sc_memt, sc->sc_memh, CAS_MAC_RX_STATUS), + bus_space_read_4(sc->sc_memt, sc->sc_memh, CAS_MAC_RX_CONFIG))); + + log(LOG_ERR, "%s: device timeout\n", sc->sc_dev.dv_xname); + ++ifp->if_oerrors; + + /* Try to get more packets going. */ + cas_init(ifp); +} + +/* + * Initialize the MII Management Interface + */ +void +cas_mifinit(struct cas_softc *sc) +{ + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t mif = sc->sc_memh; + + /* Configure the MIF in frame mode */ + sc->sc_mif_config = bus_space_read_4(t, mif, CAS_MIF_CONFIG); + sc->sc_mif_config &= ~CAS_MIF_CONFIG_BB_ENA; + bus_space_write_4(t, mif, CAS_MIF_CONFIG, sc->sc_mif_config); +} + +/* + * MII interface + * + * The Cassini MII interface supports at least three different operating modes: + * + * Bitbang mode is implemented using data, clock and output enable registers. + * + * Frame mode is implemented by loading a complete frame into the frame + * register and polling the valid bit for completion. + * + * Polling mode uses the frame register but completion is indicated by + * an interrupt. + * + */ +int +cas_mii_readreg(struct device *self, int phy, int reg) +{ + struct cas_softc *sc = (void *)self; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t mif = sc->sc_memh; + int n; + u_int32_t v; + +#if 0 +#ifdef CAS_DEBUG + if (sc->sc_debug) + printf("cas_mii_readreg: phy %d reg %d\n", phy, reg); +#endif +#endif + + /* Construct the frame command */ + v = (reg << CAS_MIF_REG_SHIFT) | (phy << CAS_MIF_PHY_SHIFT) | + CAS_MIF_FRAME_READ; + + bus_space_write_4(t, mif, CAS_MIF_FRAME, v); + for (n = 0; n < 100; n++) { + DELAY(1); + v = bus_space_read_4(t, mif, CAS_MIF_FRAME); + if (v & CAS_MIF_FRAME_TA0) + return (v & CAS_MIF_FRAME_DATA); + } + + printf("%s: mii_read timeout\n", sc->sc_dev.dv_xname); + return (0); +} + +void +cas_mii_writereg(struct device *self, int phy, int reg, int val) +{ + struct cas_softc *sc = (void *)self; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t mif = sc->sc_memh; + int n; + u_int32_t v; + +#if 0 +#ifdef CAS_DEBUG + if (sc->sc_debug) + printf("cas_mii_writereg: phy %d reg %d val %x\n", + phy, reg, val); +#endif +#endif + +#if 0 + /* Select the desired PHY in the MIF configuration register */ + v = bus_space_read_4(t, mif, CAS_MIF_CONFIG); + /* Clear PHY select bit */ + v &= ~CAS_MIF_CONFIG_PHY_SEL; + if (phy == CAS_PHYAD_EXTERNAL) + /* Set PHY select bit to get at external device */ + v |= CAS_MIF_CONFIG_PHY_SEL; + bus_space_write_4(t, mif, CAS_MIF_CONFIG, v); +#endif + /* Construct the frame command */ + v = CAS_MIF_FRAME_WRITE | + (phy << CAS_MIF_PHY_SHIFT) | + (reg << CAS_MIF_REG_SHIFT) | + (val & CAS_MIF_FRAME_DATA); + + bus_space_write_4(t, mif, CAS_MIF_FRAME, v); + for (n = 0; n < 100; n++) { + DELAY(1); + v = bus_space_read_4(t, mif, CAS_MIF_FRAME); + if (v & CAS_MIF_FRAME_TA0) + return; + } + + printf("%s: mii_write timeout\n", sc->sc_dev.dv_xname); +} + +void +cas_mii_statchg(struct device *dev) +{ + struct cas_softc *sc = (void *)dev; +#ifdef CAS_DEBUG + int instance = IFM_INST(sc->sc_mii.mii_media.ifm_cur->ifm_media); +#endif + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t mac = sc->sc_memh; + u_int32_t v; + +#ifdef CAS_DEBUG + if (sc->sc_debug) + printf("cas_mii_statchg: status change: phy = %d\n", + sc->sc_phys[instance]); +#endif + + /* Set tx full duplex options */ + bus_space_write_4(t, mac, CAS_MAC_TX_CONFIG, 0); + delay(10000); /* reg must be cleared and delay before changing. */ + v = CAS_MAC_TX_ENA_IPG0|CAS_MAC_TX_NGU|CAS_MAC_TX_NGU_LIMIT| + CAS_MAC_TX_ENABLE; + if ((IFM_OPTIONS(sc->sc_mii.mii_media_active) & IFM_FDX) != 0) { + v |= CAS_MAC_TX_IGN_CARRIER|CAS_MAC_TX_IGN_COLLIS; + } + bus_space_write_4(t, mac, CAS_MAC_TX_CONFIG, v); + + /* XIF Configuration */ + v = CAS_MAC_XIF_TX_MII_ENA; + v |= CAS_MAC_XIF_LINK_LED; + +#if 0 + /* If an external transceiver is connected, enable its MII drivers */ + sc->sc_mif_config = bus_space_read_4(t, mac, CAS_MIF_CONFIG); +#endif + + /* MII needs echo disable if half duplex. */ + if ((IFM_OPTIONS(sc->sc_mii.mii_media_active) & IFM_FDX) != 0) + /* turn on full duplex LED */ + v |= CAS_MAC_XIF_FDPLX_LED; + else + /* half duplex -- disable echo */ + v |= CAS_MAC_XIF_ECHO_DISABL; + + switch (IFM_SUBTYPE(sc->sc_mii.mii_media_active)) { + case IFM_1000_T: /* Gigabit using GMII interface */ + case IFM_1000_SX: + v |= CAS_MAC_XIF_GMII_MODE; + break; + default: + v &= ~CAS_MAC_XIF_GMII_MODE; + } + bus_space_write_4(t, mac, CAS_MAC_XIF_CONFIG, v); +} + +int +cas_pcs_readreg(struct device *self, int phy, int reg) +{ + struct cas_softc *sc = (void *)self; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t pcs = sc->sc_memh; + +#ifdef CAS_DEBUG + if (sc->sc_debug) + printf("cas_pcs_readreg: phy %d reg %d\n", phy, reg); +#endif + + if (phy != CAS_PHYAD_EXTERNAL) + return (0); + + switch (reg) { + case MII_BMCR: + reg = CAS_MII_CONTROL; + break; + case MII_BMSR: + reg = CAS_MII_STATUS; + break; + case MII_ANAR: + reg = CAS_MII_ANAR; + break; + case MII_ANLPAR: + reg = CAS_MII_ANLPAR; + break; + case MII_EXTSR: + return (EXTSR_1000XFDX|EXTSR_1000XHDX); + default: + return (0); + } + + return bus_space_read_4(t, pcs, reg); +} + +void +cas_pcs_writereg(struct device *self, int phy, int reg, int val) +{ + struct cas_softc *sc = (void *)self; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t pcs = sc->sc_memh; + +#ifdef CAS_DEBUG + if (sc->sc_debug) + printf("cas_pcs_writereg: phy %d reg %d val %x\n", + phy, reg, val); +#endif + + if (phy != CAS_PHYAD_EXTERNAL) + return; + + switch (reg) { + case MII_BMCR: + reg = CAS_MII_CONTROL; + break; + case MII_BMSR: + reg = CAS_MII_STATUS; + break; + case MII_ANAR: + reg = CAS_MII_ANAR; + break; + case MII_ANLPAR: + reg = CAS_MII_ANLPAR; + break; + default: + return; + } + + bus_space_write_4(t, pcs, reg, val); + + if (reg == CAS_MII_ANAR) { + bus_space_write_4(t, pcs, CAS_MII_SLINK_CONTROL, + CAS_MII_SLINK_LOOPBACK|CAS_MII_SLINK_EN_SYNC_D); + bus_space_write_4(t, pcs, CAS_MII_CONFIG, + CAS_MII_CONFIG_ENABLE); + } +} + +int +cas_mediachange(struct ifnet *ifp) +{ + struct cas_softc *sc = ifp->if_softc; + struct mii_data *mii = &sc->sc_mii; + + if (mii->mii_instance) { + struct mii_softc *miisc; + LIST_FOREACH(miisc, &mii->mii_phys, mii_list) + mii_phy_reset(miisc); + } + + return (mii_mediachg(&sc->sc_mii)); +} + +void +cas_mediastatus(struct ifnet *ifp, struct ifmediareq *ifmr) +{ + struct cas_softc *sc = ifp->if_softc; + + mii_pollstat(&sc->sc_mii); + ifmr->ifm_active = sc->sc_mii.mii_media_active; + ifmr->ifm_status = sc->sc_mii.mii_media_status; +} + +/* + * Process an ioctl request. + */ +int +cas_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data) +{ + struct cas_softc *sc = ifp->if_softc; + struct ifaddr *ifa = (struct ifaddr *)data; + struct ifreq *ifr = (struct ifreq *)data; + int s, error = 0; + + s = splnet(); + + if ((error = ether_ioctl(ifp, &sc->sc_arpcom, cmd, data)) > 0) { + splx(s); + return (error); + } + + switch (cmd) { + + case SIOCSIFADDR: + ifp->if_flags |= IFF_UP; + if ((ifp->if_flags & IFF_RUNNING) == 0) + cas_init(ifp); +#ifdef INET + if (ifa->ifa_addr->sa_family == AF_INET) + arp_ifinit(&sc->sc_arpcom, ifa); +#endif + break; + + case SIOCSIFFLAGS: + if (ifp->if_flags & IFF_UP) { + if ((ifp->if_flags & IFF_RUNNING) && + ((ifp->if_flags ^ sc->sc_if_flags) & + (IFF_ALLMULTI | IFF_PROMISC)) != 0) + cas_setladrf(sc); + else { + if ((ifp->if_flags & IFF_RUNNING) == 0) + cas_init(ifp); + } + } else { + if (ifp->if_flags & IFF_RUNNING) + cas_stop(ifp, 1); + } + sc->sc_if_flags = ifp->if_flags; + +#ifdef CAS_DEBUG + sc->sc_debug = (ifp->if_flags & IFF_DEBUG) != 0 ? 1 : 0; +#endif + break; + + case SIOCSIFMTU: + if (ifr->ifr_mtu > ETHERMTU || ifr->ifr_mtu < ETHERMIN) { + error = EINVAL; + } else if (ifp->if_mtu != ifr->ifr_mtu) { + ifp->if_mtu = ifr->ifr_mtu; + } + break; + + case SIOCADDMULTI: + case SIOCDELMULTI: + error = (cmd == SIOCADDMULTI) ? + ether_addmulti(ifr, &sc->sc_arpcom) : + ether_delmulti(ifr, &sc->sc_arpcom); + + if (error == ENETRESET) { + /* + * Multicast list has changed; set the hardware filter + * accordingly. + */ + if (ifp->if_flags & IFF_RUNNING) + cas_setladrf(sc); + error = 0; + } + break; + + case SIOCGIFMEDIA: + case SIOCSIFMEDIA: + error = ifmedia_ioctl(ifp, ifr, &sc->sc_media, cmd); + break; + + default: + error = EINVAL; + break; + } + + splx(s); + return (error); +} + + +void +cas_shutdown(void *arg) +{ + struct cas_softc *sc = (struct cas_softc *)arg; + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + + cas_stop(ifp, 1); +} + +/* + * Set up the logical address filter. + */ +void +cas_setladrf(struct cas_softc *sc) +{ + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + struct ether_multi *enm; + struct ether_multistep step; + struct arpcom *ac = &sc->sc_arpcom; + bus_space_tag_t t = sc->sc_memt; + bus_space_handle_t h = sc->sc_memh; + u_int32_t crc, hash[16], v; + int i; + + /* Get current RX configuration */ + v = bus_space_read_4(t, h, CAS_MAC_RX_CONFIG); + + + /* + * Turn off promiscuous mode, promiscuous group mode (all multicast), + * and hash filter. Depending on the case, the right bit will be + * enabled. + */ + v &= ~(CAS_MAC_RX_PROMISCUOUS|CAS_MAC_RX_HASH_FILTER| + CAS_MAC_RX_PROMISC_GRP); + + if ((ifp->if_flags & IFF_PROMISC) != 0) { + /* Turn on promiscuous mode */ + v |= CAS_MAC_RX_PROMISCUOUS; + ifp->if_flags |= IFF_ALLMULTI; + goto chipit; + } + + /* + * Set up multicast address filter by passing all multicast addresses + * through a crc generator, and then using the high order 8 bits as an + * index into the 256 bit logical address filter. The high order 4 + * bits selects the word, while the other 4 bits select the bit within + * the word (where bit 0 is the MSB). + */ + + /* Clear hash table */ + for (i = 0; i < 16; i++) + hash[i] = 0; + + + ETHER_FIRST_MULTI(step, ac, enm); + while (enm != NULL) { + if (bcmp(enm->enm_addrlo, enm->enm_addrhi, ETHER_ADDR_LEN)) { + /* + * We must listen to a range of multicast addresses. + * For now, just accept all multicasts, rather than + * trying to set only those filter bits needed to match + * the range. (At this time, the only use of address + * ranges is for IP multicast routing, for which the + * range is big enough to require all bits set.) + * XXX use the addr filter for this + */ + ifp->if_flags |= IFF_ALLMULTI; + v |= CAS_MAC_RX_PROMISC_GRP; + goto chipit; + } + + crc = ether_crc32_le(enm->enm_addrlo, ETHER_ADDR_LEN); + + /* Just want the 8 most significant bits. */ + crc >>= 24; + + /* Set the corresponding bit in the filter. */ + hash[crc >> 4] |= 1 << (15 - (crc & 15)); + + ETHER_NEXT_MULTI(step, enm); + } + + v |= CAS_MAC_RX_HASH_FILTER; + ifp->if_flags &= ~IFF_ALLMULTI; + + /* Now load the hash table into the chip (if we are using it) */ + for (i = 0; i < 16; i++) { + bus_space_write_4(t, h, + CAS_MAC_HASH0 + i * (CAS_MAC_HASH1-CAS_MAC_HASH0), + hash[i]); + } + +chipit: + bus_space_write_4(t, h, CAS_MAC_RX_CONFIG, v); +} + +int +cas_encap(struct cas_softc *sc, struct mbuf *mhead, u_int32_t *bixp) +{ + u_int64_t flags; + u_int32_t cur, frag, i; + bus_dmamap_t map; + + cur = frag = *bixp; + map = sc->sc_txd[cur].sd_map; + + if (bus_dmamap_load_mbuf(sc->sc_dmatag, map, mhead, + BUS_DMA_NOWAIT) != 0) { + return (ENOBUFS); + } + + if ((sc->sc_tx_cnt + map->dm_nsegs) > (CAS_NTXDESC - 2)) { + bus_dmamap_unload(sc->sc_dmatag, map); + return (ENOBUFS); + } + + bus_dmamap_sync(sc->sc_dmatag, map, 0, map->dm_mapsize, + BUS_DMASYNC_PREWRITE); + + for (i = 0; i < map->dm_nsegs; i++) { + sc->sc_txdescs[frag].gd_addr = + CAS_DMA_WRITE(sc, map->dm_segs[i].ds_addr); + flags = (map->dm_segs[i].ds_len & CAS_TD_BUFSIZE) | + (i == 0 ? CAS_TD_START_OF_PACKET : 0) | + ((i == (map->dm_nsegs - 1)) ? CAS_TD_END_OF_PACKET : 0); + sc->sc_txdescs[frag].gd_flags = CAS_DMA_WRITE(sc, flags); + bus_dmamap_sync(sc->sc_dmatag, sc->sc_cddmamap, + CAS_CDTXOFF(frag), sizeof(struct cas_desc), + BUS_DMASYNC_PREWRITE); + cur = frag; + if (++frag == CAS_NTXDESC) + frag = 0; + } + + sc->sc_tx_cnt += map->dm_nsegs; + sc->sc_txd[*bixp].sd_map = sc->sc_txd[cur].sd_map; + sc->sc_txd[cur].sd_map = map; + sc->sc_txd[cur].sd_mbuf = mhead; + + bus_space_write_4(sc->sc_memt, sc->sc_memh, CAS_TX_KICK, frag); + + *bixp = frag; + + /* sync descriptors */ + + return (0); +} + +/* + * Transmit interrupt. + */ +int +cas_tint(struct cas_softc *sc, u_int32_t status) +{ + struct ifnet *ifp = &sc->sc_arpcom.ac_if; + struct cas_sxd *sd; + u_int32_t cons, hwcons; + + hwcons = status >> 19; + cons = sc->sc_tx_cons; + while (cons != hwcons) { + sd = &sc->sc_txd[cons]; + if (sd->sd_mbuf != NULL) { + bus_dmamap_sync(sc->sc_dmatag, sd->sd_map, 0, + sd->sd_map->dm_mapsize, BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(sc->sc_dmatag, sd->sd_map); + m_freem(sd->sd_mbuf); + sd->sd_mbuf = NULL; + } + sc->sc_tx_cnt--; + ifp->if_opackets++; + if (++cons == CAS_NTXDESC) + cons = 0; + } + sc->sc_tx_cons = cons; + + cas_start(ifp); + + if (sc->sc_tx_cnt == 0) + ifp->if_timer = 0; + + return (1); +} + +void +cas_start(struct ifnet *ifp) +{ + struct cas_softc *sc = ifp->if_softc; + struct mbuf *m; + u_int32_t bix; + + if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) + return; + + bix = sc->sc_tx_prod; + while (sc->sc_txd[bix].sd_mbuf == NULL) { + IFQ_POLL(&ifp->if_snd, m); + if (m == NULL) + break; + +#if NBPFILTER > 0 + /* + * If BPF is listening on this interface, let it see the + * packet before we commit it to the wire. + */ + if (ifp->if_bpf) + bpf_mtap(ifp->if_bpf, m, BPF_DIRECTION_OUT); +#endif + + /* + * Encapsulate this packet and start it going... + * or fail... + */ + if (cas_encap(sc, m, &bix)) { + ifp->if_timer = 2; + break; + } + + IFQ_DEQUEUE(&ifp->if_snd, m); + ifp->if_timer = 5; + } + + sc->sc_tx_prod = bix; +} diff --git a/sys/dev/pci/if_casreg.h b/sys/dev/pci/if_casreg.h new file mode 100644 index 00000000000..414c4a3639a --- /dev/null +++ b/sys/dev/pci/if_casreg.h @@ -0,0 +1,555 @@ +/* $OpenBSD: if_casreg.h,v 1.1 2007/02/24 20:13:34 kettenis Exp $ */ + +/* + * + * Copyright (C) 2001 Eduardo Horvath. + * 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 ``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. + * + */ + +#ifndef _IF_CASREG_H +#define _IF_CASREG_H + +/* + * Register definitions for Sun Cassini ethernet controllers. + */ + +/* + * First bank: this registers live at the start of the PCI + * mapping, and at the start of the second bank of the SBUS + * version. + */ +#define CAS_SEB_STATE 0x0000 /* SEB state reg, R/O */ +#define CAS_CONFIG 0x0004 /* config reg */ +#define CAS_STATUS 0x000c /* status reg */ +/* Note: Reading the status reg clears bits 0-6 */ +#define CAS_INTMASK 0x0010 +#define CAS_INTACK 0x0014 /* Interrupt acknowledge, W/O */ +#define CAS_STATUS_ALIAS 0x001c +/* Note: Same as CAS_STATUS but reading it does not clear bits. */ + +#define CAS_ERROR_STATUS 0x1000 /* PCI error status R/C */ +#define CAS_ERROR_MASK 0x0004 +#define CAS_BIF_CONFIG 0x0008 /* BIF config reg */ +#define CAS_BIF_DIAG 0x000c +#define CAS_RESET 0x1010 /* Software reset register */ + +/* Bits in CAS_SEB register */ +#define CAS_SEB_ARB 0x000000002 /* Arbitration status */ +#define CAS_SEB_RXWON 0x000000004 + +/* Bits in CAS_CONFIG register */ +#define CAS_CONFIG_BURST_64 0x000000000 /* 0->infinity, 1->64KB */ +#define CAS_CONFIG_BURST_INF 0x000000001 /* 0->infinity, 1->64KB */ +#define CAS_CONFIG_TXDMA_LIMIT 0x00000003e +#define CAS_CONFIG_RXDMA_LIMIT 0x0000007c0 + +#define CAS_CONFIG_TXDMA_LIMIT_SHIFT 1 +#define CAS_CONFIG_RXDMA_LIMIT_SHIFT 6 + +/* Top part of CAS_STATUS has TX completion information */ +#define CAS_STATUS_TX_COMPL 0xfff800000 /* TX completion reg. */ + +/* + * Interrupt bits, for both the CAS_STATUS and CAS_INTMASK regs. + * Bits 0-6 auto-clear when read. + */ +#define CAS_INTR_TX_INTME 0x000000001 /* Frame w/INTME bit set sent */ +#define CAS_INTR_TX_EMPTY 0x000000002 /* TX ring empty */ +#define CAS_INTR_TX_DONE 0x000000004 /* TX complete */ +#define CAS_INTR_RX_DONE 0x000000010 /* Got a packet */ +#define CAS_INTR_RX_NOBUF 0x000000020 +#define CAS_INTR_RX_TAG_ERR 0x000000040 +#define CAS_INTR_PCS 0x000002000 /* Physical Code Sub-layer */ +#define CAS_INTR_TX_MAC 0x000004000 +#define CAS_INTR_RX_MAC 0x000008000 +#define CAS_INTR_MAC_CONTROL 0x000010000 /* MAC control interrupt */ +#define CAS_INTR_MIF 0x000020000 +#define CAS_INTR_BERR 0x000040000 /* Bus error interrupt */ +#define CAS_INTR_BITS "\020" \ + "\1INTME\2TXEMPTY\3TXDONE" \ + "\5RXDONE\6RXNOBUF\7RX_TAG_ERR" \ + "\16PCS\17TXMAC\20RXMAC" \ + "\21MACCONTROL\22MIF\23BERR" + +/* CAS_ERROR_STATUS and CAS_ERROR_MASK PCI error bits */ +#define CAS_ERROR_STAT_BADACK 0x000000001 /* No ACK64# */ +#define CAS_ERROR_STAT_DTRTO 0x000000002 /* Delayed xaction timeout */ +#define CAS_ERROR_STAT_OTHERS 0x000000004 + +/* CAS_BIF_CONFIG register bits */ +#define CAS_BIF_CONFIG_SLOWCLK 0x000000001 /* Parity error timing */ +#define CAS_BIF_CONFIG_HOST_64 0x000000002 /* 64-bit host */ +#define CAS_BIF_CONFIG_B64D_DIS 0x000000004 /* no 64-bit data cycle */ +#define CAS_BIF_CONFIG_M66EN 0x000000008 + +/* CAS_RESET register bits -- TX and RX self clear when complete. */ +#define CAS_RESET_TX 0x000000001 /* Reset TX half */ +#define CAS_RESET_RX 0x000000002 /* Reset RX half */ +#define CAS_RESET_RSTOUT 0x000000004 /* Force PCI RSTOUT# */ + +/* GEM TX DMA registers */ +#define CAS_TX_CONFIG 0x2004 + +#define CAS_TX_FIFO_WR_PTR 0x2014 /* FIFO write pointer */ +#define CAS_TX_FIFO_SDWR_PTR 0x2018 /* FIFO shadow write pointer */ +#define CAS_TX_FIFO_RD_PTR 0x201c /* FIFO read pointer */ +#define CAS_TX_FIFO_SDRD_PTR 0x2020 /* FIFO shadow read pointer */ +#define CAS_TX_FIFO_PKT_CNT 0x2024 /* FIFO packet counter */ + +#define CAS_TX_STATE_MACHINE 0x2028 /* ETX state machine reg */ +#define CAS_TX_DATA_PTR 0x2030 /* ETX state machine reg (64-bit)*/ + +#define CAS_TX_KICK1 0x2038 /* Write last valid desc + 1 */ +#define CAS_TX_KICK2 0x203c +#define CAS_TX_KICK3 0x2040 +#define CAS_TX_KICK4 0x2044 +#define CAS_TX_COMPLETION1 0x2048 +#define CAS_TX_COMPLETION2 0x2048 +#define CAS_TX_COMPLETION3 0x2048 +#define CAS_TX_COMPLETION4 0x2048 +#define CAS_TX_RING_PTR_LO1 0x2060 +#define CAS_TX_RING_PTR_HI1 0x2064 +#define CAS_TX_RING_PTR_LO2 0x2068 +#define CAS_TX_RING_PTR_HI2 0x206c +#define CAS_TX_RING_PTR_LO3 0x2070 +#define CAS_TX_RING_PTR_HI3 0x2074 +#define CAS_TX_RING_PTR_LO4 0x2078 +#define CAS_TX_RING_PTR_HI4 0x207c +#define CAS_TX_MAXBURST1 0x2080 +#define CAS_TX_MAXBURST2 0x2084 +#define CAS_TX_MAXBURST3 0x2088 +#define CAS_TX_MAXBURST4 0x208c + +#define CAS_TX_KICK CAS_TX_KICK3 +#define CAS_TX_RING_PTR_LO CAS_TX_RING_PTR_LO3 +#define CAS_TX_RING_PTR_HI CAS_TX_RING_PTR_HI3 + +#define CAS_TX_FIFO_ADDRESS 0x2104 +#define CAS_TX_FIFO_TAG 0x2108 +#define CAS_TX_FIFO_DATA_LO 0x210c +#define CAS_TX_FIFO_DATA_HI_T1 0x2110 +#define CAS_TX_FIFO_DATA_HI_T0 0x2114 +#define CAS_TX_FIFO_SIZE 0x2118 +#define CAS_TX_DEBUG 0x3028 + +/* CAS_TX_CONFIG register bits. */ +#define CAS_TX_CONFIG_TXDMA_EN 0x00000001 /* TX DMA enable */ +#define CAS_TX_CONFIG_TXRING_SZ 0x0000003c /* TX ring size */ +#define CAS_TX_CONFIG_PACED 0x00100000 /* TX_all_int modifier */ + +#define CAS_RING_SZ_32 0 /* 32 descriptors */ +#define CAS_RING_SZ_64 1 +#define CAS_RING_SZ_128 2 +#define CAS_RING_SZ_256 3 +#define CAS_RING_SZ_512 4 +#define CAS_RING_SZ_1024 5 +#define CAS_RING_SZ_2048 6 +#define CAS_RING_SZ_4096 7 +#define CAS_RING_SZ_8192 8 + +/* CAS_TX_COMPLETION register bits */ +#define CAS_TX_COMPLETION_MASK 0x00001fff /* # of last descriptor */ + +/* GEM RX DMA registers */ +#define CAS_RX_CONFIG 0x4000 +#define CAS_RX_RING_PTR_LO 0x4004 /* 64-bits unaligned GAK! */ +#define CAS_RX_RING_PTR_HI 0x4008 /* 64-bits unaligned GAK! */ + +#define CAS_RX_FIFO_WR_PTR 0x400c /* FIFO write pointer */ +#define CAS_RX_FIFO_SDWR_PTR 0x4010 /* FIFO shadow write pointer */ +#define CAS_RX_FIFO_RD_PTR 0x4014 /* FIFO read pointer */ +#define CAS_RX_FIFO_PKT_CNT 0x4018 /* FIFO packet counter */ + +#define CAS_RX_STATE_MACHINE 0x401c /* ERX state machine reg */ +#define CAS_RX_PAUSE_THRESH 0x4020 + +#define CAS_RX_DATA_PTR_LO 0x4024 /* ERX state machine reg */ +#define CAS_RX_DATA_PTR_HI 0x4028 /* Damn thing is unaligned */ + +#define CAS_RX_KICK 0x4100 /* Write last valid desc + 1 */ +#define CAS_RX_COMPLETION 0x4104 /* First pending desc */ +#define CAS_RX_BLANKING 0x4108 /* Interrupt blanking reg */ + +#define CAS_RX_FIFO_ADDRESS 0x410c +#define CAS_RX_FIFO_TAG 0x4110 +#define CAS_RX_FIFO_DATA_LO 0x4114 +#define CAS_RX_FIFO_DATA_HI_T1 0x4118 +#define CAS_RX_FIFO_DATA_HI_T0 0x411c +#define CAS_RX_FIFO_SIZE 0x4120 + +/* CAS_RX_CONFIG register bits. */ +#define CAS_RX_CONFIG_RXDMA_EN 0x00000001 /* RX DMA enable */ +#define CAS_RX_CONFIG_RXRING_SZ 0x0000001e /* RX ring size */ +#define CAS_RX_CONFIG_BATCH_DIS 0x00000020 /* desc batching disable */ +#define CAS_RX_CONFIG_FBOFF 0x00001c00 /* first byte offset */ +#define CAS_RX_CONFIG_CXM_START 0x000fe000 /* cksum start offset bytes */ +#define CAS_RX_CONFIG_FIFO_THRS 0x07000000 /* fifo threshold size */ + +#define CAS_THRSH_64 0 +#define CAS_THRSH_128 1 +#define CAS_THRSH_256 2 +#define CAS_THRSH_512 3 +#define CAS_THRSH_1024 4 +#define CAS_THRSH_2048 5 + +#define CAS_RX_CONFIG_FIFO_THRS_SHIFT 24 +#define CAS_RX_CONFIG_FBOFF_SHFT 10 +#define CAS_RX_CONFIG_CXM_START_SHFT 13 + +/* CAS_RX_PAUSE_THRESH register bits -- sizes in multiples of 64 bytes */ +#define CAS_RX_PTH_XOFF_THRESH 0x000001ff +#define CAS_RX_PTH_XON_THRESH 0x07fc0000 + +/* CAS_RX_BLANKING register bits */ +#define CAS_RX_BLANKING_PACKETS 0x000001ff /* Delay intr for x packets */ +#define CAS_RX_BLANKING_TIME 0x03fc0000 /* Delay intr for x ticks */ +/* One tick is 1048 PCI clocs, or 16us at 66MHz */ + +/* CAS_MAC registers */ +#define CAS_MAC_TXRESET 0x6000 /* Store 1, cleared when done */ +#define CAS_MAC_RXRESET 0x6004 /* ditto */ +#define CAS_MAC_SEND_PAUSE_CMD 0x6008 +#define CAS_MAC_TX_STATUS 0x6010 +#define CAS_MAC_RX_STATUS 0x6014 +#define CAS_MAC_CONTROL_STATUS 0x6018 /* MAC control status reg */ +#define CAS_MAC_TX_MASK 0x6020 /* TX MAC mask register */ +#define CAS_MAC_RX_MASK 0x6024 +#define CAS_MAC_CONTROL_MASK 0x6028 +#define CAS_MAC_TX_CONFIG 0x6030 +#define CAS_MAC_RX_CONFIG 0x6034 +#define CAS_MAC_CONTROL_CONFIG 0x6038 +#define CAS_MAC_XIF_CONFIG 0x603c +#define CAS_MAC_IPG0 0x6040 /* inter packet gap 0 */ +#define CAS_MAC_IPG1 0x6044 /* inter packet gap 1 */ +#define CAS_MAC_IPG2 0x6048 /* inter packet gap 2 */ +#define CAS_MAC_SLOT_TIME 0x604c /* slot time, bits 0-7 */ +#define CAS_MAC_MAC_MIN_FRAME 0x6050 +#define CAS_MAC_MAC_MAX_FRAME 0x6054 +#define CAS_MAC_PREAMBLE_LEN 0x6058 +#define CAS_MAC_JAM_SIZE 0x605c +#define CAS_MAC_ATTEMPT_LIMIT 0x6060 +#define CAS_MAC_CONTROL_TYPE 0x6064 + +#define CAS_MAC_ADDR0 0x6080 /* Normal MAC address 0 */ +#define CAS_MAC_ADDR1 0x6084 +#define CAS_MAC_ADDR2 0x6088 +#define CAS_MAC_ADDR3 0x608c /* Alternate MAC address 0 */ +#define CAS_MAC_ADDR4 0x6090 +#define CAS_MAC_ADDR5 0x6094 +#define CAS_MAC_ADDR42 0x6128 /* Control MAC address 0 */ +#define CAS_MAC_ADDR43 0x612c +#define CAS_MAC_ADDR44 0x6130 + +#define CAS_MAC_ADDR_FILTER0 0x614c +#define CAS_MAC_ADDR_FILTER1 0x6150 +#define CAS_MAC_ADDR_FILTER2 0x6154 +#define CAS_MAC_ADR_FLT_MASK1_2 0x6158 /* Address filter mask 1,2 */ +#define CAS_MAC_ADR_FLT_MASK0 0x615c /* Address filter mask 0 reg */ + +#define CAS_MAC_HASH0 0x6160 /* Hash table 0 */ +#define CAS_MAC_HASH1 0x6164 +#define CAS_MAC_HASH2 0x6168 +#define CAS_MAC_HASH3 0x616c +#define CAS_MAC_HASH4 0x6170 +#define CAS_MAC_HASH5 0x6174 +#define CAS_MAC_HASH6 0x6178 +#define CAS_MAC_HASH7 0x617c +#define CAS_MAC_HASH8 0x6180 +#define CAS_MAC_HASH9 0x6184 +#define CAS_MAC_HASH10 0x6188 +#define CAS_MAC_HASH11 0x618c +#define CAS_MAC_HASH12 0x6190 +#define CAS_MAC_HASH13 0x6194 +#define CAS_MAC_HASH14 0x6198 +#define CAS_MAC_HASH15 0x619c + +#define CAS_MAC_NORM_COLL_CNT 0x61a0 /* Normal collision counter */ +#define CAS_MAC_FIRST_COLL_CNT 0x61a4 /* 1st successful collision cntr */ +#define CAS_MAC_EXCESS_COLL_CNT 0x61a8 /* Excess collision counter */ +#define CAS_MAC_LATE_COLL_CNT 0x61ac /* Late collision counter */ +#define CAS_MAC_DEFER_TMR_CNT 0x61b0 /* defer timer counter */ +#define CAS_MAC_PEAK_ATTEMPTS 0x61b4 +#define CAS_MAC_RX_FRAME_COUNT 0x61b8 +#define CAS_MAC_RX_LEN_ERR_CNT 0x61bc +#define CAS_MAC_RX_ALIGN_ERR 0x61c0 +#define CAS_MAC_RX_CRC_ERR_CNT 0x61c4 +#define CAS_MAC_RX_CODE_VIOL 0x61c8 +#define CAS_MAC_RANDOM_SEED 0x61cc +#define CAS_MAC_MAC_STATE 0x61d0 /* MAC sstate machine reg */ + +/* CAS_MAC_SEND_PAUSE_CMD register bits */ +#define CAS_MAC_PAUSE_CMD_TIME 0x0000ffff +#define CAS_MAC_PAUSE_CMD_SEND 0x00010000 + +/* CAS_MAC_TX_STATUS and _MASK register bits */ +#define CAS_MAC_TX_XMIT_DONE 0x00000001 +#define CAS_MAC_TX_UNDERRUN 0x00000002 +#define CAS_MAC_TX_PKT_TOO_LONG 0x00000004 +#define CAS_MAC_TX_NCC_EXP 0x00000008 /* Normal collision cnt exp */ +#define CAS_MAC_TX_ECC_EXP 0x00000010 +#define CAS_MAC_TX_LCC_EXP 0x00000020 +#define CAS_MAC_TX_FCC_EXP 0x00000040 +#define CAS_MAC_TX_DEFER_EXP 0x00000080 +#define CAS_MAC_TX_PEAK_EXP 0x00000100 + +/* CAS_MAC_RX_STATUS and _MASK register bits */ +#define CAS_MAC_RX_DONE 0x00000001 +#define CAS_MAC_RX_OVERFLOW 0x00000002 +#define CAS_MAC_RX_FRAME_CNT 0x00000004 +#define CAS_MAC_RX_ALIGN_EXP 0x00000008 +#define CAS_MAC_RX_CRC_EXP 0x00000010 +#define CAS_MAC_RX_LEN_EXP 0x00000020 +#define CAS_MAC_RX_CVI_EXP 0x00000040 /* Code violation */ + +/* CAS_MAC_CONTROL_STATUS and CAS_MAC_CONTROL_MASK register bits */ +#define CAS_MAC_PAUSED 0x00000001 /* Pause received */ +#define CAS_MAC_PAUSE 0x00000002 /* enter pause state */ +#define CAS_MAC_RESUME 0x00000004 /* exit pause state */ +#define CAS_MAC_PAUSE_TIME 0xffff0000 + +/* CAS_MAC_XIF_CONFIG register bits */ +#define CAS_MAC_XIF_TX_MII_ENA 0x00000001 /* Enable XIF output drivers */ +#define CAS_MAC_XIF_MII_LOOPBK 0x00000002 /* Enable MII loopback mode */ +#define CAS_MAC_XIF_ECHO_DISABL 0x00000004 /* Disable echo */ +#define CAS_MAC_XIF_GMII_MODE 0x00000008 /* Select GMII/MII mode */ +#define CAS_MAC_XIF_MII_BUF_ENA 0x00000010 /* Enable MII recv buffers */ +#define CAS_MAC_XIF_LINK_LED 0x00000020 /* force link LED active */ +#define CAS_MAC_XIF_FDPLX_LED 0x00000040 /* force FDPLX LED active */ + +/* CAS_MAC_SLOT_TIME register bits */ +#define CAS_MAC_SLOT_INT 0x40 +#define CAS_MAC_SLOT_EXT 0x200 /* external phy */ + +/* CAS_MAC_TX_CONFIG register bits */ +#define CAS_MAC_TX_ENABLE 0x00000001 /* TX enable */ +#define CAS_MAC_TX_IGN_CARRIER 0x00000002 /* Ignore carrier sense */ +#define CAS_MAC_TX_IGN_COLLIS 0x00000004 /* ignore collisions */ +#define CAS_MAC_TX_ENA_IPG0 0x00000008 /* extend Rx-to-TX IPG */ +#define CAS_MAC_TX_NGU 0x00000010 /* Never give up */ +#define CAS_MAC_TX_NGU_LIMIT 0x00000020 /* Never give up limit */ +#define CAS_MAC_TX_NO_BACKOFF 0x00000040 +#define CAS_MAC_TX_SLOWDOWN 0x00000080 +#define CAS_MAC_TX_NO_FCS 0x00000100 /* no FCS will be generated */ +#define CAS_MAC_TX_CARR_EXTEND 0x00000200 /* Ena TX Carrier Extension */ +/* Carrier Extension is required for half duplex Gbps operation */ + +/* CAS_MAC_RX_CONFIG register bits */ +#define CAS_MAC_RX_ENABLE 0x00000001 /* RX enable */ +#define CAS_MAC_RX_STRIP_PAD 0x00000002 /* strip pad bytes */ +#define CAS_MAC_RX_STRIP_CRC 0x00000004 +#define CAS_MAC_RX_PROMISCUOUS 0x00000008 /* promiscuous mode */ +#define CAS_MAC_RX_PROMISC_GRP 0x00000010 /* promiscuous group mode */ +#define CAS_MAC_RX_HASH_FILTER 0x00000020 /* enable hash filter */ +#define CAS_MAC_RX_ADDR_FILTER 0x00000040 /* enable address filter */ +#define CAS_MAC_RX_ERRCHK_DIS 0x00000080 /* disable error checking */ +#define CAS_MAC_RX_CARR_EXTEND 0x00000100 /* Ena RX Carrier Extension */ +/* + * Carrier Extension enables reception of packet bursts generated by + * senders with carrier extension enabled. + */ + +/* CAS_MAC_CONTROL_CONFIG bits */ +#define CAS_MAC_CC_TX_PAUSE 0x00000001 /* send pause enabled */ +#define CAS_MAC_CC_RX_PAUSE 0x00000002 /* receive pause enabled */ +#define CAS_MAC_CC_PASS_PAUSE 0x00000004 /* pass pause up */ + +/* Cassini MIF registers */ +/* Bit bang registers use low bit only */ +#define CAS_MIF_BB_CLOCK 0x6200 /* bit bang clock */ +#define CAS_MIF_BB_DATA 0x6204 /* bit bang data */ +#define CAS_MIF_BB_OUTPUT_ENAB 0x6208 +#define CAS_MIF_FRAME 0x620c /* MIF frame - ctl and data */ +#define CAS_MIF_CONFIG 0x6210 +#define CAS_MIF_INTERRUPT_MASK 0x6214 +#define CAS_MIF_BASIC_STATUS 0x6218 +#define CAS_MIF_STATE_MACHINE 0x621c + +/* CAS_MIF_FRAME bits */ +#define CAS_MIF_FRAME_DATA 0x0000ffff +#define CAS_MIF_FRAME_TA0 0x00010000 /* TA bit, 1 for completion */ +#define CAS_MIF_FRAME_TA1 0x00020000 /* TA bits */ +#define CAS_MIF_FRAME_REG_ADDR 0x007c0000 +#define CAS_MIF_FRAME_PHY_ADDR 0x0f800000 /* phy address, should be 0 */ +#define CAS_MIF_FRAME_OP 0x30000000 /* operation - write/read */ +#define CAS_MIF_FRAME_START 0xc0000000 /* START bits */ + +#define CAS_MIF_FRAME_READ 0x60020000 +#define CAS_MIF_FRAME_WRITE 0x50020000 + +#define CAS_MIF_REG_SHIFT 18 +#define CAS_MIF_PHY_SHIFT 23 + +/* CAS_MIF_CONFIG register bits */ +#define CAS_MIF_CONFIG_PHY_SEL 0x00000001 /* PHY select, 0=MDIO0 */ +#define CAS_MIF_CONFIG_POLL_ENA 0x00000002 /* poll enable */ +#define CAS_MIF_CONFIG_BB_ENA 0x00000004 /* bit bang enable */ +#define CAS_MIF_CONFIG_REG_ADR 0x000000f8 /* poll register address */ +#define CAS_MIF_CONFIG_MDI0 0x00000100 /* MDIO_0 Data/MDIO_0 atached */ +#define CAS_MIF_CONFIG_MDI1 0x00000200 /* MDIO_1 Data/MDIO_1 atached */ +#define CAS_MIF_CONFIG_PHY_ADR 0x00007c00 /* poll PHY address */ +/* MDI0 is onboard transceiver MID1 is external, PHYAD for both is 0 */ + +/* CAS_MIF_BASIC_STATUS and CAS_MIF_INTERRUPT_MASK bits */ +#define CAS_MIF_STATUS 0x0000ffff +#define CAS_MIF_BASIC 0xffff0000 +/* + * The Basic part is the last value read in the POLL field of the config + * register. + * + * The status part indicates the bits that have changed. + */ + +/* Cassini PCS/Serial link registers */ +#define CAS_MII_CONTROL 0x9000 +#define CAS_MII_STATUS 0x9004 +#define CAS_MII_ANAR 0x9008 /* MII advertisement reg */ +#define CAS_MII_ANLPAR 0x900c /* Link Partner Ability Reg */ +#define CAS_MII_CONFIG 0x9010 +#define CAS_MII_STATE_MACHINE 0x9014 +#define CAS_MII_INTERRUP_STATUS 0x9018 /* PCS interrupt state */ +#define CAS_MII_DATAPATH_MODE 0x9050 +#define CAS_MII_SLINK_CONTROL 0x9054 /* Serial link control */ +#define CAS_MII_OUTPUT_SELECT 0x9058 +#define CAS_MII_SLINK_STATUS 0x905c /* serial link status */ + +/* CAS_MII_CONTROL bits */ +#define CAS_MII_CONTROL_RESET 0x00008000 +#define CAS_MII_CONTROL_LOOPBK 0x00004000 /* 10-bit i/f loopback */ +#define CAS_MII_CONTROL_1000M 0x00002000 /* speed select, always 0 */ +#define CAS_MII_CONTROL_AUTONEG 0x00001000 /* auto negotiation enabled */ +#define CAS_MII_CONTROL_POWERDN 0x00000800 +#define CAS_MII_CONTROL_ISOLATE 0x00000400 /* isolate phy from mii */ +#define CAS_MII_CONTROL_RAN 0x00000200 /* restart auto negotiation */ +#define CAS_MII_CONTROL_FDUPLEX 0x00000100 /* full duplex, always 0 */ +#define CAS_MII_CONTROL_COL_TST 0x00000080 /* collision test */ + +/* CAS_MII_STATUS reg - PCS "BMSR" (Basic Mode Status Reg) */ +#define CAS_MII_STATUS_GB_FDX 0x00000400 /* can perform GBit FDX */ +#define CAS_MII_STATUS_GB_HDX 0x00000200 /* can perform GBit HDX */ +#define CAS_MII_STATUS_UNK 0x00000100 +#define CAS_MII_STATUS_ANEG_CPT 0x00000020 /* auto negotiate compete */ +#define CAS_MII_STATUS_REM_FLT 0x00000010 /* remote fault detected */ +#define CAS_MII_STATUS_ACFG 0x00000008 /* can auto negotiate */ +#define CAS_MII_STATUS_LINK_STS 0x00000004 /* link status */ +#define CAS_MII_STATUS_JABBER 0x00000002 /* jabber condition detected */ +#define CAS_MII_STATUS_EXTCAP 0x00000001 /* extended register capability */ + +/* CAS_MII_ANAR and CAS_MII_ANLPAR reg bits */ +#define CAS_MII_ANEG_NP 0x00008000 /* next page bit */ +#define CAS_MII_ANEG_ACK 0x00004000 /* ack reception of */ + /* Link Partner Capability */ +#define CAS_MII_ANEG_RF 0x00003000 /* advertise remote fault cap */ +#define CAS_MII_ANEG_ASYM_PAUSE 0x00000100 /* asymmetric pause */ +#define CAS_MII_ANEG_SYM_PAUSE 0x00000080 /* symmetric pause */ +#define CAS_MII_ANEG_HLF_DUPLX 0x00000040 +#define CAS_MII_ANEG_FUL_DUPLX 0x00000020 + +/* CAS_MII_CONFIG reg */ +#define CAS_MII_CONFIG_TIMER 0x0000000e /* link monitor timer values */ +#define CAS_MII_CONFIG_ANTO 0x00000020 /* 10ms ANEG timer override */ +#define CAS_MII_CONFIG_JS 0x00000018 /* Jitter Study, 0 normal + * 1 high freq, 2 low freq */ +#define CAS_MII_CONFIG_SDL 0x00000004 /* Signal Detect active low */ +#define CAS_MII_CONFIG_SDO 0x00000002 /* Signal Detect Override */ +#define CAS_MII_CONFIG_ENABLE 0x00000001 /* Enable PCS */ + +/* + * CAS_MII_STATE_MACHINE + * XXX These are best guesses from observed behavior. + */ +#define CAS_MII_FSM_STOP 0x00000000 /* stopped */ +#define CAS_MII_FSM_RUN 0x00000001 /* running */ +#define CAS_MII_FSM_UNKWN 0x00000100 /* unknown */ +#define CAS_MII_FSM_DONE 0x00000101 /* complete */ + +/* + * CAS_MII_INTERRUP_STATUS reg + * No mask register; mask with the global interrupt mask register. + */ +#define CAS_MII_INTERRUP_LINK 0x00000002 /* PCS link status change */ + +/* CAS_MII_DATAPATH_MODE reg */ +#define CAS_MII_DATAPATH_SERIAL 0x00000001 /* Serial link */ +#define CAS_MII_DATAPATH_SERDES 0x00000002 /* Use PCS via 10bit interfac */ +#define CAS_MII_DATAPATH_MII 0x00000004 /* Use {G}MII, not PCS */ +#define CAS_MII_DATAPATH_MIIOUT 0x00000008 /* enable serial output on GMII */ + +/* CAS_MII_SLINK_CONTROL reg */ +#define CAS_MII_SLINK_LOOPBACK 0x00000001 /* enable loopback at sl, logic + * reversed for SERDES */ +#define CAS_MII_SLINK_EN_SYNC_D 0x00000002 /* enable sync detection */ +#define CAS_MII_SLINK_LOCK_REF 0x00000004 /* lock reference clock */ +#define CAS_MII_SLINK_EMPHASIS 0x00000008 /* enable emphasis */ +#define CAS_MII_SLINK_SELFTEST 0x000001c0 +#define CAS_MII_SLINK_POWER_OFF 0x00000200 /* Power down serial link */ + +/* CAS_MII_SLINK_STATUS reg */ +#define CAS_MII_SLINK_TEST 0x00000000 /* undergoing test */ +#define CAS_MII_SLINK_LOCKED 0x00000001 /* waiting 500us lockrefn */ +#define CAS_MII_SLINK_COMMA 0x00000002 /* waiting for comma detect */ +#define CAS_MII_SLINK_SYNC 0x00000003 /* recv data synchronized */ + +/* Wired GEM PHY addresses */ +#define CAS_PHYAD_INTERNAL 1 +#define CAS_PHYAD_EXTERNAL 0 + +/* + * GEM descriptor table structures. + */ +struct cas_desc { + uint64_t gd_flags; + uint64_t gd_addr; +}; + +/* Transmit flags */ +#define CAS_TD_BUFSIZE 0x0000000000007fffLL +#define CAS_TD_CXSUM_START 0x00000000001f8000LL /* Cxsum start offset */ +#define CAS_TD_CXSUM_STARTSHFT 15 +#define CAS_TD_CXSUM_STUFF 0x000000001fe00000LL /* Cxsum stuff offset */ +#define CAS_TD_CXSUM_STUFFSHFT 21 +#define CAS_TD_CXSUM_ENABLE 0x0000000020000000LL /* Cxsum generation enable */ +#define CAS_TD_END_OF_PACKET 0x0000000040000000LL +#define CAS_TD_START_OF_PACKET 0x0000000080000000LL +#define CAS_TD_INTERRUPT_ME 0x0000000100000000LL /* Interrupt me now */ +#define CAS_TD_NO_CRC 0x0000000200000000LL /* do not insert crc */ +/* + * Only need to set CAS_TD_CXSUM_ENABLE, CAS_TD_CXSUM_STUFF, + * CAS_TD_CXSUM_START, and CAS_TD_INTERRUPT_ME in 1st descriptor of a group. + */ + +/* Receive flags */ +#define CAS_RD_CHECKSUM 0x000000000000ffffLL /* is the complement */ +#define CAS_RD_BUFSIZE 0x000000007fff0000LL +#define CAS_RD_OWN 0x0000000080000000LL /* 1 - owned by h/w */ +#define CAS_RD_HASHVAL 0x0ffff00000000000LL +#define CAS_RD_HASH_PASS 0x1000000000000000LL /* passed hash filter */ +#define CAS_RD_ALTERNATE_MAC 0x2000000000000000LL /* Alternate MAC adrs */ +#define CAS_RD_BAD_CRC 0x4000000000000000LL + +#define CAS_RD_BUFSHIFT 16 +#define CAS_RD_BUFLEN(x) (((x)&CAS_RD_BUFSIZE)>>CAS_RD_BUFSHIFT) + +#endif /* _IF_CASREG_H */ diff --git a/sys/dev/pci/if_casvar.h b/sys/dev/pci/if_casvar.h new file mode 100644 index 00000000000..e50e3aa8ea2 --- /dev/null +++ b/sys/dev/pci/if_casvar.h @@ -0,0 +1,263 @@ +/* $OpenBSD: if_casvar.h,v 1.1 2007/02/24 20:13:34 kettenis Exp $ */ + +/* + * + * Copyright (C) 2001 Eduardo Horvath. + * 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 ``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. + * + */ + +#ifndef _IF_CASVAR_H +#define _IF_CASVAR_H + +#include <sys/queue.h> +#include <sys/timeout.h> + +/* + * Misc. definitions for Sun Cassini ethernet controllers. + */ + +/* + * Transmit descriptor list size. This is arbitrary, but allocate + * enough descriptors for 64 pending transmissions and 16 segments + * per packet. + */ +#define CAS_NTXSEGS 16 + +#define CAS_TXQUEUELEN 64 +#define CAS_NTXDESC (CAS_TXQUEUELEN * CAS_NTXSEGS) +#define CAS_NTXDESC_MASK (CAS_NTXDESC - 1) +#define CAS_NEXTTX(x) ((x + 1) & CAS_NTXDESC_MASK) + +struct cas_sxd { + struct mbuf *sd_mbuf; + bus_dmamap_t sd_map; +}; + +/* + * Receive descriptor list size. We have one Rx buffer per incoming + * packet, so this logic is a little simpler. + */ +#define CAS_NRXDESC 128 +#define CAS_NRXDESC_MASK (CAS_NRXDESC - 1) +#define CAS_NEXTRX(x) ((x + 1) & CAS_NRXDESC_MASK) + +/* + * Control structures are DMA'd to the GEM chip. We allocate them in + * a single clump that maps to a single DMA segment to make several things + * easier. + */ +struct cas_control_data { + /* + * The transmit descriptors. + */ + struct cas_desc gcd_txdescs[CAS_NTXDESC]; + + /* + * The receive descriptors. + */ + struct cas_desc gcd_rxdescs[CAS_NRXDESC]; +}; + +#define CAS_CDOFF(x) offsetof(struct cas_control_data, x) +#define CAS_CDTXOFF(x) CAS_CDOFF(gcd_txdescs[(x)]) +#define CAS_CDRXOFF(x) CAS_CDOFF(gcd_rxdescs[(x)]) + +/* + * Software state for receive jobs. + */ +struct cas_rxsoft { + struct mbuf *rxs_mbuf; /* head of our mbuf chain */ + bus_dmamap_t rxs_dmamap; /* our DMA map */ +}; + + +/* + * Table which describes the transmit threshold mode. We generally + * start at index 0. Whenever we get a transmit underrun, we increment + * our index, falling back if we encounter the NULL terminator. + */ +struct cas_txthresh_tab { + u_int32_t txth_opmode; /* OPMODE bits */ + const char *txth_name; /* name of mode */ +}; + +/* + * Some misc. statics, useful for debugging. + */ +struct cas_stats { + u_long ts_tx_uf; /* transmit underflow errors */ + u_long ts_tx_to; /* transmit jabber timeouts */ + u_long ts_tx_ec; /* excessive collision count */ + u_long ts_tx_lc; /* late collision count */ +}; + +/* + * Software state per device. + */ +struct cas_softc { + struct device sc_dev; /* generic device information */ + struct arpcom sc_arpcom; /* ethernet common data */ + struct mii_data sc_mii; /* MII media control */ +#define sc_media sc_mii.mii_media/* shorthand */ + struct timeout sc_tick_ch; /* tick callout */ + + bus_space_tag_t sc_memt; + bus_space_handle_t sc_memh; + void *sc_ih; + + bus_dma_tag_t sc_dmatag; /* bus dma tag */ + bus_dmamap_t sc_dmamap; /* bus dma handle */ + int sc_burst; /* DVMA burst size in effect */ + int sc_phys[2]; /* MII instance -> PHY map */ + + int sc_if_flags; + + int sc_mif_config; /* Selected MII reg setting */ + + void *sc_sdhook; /* shutdown hook */ + void *sc_powerhook; /* power management hook */ + + struct cas_stats sc_stats; /* debugging stats */ + + /* + * Ring buffer DMA stuff. + */ + bus_dma_segment_t sc_cdseg; /* control data memory */ + int sc_cdnseg; /* number of segments */ + bus_dmamap_t sc_cddmamap; /* control data DMA map */ +#define sc_cddma sc_cddmamap->dm_segs[0].ds_addr + + /* + * Software state for transmit and receive descriptors. + */ + struct cas_sxd sc_txd[CAS_NTXDESC]; + u_int32_t sc_tx_cnt, sc_tx_prod, sc_tx_cons; + + struct cas_rxsoft sc_rxsoft[CAS_NRXDESC]; + + /* + * Control data structures. + */ + struct cas_control_data *sc_control_data; +#define sc_txdescs sc_control_data->gcd_txdescs +#define sc_rxdescs sc_control_data->gcd_rxdescs + + int sc_txfree; /* number of free Tx descriptors */ + int sc_txnext; /* next ready Tx descriptor */ + + u_int32_t sc_tdctl_ch; /* conditional desc chaining */ + u_int32_t sc_tdctl_er; /* conditional desc end-of-ring */ + + u_int32_t sc_setup_fsls; /* FS|LS on setup descriptor */ + + int sc_rxptr; /* next ready RX descriptor/descsoft */ + int sc_rxfifosize; + + /* ========== */ + int sc_inited; + int sc_debug; + void *sc_sh; /* shutdownhook cookie */ +}; + +#define CAS_DMA_READ(sc, v) letoh64(v) +#define CAS_DMA_WRITE(sc, v) htole64(v) + +/* + * This macro returns the current media entry for *non-MII* media. + */ +#define CAS_CURRENT_MEDIA(sc) \ + (IFM_SUBTYPE((sc)->sc_mii.mii_media.ifm_cur->ifm_media) != IFM_AUTO ? \ + (sc)->sc_mii.mii_media.ifm_cur : (sc)->sc_nway_active) + +/* + * This macro determines if a change to media-related OPMODE bits requires + * a chip reset. + */ +#define CAS_MEDIA_NEEDSRESET(sc, newbits) \ + (((sc)->sc_opmode & OPMODE_MEDIA_BITS) != \ + ((newbits) & OPMODE_MEDIA_BITS)) + +#define CAS_CDTXADDR(sc, x) ((sc)->sc_cddma + CAS_CDTXOFF((x))) +#define CAS_CDRXADDR(sc, x) ((sc)->sc_cddma + CAS_CDRXOFF((x))) + +#define CAS_CDSPADDR(sc) ((sc)->sc_cddma + CAS_CDSPOFF) + +#define CAS_CDTXSYNC(sc, x, n, ops) \ +do { \ + int __x, __n; \ + \ + __x = (x); \ + __n = (n); \ + \ + /* If it will wrap around, sync to the end of the ring. */ \ + if ((__x + __n) > CAS_NTXDESC) { \ + bus_dmamap_sync((sc)->sc_dmatag, (sc)->sc_cddmamap, \ + CAS_CDTXOFF(__x), sizeof(struct cas_desc) * \ + (CAS_NTXDESC - __x), (ops)); \ + __n -= (CAS_NTXDESC - __x); \ + __x = 0; \ + } \ + \ + /* Now sync whatever is left. */ \ + bus_dmamap_sync((sc)->sc_dmatag, (sc)->sc_cddmamap, \ + CAS_CDTXOFF(__x), sizeof(struct cas_desc) * __n, (ops)); \ +} while (0) + +#define CAS_CDRXSYNC(sc, x, ops) \ + bus_dmamap_sync((sc)->sc_dmatag, (sc)->sc_cddmamap, \ + CAS_CDRXOFF((x)), sizeof(struct cas_desc), (ops)) + +#define CAS_CDSPSYNC(sc, ops) \ + bus_dmamap_sync((sc)->sc_dmatag, (sc)->sc_cddmamap, \ + CAS_CDSPOFF, CAS_SETUP_PACKET_LEN, (ops)) + +#define CAS_INIT_RXDESC(sc, x) \ +do { \ + struct cas_rxsoft *__rxs = &sc->sc_rxsoft[(x)]; \ + struct cas_desc *__rxd = &sc->sc_rxdescs[(x)]; \ + struct mbuf *__m = __rxs->rxs_mbuf; \ + \ + __m->m_data = __m->m_ext.ext_buf; \ + __rxd->gd_addr = \ + CAS_DMA_WRITE((sc), __rxs->rxs_dmamap->dm_segs[0].ds_addr); \ + __rxd->gd_flags = \ + CAS_DMA_WRITE((sc), \ + (((__m->m_ext.ext_size)<<CAS_RD_BUFSHIFT) \ + & CAS_RD_BUFSIZE) | CAS_RD_OWN); \ + CAS_CDRXSYNC((sc), (x), BUS_DMASYNC_PREREAD|BUS_DMASYNC_PREWRITE); \ +} while (0) + +#ifdef _KERNEL +int cas_mediachange(struct ifnet *); +void cas_mediastatus(struct ifnet *, struct ifmediareq *); + +void cas_config(struct cas_softc *); +void cas_reset(struct cas_softc *); +int cas_intr(void *); +#endif /* _KERNEL */ + + +#endif |