summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/dev/pci/if_cas.c1906
-rw-r--r--sys/dev/pci/if_casreg.h555
-rw-r--r--sys/dev/pci/if_casvar.h263
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