1119051cbSMarius Strobl /*- 24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause 3718cf2ccSPedro F. Giffuni * 4119051cbSMarius Strobl * Copyright (C) 2001 Eduardo Horvath. 5119051cbSMarius Strobl * Copyright (c) 2001-2003 Thomas Moestl 6119051cbSMarius Strobl * Copyright (c) 2007-2009 Marius Strobl <marius@FreeBSD.org> 7119051cbSMarius Strobl * All rights reserved. 8119051cbSMarius Strobl * 9119051cbSMarius Strobl * Redistribution and use in source and binary forms, with or without 10119051cbSMarius Strobl * modification, are permitted provided that the following conditions 11119051cbSMarius Strobl * are met: 12119051cbSMarius Strobl * 1. Redistributions of source code must retain the above copyright 13119051cbSMarius Strobl * notice, this list of conditions and the following disclaimer. 14119051cbSMarius Strobl * 2. Redistributions in binary form must reproduce the above copyright 15119051cbSMarius Strobl * notice, this list of conditions and the following disclaimer in the 16119051cbSMarius Strobl * documentation and/or other materials provided with the distribution. 17119051cbSMarius Strobl * 18119051cbSMarius Strobl * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND 19119051cbSMarius Strobl * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20119051cbSMarius Strobl * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21119051cbSMarius Strobl * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE 22119051cbSMarius Strobl * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 23119051cbSMarius Strobl * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 24119051cbSMarius Strobl * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 25119051cbSMarius Strobl * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 26119051cbSMarius Strobl * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 27119051cbSMarius Strobl * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 28119051cbSMarius Strobl * SUCH DAMAGE. 29119051cbSMarius Strobl * 30119051cbSMarius Strobl * from: NetBSD: gem.c,v 1.21 2002/06/01 23:50:58 lukem Exp 31119051cbSMarius Strobl * from: FreeBSD: if_gem.c 182060 2008-08-23 15:03:26Z marius 32119051cbSMarius Strobl */ 33119051cbSMarius Strobl 34119051cbSMarius Strobl #include <sys/cdefs.h> 35119051cbSMarius Strobl /* 36119051cbSMarius Strobl * driver for Sun Cassini/Cassini+ and National Semiconductor DP83065 37119051cbSMarius Strobl * Saturn Gigabit Ethernet controllers 38119051cbSMarius Strobl */ 39119051cbSMarius Strobl 40119051cbSMarius Strobl #if 0 41119051cbSMarius Strobl #define CAS_DEBUG 42119051cbSMarius Strobl #endif 43119051cbSMarius Strobl 44119051cbSMarius Strobl #include <sys/param.h> 45119051cbSMarius Strobl #include <sys/systm.h> 46119051cbSMarius Strobl #include <sys/bus.h> 47119051cbSMarius Strobl #include <sys/callout.h> 48119051cbSMarius Strobl #include <sys/endian.h> 49119051cbSMarius Strobl #include <sys/mbuf.h> 50119051cbSMarius Strobl #include <sys/malloc.h> 51119051cbSMarius Strobl #include <sys/kernel.h> 52119051cbSMarius Strobl #include <sys/lock.h> 53119051cbSMarius Strobl #include <sys/module.h> 54119051cbSMarius Strobl #include <sys/mutex.h> 55119051cbSMarius Strobl #include <sys/refcount.h> 56119051cbSMarius Strobl #include <sys/resource.h> 57119051cbSMarius Strobl #include <sys/rman.h> 58119051cbSMarius Strobl #include <sys/socket.h> 59119051cbSMarius Strobl #include <sys/sockio.h> 6073591c46SMarius Strobl #include <sys/taskqueue.h> 61119051cbSMarius Strobl 62119051cbSMarius Strobl #include <net/bpf.h> 63119051cbSMarius Strobl #include <net/ethernet.h> 64119051cbSMarius Strobl #include <net/if.h> 6576039bc8SGleb Smirnoff #include <net/if_var.h> 66119051cbSMarius Strobl #include <net/if_arp.h> 67119051cbSMarius Strobl #include <net/if_dl.h> 68119051cbSMarius Strobl #include <net/if_media.h> 69119051cbSMarius Strobl #include <net/if_types.h> 70119051cbSMarius Strobl #include <net/if_vlan_var.h> 71119051cbSMarius Strobl 72119051cbSMarius Strobl #include <netinet/in.h> 73119051cbSMarius Strobl #include <netinet/in_systm.h> 74119051cbSMarius Strobl #include <netinet/ip.h> 75119051cbSMarius Strobl #include <netinet/tcp.h> 76119051cbSMarius Strobl #include <netinet/udp.h> 77119051cbSMarius Strobl 78119051cbSMarius Strobl #include <machine/bus.h> 7958aa35d4SWarner Losh #if defined(__powerpc__) 80917825acSMarius Strobl #include <dev/ofw/ofw_bus.h> 81119051cbSMarius Strobl #include <dev/ofw/openfirm.h> 82119051cbSMarius Strobl #include <machine/ofw_machdep.h> 83119051cbSMarius Strobl #endif 84119051cbSMarius Strobl #include <machine/resource.h> 85119051cbSMarius Strobl 86119051cbSMarius Strobl #include <dev/mii/mii.h> 87119051cbSMarius Strobl #include <dev/mii/miivar.h> 88119051cbSMarius Strobl 89119051cbSMarius Strobl #include <dev/cas/if_casreg.h> 90119051cbSMarius Strobl #include <dev/cas/if_casvar.h> 91119051cbSMarius Strobl 92119051cbSMarius Strobl #include <dev/pci/pcireg.h> 93119051cbSMarius Strobl #include <dev/pci/pcivar.h> 94119051cbSMarius Strobl 95119051cbSMarius Strobl #include "miibus_if.h" 96119051cbSMarius Strobl 97119051cbSMarius Strobl #define RINGASSERT(n , min, max) \ 98119051cbSMarius Strobl CTASSERT(powerof2(n) && (n) >= (min) && (n) <= (max)) 99119051cbSMarius Strobl 100119051cbSMarius Strobl RINGASSERT(CAS_NRXCOMP, 128, 32768); 101119051cbSMarius Strobl RINGASSERT(CAS_NRXDESC, 32, 8192); 102119051cbSMarius Strobl RINGASSERT(CAS_NRXDESC2, 32, 8192); 103119051cbSMarius Strobl RINGASSERT(CAS_NTXDESC, 32, 8192); 104119051cbSMarius Strobl 105119051cbSMarius Strobl #undef RINGASSERT 106119051cbSMarius Strobl 107119051cbSMarius Strobl #define CCDASSERT(m, a) \ 108119051cbSMarius Strobl CTASSERT((offsetof(struct cas_control_data, m) & ((a) - 1)) == 0) 109119051cbSMarius Strobl 110119051cbSMarius Strobl CCDASSERT(ccd_rxcomps, CAS_RX_COMP_ALIGN); 111119051cbSMarius Strobl CCDASSERT(ccd_rxdescs, CAS_RX_DESC_ALIGN); 112119051cbSMarius Strobl CCDASSERT(ccd_rxdescs2, CAS_RX_DESC_ALIGN); 113119051cbSMarius Strobl 114119051cbSMarius Strobl #undef CCDASSERT 115119051cbSMarius Strobl 116119051cbSMarius Strobl #define CAS_TRIES 10000 117119051cbSMarius Strobl 118119051cbSMarius Strobl /* 119119051cbSMarius Strobl * According to documentation, the hardware has support for basic TCP 120119051cbSMarius Strobl * checksum offloading only, in practice this can be also used for UDP 121119051cbSMarius Strobl * however (i.e. the problem of previous Sun NICs that a checksum of 0x0 122119051cbSMarius Strobl * is not converted to 0xffff no longer exists). 123119051cbSMarius Strobl */ 124119051cbSMarius Strobl #define CAS_CSUM_FEATURES (CSUM_TCP | CSUM_UDP) 125119051cbSMarius Strobl 126119051cbSMarius Strobl static inline void cas_add_rxdesc(struct cas_softc *sc, u_int idx); 127119051cbSMarius Strobl static int cas_attach(struct cas_softc *sc); 128119051cbSMarius Strobl static int cas_bitwait(struct cas_softc *sc, bus_addr_t r, uint32_t clr, 129119051cbSMarius Strobl uint32_t set); 130119051cbSMarius Strobl static void cas_cddma_callback(void *xsc, bus_dma_segment_t *segs, 131119051cbSMarius Strobl int nsegs, int error); 132119051cbSMarius Strobl static void cas_detach(struct cas_softc *sc); 133119051cbSMarius Strobl static int cas_disable_rx(struct cas_softc *sc); 134119051cbSMarius Strobl static int cas_disable_tx(struct cas_softc *sc); 135119051cbSMarius Strobl static void cas_eint(struct cas_softc *sc, u_int status); 136e8fd18f3SGleb Smirnoff static void cas_free(struct mbuf *m); 137119051cbSMarius Strobl static void cas_init(void *xsc); 138119051cbSMarius Strobl static void cas_init_locked(struct cas_softc *sc); 139119051cbSMarius Strobl static void cas_init_regs(struct cas_softc *sc); 14073591c46SMarius Strobl static int cas_intr(void *v); 14173591c46SMarius Strobl static void cas_intr_task(void *arg, int pending __unused); 14244e65355SJustin Hibbits static int cas_ioctl(if_t ifp, u_long cmd, caddr_t data); 143119051cbSMarius Strobl static int cas_load_txmbuf(struct cas_softc *sc, struct mbuf **m_head); 14444e65355SJustin Hibbits static int cas_mediachange(if_t ifp); 14544e65355SJustin Hibbits static void cas_mediastatus(if_t ifp, struct ifmediareq *ifmr); 146119051cbSMarius Strobl static void cas_meminit(struct cas_softc *sc); 147119051cbSMarius Strobl static void cas_mifinit(struct cas_softc *sc); 148119051cbSMarius Strobl static int cas_mii_readreg(device_t dev, int phy, int reg); 149119051cbSMarius Strobl static void cas_mii_statchg(device_t dev); 150119051cbSMarius Strobl static int cas_mii_writereg(device_t dev, int phy, int reg, int val); 151119051cbSMarius Strobl static void cas_reset(struct cas_softc *sc); 152119051cbSMarius Strobl static int cas_reset_rx(struct cas_softc *sc); 153119051cbSMarius Strobl static int cas_reset_tx(struct cas_softc *sc); 154119051cbSMarius Strobl static void cas_resume(struct cas_softc *sc); 155119051cbSMarius Strobl static u_int cas_descsize(u_int sz); 156119051cbSMarius Strobl static void cas_rint(struct cas_softc *sc); 157119051cbSMarius Strobl static void cas_rint_timeout(void *arg); 158119051cbSMarius Strobl static inline void cas_rxcksum(struct mbuf *m, uint16_t cksum); 159119051cbSMarius Strobl static inline void cas_rxcompinit(struct cas_rx_comp *rxcomp); 160119051cbSMarius Strobl static u_int cas_rxcompsize(u_int sz); 161119051cbSMarius Strobl static void cas_rxdma_callback(void *xsc, bus_dma_segment_t *segs, 162119051cbSMarius Strobl int nsegs, int error); 163119051cbSMarius Strobl static void cas_setladrf(struct cas_softc *sc); 16444e65355SJustin Hibbits static void cas_start(if_t ifp); 16544e65355SJustin Hibbits static void cas_stop(if_t ifp); 166119051cbSMarius Strobl static void cas_suspend(struct cas_softc *sc); 167119051cbSMarius Strobl static void cas_tick(void *arg); 168119051cbSMarius Strobl static void cas_tint(struct cas_softc *sc); 16973591c46SMarius Strobl static void cas_tx_task(void *arg, int pending __unused); 170119051cbSMarius Strobl static inline void cas_txkick(struct cas_softc *sc); 17173591c46SMarius Strobl static void cas_watchdog(struct cas_softc *sc); 172119051cbSMarius Strobl 173119051cbSMarius Strobl MODULE_DEPEND(cas, ether, 1, 1, 1); 174119051cbSMarius Strobl MODULE_DEPEND(cas, miibus, 1, 1, 1); 175119051cbSMarius Strobl 176119051cbSMarius Strobl #ifdef CAS_DEBUG 177119051cbSMarius Strobl #include <sys/ktr.h> 178651aa2d8SAttilio Rao #define KTR_CAS KTR_SPARE2 179119051cbSMarius Strobl #endif 180119051cbSMarius Strobl 181119051cbSMarius Strobl static int 182119051cbSMarius Strobl cas_attach(struct cas_softc *sc) 183119051cbSMarius Strobl { 184119051cbSMarius Strobl struct cas_txsoft *txs; 18544e65355SJustin Hibbits if_t ifp; 186119051cbSMarius Strobl int error, i; 187119051cbSMarius Strobl uint32_t v; 188119051cbSMarius Strobl 189119051cbSMarius Strobl /* Set up ifnet structure. */ 190119051cbSMarius Strobl ifp = sc->sc_ifp = if_alloc(IFT_ETHER); 19144e65355SJustin Hibbits if_setsoftc(ifp, sc); 192119051cbSMarius Strobl if_initname(ifp, device_get_name(sc->sc_dev), 193119051cbSMarius Strobl device_get_unit(sc->sc_dev)); 19444e65355SJustin Hibbits if_setflags(ifp, IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST); 19544e65355SJustin Hibbits if_setstartfn(ifp, cas_start); 19644e65355SJustin Hibbits if_setioctlfn(ifp, cas_ioctl); 19744e65355SJustin Hibbits if_setinitfn(ifp, cas_init); 19844e65355SJustin Hibbits if_setsendqlen(ifp, CAS_TXQUEUELEN); 19944e65355SJustin Hibbits if_setsendqready(ifp); 200119051cbSMarius Strobl 201119051cbSMarius Strobl callout_init_mtx(&sc->sc_tick_ch, &sc->sc_mtx, 0); 202fcdbb666SMarius Strobl callout_init_mtx(&sc->sc_rx_ch, &sc->sc_mtx, 0); 20373591c46SMarius Strobl /* Create local taskq. */ 2046c3e93cbSGleb Smirnoff NET_TASK_INIT(&sc->sc_intr_task, 0, cas_intr_task, sc); 20573591c46SMarius Strobl TASK_INIT(&sc->sc_tx_task, 1, cas_tx_task, ifp); 20673591c46SMarius Strobl sc->sc_tq = taskqueue_create_fast("cas_taskq", M_WAITOK, 20773591c46SMarius Strobl taskqueue_thread_enqueue, &sc->sc_tq); 2087c2fdcacSMarius Strobl error = taskqueue_start_threads(&sc->sc_tq, 1, PI_NET, "%s taskq", 20973591c46SMarius Strobl device_get_nameunit(sc->sc_dev)); 2107c2fdcacSMarius Strobl if (error != 0) { 2117c2fdcacSMarius Strobl device_printf(sc->sc_dev, "could not start threads\n"); 2127c2fdcacSMarius Strobl goto fail_taskq; 2137c2fdcacSMarius Strobl } 214119051cbSMarius Strobl 215119051cbSMarius Strobl /* Make sure the chip is stopped. */ 216119051cbSMarius Strobl cas_reset(sc); 217119051cbSMarius Strobl 218119051cbSMarius Strobl error = bus_dma_tag_create(bus_get_dma_tag(sc->sc_dev), 1, 0, 219119051cbSMarius Strobl BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 220119051cbSMarius Strobl BUS_SPACE_MAXSIZE, 0, BUS_SPACE_MAXSIZE, 0, NULL, NULL, 221119051cbSMarius Strobl &sc->sc_pdmatag); 222119051cbSMarius Strobl if (error != 0) 22373591c46SMarius Strobl goto fail_taskq; 224119051cbSMarius Strobl 225119051cbSMarius Strobl error = bus_dma_tag_create(sc->sc_pdmatag, 1, 0, 226119051cbSMarius Strobl BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 227119051cbSMarius Strobl CAS_PAGE_SIZE, 1, CAS_PAGE_SIZE, 0, NULL, NULL, &sc->sc_rdmatag); 228119051cbSMarius Strobl if (error != 0) 229119051cbSMarius Strobl goto fail_ptag; 230119051cbSMarius Strobl 231119051cbSMarius Strobl error = bus_dma_tag_create(sc->sc_pdmatag, 1, 0, 232119051cbSMarius Strobl BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 233119051cbSMarius Strobl MCLBYTES * CAS_NTXSEGS, CAS_NTXSEGS, MCLBYTES, 234119051cbSMarius Strobl BUS_DMA_ALLOCNOW, NULL, NULL, &sc->sc_tdmatag); 235119051cbSMarius Strobl if (error != 0) 236119051cbSMarius Strobl goto fail_rtag; 237119051cbSMarius Strobl 238119051cbSMarius Strobl error = bus_dma_tag_create(sc->sc_pdmatag, CAS_TX_DESC_ALIGN, 0, 239119051cbSMarius Strobl BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 240119051cbSMarius Strobl sizeof(struct cas_control_data), 1, 241119051cbSMarius Strobl sizeof(struct cas_control_data), 0, 242119051cbSMarius Strobl NULL, NULL, &sc->sc_cdmatag); 243119051cbSMarius Strobl if (error != 0) 244119051cbSMarius Strobl goto fail_ttag; 245119051cbSMarius Strobl 246119051cbSMarius Strobl /* 247119051cbSMarius Strobl * Allocate the control data structures, create and load the 248119051cbSMarius Strobl * DMA map for it. 249119051cbSMarius Strobl */ 250119051cbSMarius Strobl if ((error = bus_dmamem_alloc(sc->sc_cdmatag, 251119051cbSMarius Strobl (void **)&sc->sc_control_data, 252119051cbSMarius Strobl BUS_DMA_WAITOK | BUS_DMA_COHERENT | BUS_DMA_ZERO, 253119051cbSMarius Strobl &sc->sc_cddmamap)) != 0) { 254119051cbSMarius Strobl device_printf(sc->sc_dev, 255119051cbSMarius Strobl "unable to allocate control data, error = %d\n", error); 256119051cbSMarius Strobl goto fail_ctag; 257119051cbSMarius Strobl } 258119051cbSMarius Strobl 259119051cbSMarius Strobl sc->sc_cddma = 0; 260119051cbSMarius Strobl if ((error = bus_dmamap_load(sc->sc_cdmatag, sc->sc_cddmamap, 261119051cbSMarius Strobl sc->sc_control_data, sizeof(struct cas_control_data), 262119051cbSMarius Strobl cas_cddma_callback, sc, 0)) != 0 || sc->sc_cddma == 0) { 263119051cbSMarius Strobl device_printf(sc->sc_dev, 264119051cbSMarius Strobl "unable to load control data DMA map, error = %d\n", 265119051cbSMarius Strobl error); 266119051cbSMarius Strobl goto fail_cmem; 267119051cbSMarius Strobl } 268119051cbSMarius Strobl 269119051cbSMarius Strobl /* 270119051cbSMarius Strobl * Initialize the transmit job descriptors. 271119051cbSMarius Strobl */ 272119051cbSMarius Strobl STAILQ_INIT(&sc->sc_txfreeq); 273119051cbSMarius Strobl STAILQ_INIT(&sc->sc_txdirtyq); 274119051cbSMarius Strobl 275119051cbSMarius Strobl /* 276119051cbSMarius Strobl * Create the transmit buffer DMA maps. 277119051cbSMarius Strobl */ 278119051cbSMarius Strobl error = ENOMEM; 279119051cbSMarius Strobl for (i = 0; i < CAS_TXQUEUELEN; i++) { 280119051cbSMarius Strobl txs = &sc->sc_txsoft[i]; 281119051cbSMarius Strobl txs->txs_mbuf = NULL; 282119051cbSMarius Strobl txs->txs_ndescs = 0; 283119051cbSMarius Strobl if ((error = bus_dmamap_create(sc->sc_tdmatag, 0, 284119051cbSMarius Strobl &txs->txs_dmamap)) != 0) { 285119051cbSMarius Strobl device_printf(sc->sc_dev, 286119051cbSMarius Strobl "unable to create TX DMA map %d, error = %d\n", 287119051cbSMarius Strobl i, error); 288119051cbSMarius Strobl goto fail_txd; 289119051cbSMarius Strobl } 290119051cbSMarius Strobl STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 291119051cbSMarius Strobl } 292119051cbSMarius Strobl 293119051cbSMarius Strobl /* 294119051cbSMarius Strobl * Allocate the receive buffers, create and load the DMA maps 295119051cbSMarius Strobl * for them. 296119051cbSMarius Strobl */ 297119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) { 298119051cbSMarius Strobl if ((error = bus_dmamem_alloc(sc->sc_rdmatag, 299119051cbSMarius Strobl &sc->sc_rxdsoft[i].rxds_buf, BUS_DMA_WAITOK, 300119051cbSMarius Strobl &sc->sc_rxdsoft[i].rxds_dmamap)) != 0) { 301119051cbSMarius Strobl device_printf(sc->sc_dev, 302119051cbSMarius Strobl "unable to allocate RX buffer %d, error = %d\n", 303119051cbSMarius Strobl i, error); 304119051cbSMarius Strobl goto fail_rxmem; 305119051cbSMarius Strobl } 306119051cbSMarius Strobl 307119051cbSMarius Strobl sc->sc_rxdptr = i; 308119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_paddr = 0; 309119051cbSMarius Strobl if ((error = bus_dmamap_load(sc->sc_rdmatag, 310119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap, sc->sc_rxdsoft[i].rxds_buf, 311119051cbSMarius Strobl CAS_PAGE_SIZE, cas_rxdma_callback, sc, 0)) != 0 || 312119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_paddr == 0) { 313119051cbSMarius Strobl device_printf(sc->sc_dev, 314119051cbSMarius Strobl "unable to load RX DMA map %d, error = %d\n", 315119051cbSMarius Strobl i, error); 316119051cbSMarius Strobl goto fail_rxmap; 317119051cbSMarius Strobl } 318119051cbSMarius Strobl } 319119051cbSMarius Strobl 320917825acSMarius Strobl if ((sc->sc_flags & CAS_SERDES) == 0) { 321119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_DATAPATH, CAS_PCS_DATAPATH_MII); 322917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_DATAPATH, 4, 323917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 324119051cbSMarius Strobl cas_mifinit(sc); 325119051cbSMarius Strobl /* 326119051cbSMarius Strobl * Look for an external PHY. 327119051cbSMarius Strobl */ 328119051cbSMarius Strobl error = ENXIO; 329119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MIF_CONF); 330119051cbSMarius Strobl if ((v & CAS_MIF_CONF_MDI1) != 0) { 331119051cbSMarius Strobl v |= CAS_MIF_CONF_PHY_SELECT; 332119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MIF_CONF, v); 333917825acSMarius Strobl CAS_BARRIER(sc, CAS_MIF_CONF, 4, 334917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 335917825acSMarius Strobl /* Enable/unfreeze the GMII pins of Saturn. */ 336917825acSMarius Strobl if (sc->sc_variant == CAS_SATURN) { 3377c2fdcacSMarius Strobl CAS_WRITE_4(sc, CAS_SATURN_PCFG, 3387c2fdcacSMarius Strobl CAS_READ_4(sc, CAS_SATURN_PCFG) & 3397c2fdcacSMarius Strobl ~CAS_SATURN_PCFG_FSI); 340917825acSMarius Strobl CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 341917825acSMarius Strobl BUS_SPACE_BARRIER_READ | 342917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 3437c2fdcacSMarius Strobl DELAY(10000); 344917825acSMarius Strobl } 3458e5d93dbSMarius Strobl error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 3468e5d93dbSMarius Strobl cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 34736334c57SMarius Strobl MII_PHY_ANY, MII_OFFSET_ANY, MIIF_DOPAUSE); 348119051cbSMarius Strobl } 349119051cbSMarius Strobl /* 350119051cbSMarius Strobl * Fall back on an internal PHY if no external PHY was found. 351119051cbSMarius Strobl */ 352119051cbSMarius Strobl if (error != 0 && (v & CAS_MIF_CONF_MDI0) != 0) { 353119051cbSMarius Strobl v &= ~CAS_MIF_CONF_PHY_SELECT; 354119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MIF_CONF, v); 355917825acSMarius Strobl CAS_BARRIER(sc, CAS_MIF_CONF, 4, 356917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 357917825acSMarius Strobl /* Freeze the GMII pins of Saturn for saving power. */ 358917825acSMarius Strobl if (sc->sc_variant == CAS_SATURN) { 359917825acSMarius Strobl CAS_WRITE_4(sc, CAS_SATURN_PCFG, 3607c2fdcacSMarius Strobl CAS_READ_4(sc, CAS_SATURN_PCFG) | 361917825acSMarius Strobl CAS_SATURN_PCFG_FSI); 362917825acSMarius Strobl CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 363917825acSMarius Strobl BUS_SPACE_BARRIER_READ | 364917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 3657c2fdcacSMarius Strobl DELAY(10000); 366917825acSMarius Strobl } 3678e5d93dbSMarius Strobl error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 3688e5d93dbSMarius Strobl cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 36936334c57SMarius Strobl MII_PHY_ANY, MII_OFFSET_ANY, MIIF_DOPAUSE); 370119051cbSMarius Strobl } 371917825acSMarius Strobl } else { 372119051cbSMarius Strobl /* 373917825acSMarius Strobl * Use the external PCS SERDES. 374119051cbSMarius Strobl */ 375119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_DATAPATH, CAS_PCS_DATAPATH_SERDES); 376917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_DATAPATH, 4, BUS_SPACE_BARRIER_WRITE); 377917825acSMarius Strobl /* Enable/unfreeze the SERDES pins of Saturn. */ 378917825acSMarius Strobl if (sc->sc_variant == CAS_SATURN) { 379917825acSMarius Strobl CAS_WRITE_4(sc, CAS_SATURN_PCFG, 0); 380917825acSMarius Strobl CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 381917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 382917825acSMarius Strobl } 383119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_SERDES_CTRL, CAS_PCS_SERDES_CTRL_ESD); 384917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_SERDES_CTRL, 4, 385917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 386917825acSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_CONF, CAS_PCS_CONF_EN); 387917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_CONF, 4, 388917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 3898e5d93dbSMarius Strobl error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 3908e5d93dbSMarius Strobl cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 39136334c57SMarius Strobl CAS_PHYAD_EXTERNAL, MII_OFFSET_ANY, MIIF_DOPAUSE); 392119051cbSMarius Strobl } 393119051cbSMarius Strobl if (error != 0) { 3948e5d93dbSMarius Strobl device_printf(sc->sc_dev, "attaching PHYs failed\n"); 395119051cbSMarius Strobl goto fail_rxmap; 396119051cbSMarius Strobl } 397119051cbSMarius Strobl sc->sc_mii = device_get_softc(sc->sc_miibus); 398119051cbSMarius Strobl 399119051cbSMarius Strobl /* 400119051cbSMarius Strobl * From this point forward, the attachment cannot fail. A failure 401119051cbSMarius Strobl * before this point releases all resources that may have been 402119051cbSMarius Strobl * allocated. 403119051cbSMarius Strobl */ 404119051cbSMarius Strobl 405119051cbSMarius Strobl /* Announce FIFO sizes. */ 406119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_TX_FIFO_SIZE); 407119051cbSMarius Strobl device_printf(sc->sc_dev, "%ukB RX FIFO, %ukB TX FIFO\n", 408119051cbSMarius Strobl CAS_RX_FIFO_SIZE / 1024, v / 16); 409119051cbSMarius Strobl 410119051cbSMarius Strobl /* Attach the interface. */ 411119051cbSMarius Strobl ether_ifattach(ifp, sc->sc_enaddr); 412119051cbSMarius Strobl 413119051cbSMarius Strobl /* 414119051cbSMarius Strobl * Tell the upper layer(s) we support long frames/checksum offloads. 415119051cbSMarius Strobl */ 41644e65355SJustin Hibbits if_setifheaderlen(ifp, sizeof(struct ether_vlan_header)); 41744e65355SJustin Hibbits if_setcapabilities(ifp, IFCAP_VLAN_MTU); 418119051cbSMarius Strobl if ((sc->sc_flags & CAS_NO_CSUM) == 0) { 41944e65355SJustin Hibbits if_setcapabilitiesbit(ifp, IFCAP_HWCSUM, 0); 42044e65355SJustin Hibbits if_sethwassist(ifp, CAS_CSUM_FEATURES); 421119051cbSMarius Strobl } 42244e65355SJustin Hibbits if_setcapenable(ifp, if_getcapabilities(ifp)); 423119051cbSMarius Strobl 424119051cbSMarius Strobl return (0); 425119051cbSMarius Strobl 426119051cbSMarius Strobl /* 427119051cbSMarius Strobl * Free any resources we've allocated during the failed attach 428119051cbSMarius Strobl * attempt. Do this in reverse order and fall through. 429119051cbSMarius Strobl */ 430119051cbSMarius Strobl fail_rxmap: 431119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 432119051cbSMarius Strobl if (sc->sc_rxdsoft[i].rxds_paddr != 0) 433119051cbSMarius Strobl bus_dmamap_unload(sc->sc_rdmatag, 434119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap); 435119051cbSMarius Strobl fail_rxmem: 436119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 437119051cbSMarius Strobl if (sc->sc_rxdsoft[i].rxds_buf != NULL) 438119051cbSMarius Strobl bus_dmamem_free(sc->sc_rdmatag, 439119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_buf, 440119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap); 441119051cbSMarius Strobl fail_txd: 442119051cbSMarius Strobl for (i = 0; i < CAS_TXQUEUELEN; i++) 443119051cbSMarius Strobl if (sc->sc_txsoft[i].txs_dmamap != NULL) 444119051cbSMarius Strobl bus_dmamap_destroy(sc->sc_tdmatag, 445119051cbSMarius Strobl sc->sc_txsoft[i].txs_dmamap); 446119051cbSMarius Strobl bus_dmamap_unload(sc->sc_cdmatag, sc->sc_cddmamap); 447119051cbSMarius Strobl fail_cmem: 448119051cbSMarius Strobl bus_dmamem_free(sc->sc_cdmatag, sc->sc_control_data, 449119051cbSMarius Strobl sc->sc_cddmamap); 450119051cbSMarius Strobl fail_ctag: 451119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_cdmatag); 452119051cbSMarius Strobl fail_ttag: 453119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_tdmatag); 454119051cbSMarius Strobl fail_rtag: 455119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_rdmatag); 456119051cbSMarius Strobl fail_ptag: 457119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_pdmatag); 45873591c46SMarius Strobl fail_taskq: 45973591c46SMarius Strobl taskqueue_free(sc->sc_tq); 460119051cbSMarius Strobl if_free(ifp); 461119051cbSMarius Strobl return (error); 462119051cbSMarius Strobl } 463119051cbSMarius Strobl 464119051cbSMarius Strobl static void 465119051cbSMarius Strobl cas_detach(struct cas_softc *sc) 466119051cbSMarius Strobl { 46744e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 468119051cbSMarius Strobl int i; 469119051cbSMarius Strobl 47073591c46SMarius Strobl ether_ifdetach(ifp); 471119051cbSMarius Strobl CAS_LOCK(sc); 472119051cbSMarius Strobl cas_stop(ifp); 473119051cbSMarius Strobl CAS_UNLOCK(sc); 474119051cbSMarius Strobl callout_drain(&sc->sc_tick_ch); 475119051cbSMarius Strobl callout_drain(&sc->sc_rx_ch); 47673591c46SMarius Strobl taskqueue_drain(sc->sc_tq, &sc->sc_intr_task); 47773591c46SMarius Strobl taskqueue_drain(sc->sc_tq, &sc->sc_tx_task); 478119051cbSMarius Strobl if_free(ifp); 47973591c46SMarius Strobl taskqueue_free(sc->sc_tq); 480*11a91178SJohn Baldwin bus_generic_detach(sc->sc_dev); 481119051cbSMarius Strobl 482119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 483119051cbSMarius Strobl if (sc->sc_rxdsoft[i].rxds_dmamap != NULL) 484119051cbSMarius Strobl bus_dmamap_sync(sc->sc_rdmatag, 485119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap, 486119051cbSMarius Strobl BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 487119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 488119051cbSMarius Strobl if (sc->sc_rxdsoft[i].rxds_paddr != 0) 489119051cbSMarius Strobl bus_dmamap_unload(sc->sc_rdmatag, 490119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap); 491119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 492119051cbSMarius Strobl if (sc->sc_rxdsoft[i].rxds_buf != NULL) 493119051cbSMarius Strobl bus_dmamem_free(sc->sc_rdmatag, 494119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_buf, 495119051cbSMarius Strobl sc->sc_rxdsoft[i].rxds_dmamap); 496119051cbSMarius Strobl for (i = 0; i < CAS_TXQUEUELEN; i++) 497119051cbSMarius Strobl if (sc->sc_txsoft[i].txs_dmamap != NULL) 498119051cbSMarius Strobl bus_dmamap_destroy(sc->sc_tdmatag, 499119051cbSMarius Strobl sc->sc_txsoft[i].txs_dmamap); 500119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 501119051cbSMarius Strobl bus_dmamap_unload(sc->sc_cdmatag, sc->sc_cddmamap); 502119051cbSMarius Strobl bus_dmamem_free(sc->sc_cdmatag, sc->sc_control_data, 503119051cbSMarius Strobl sc->sc_cddmamap); 504119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_cdmatag); 505119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_tdmatag); 506119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_rdmatag); 507119051cbSMarius Strobl bus_dma_tag_destroy(sc->sc_pdmatag); 508119051cbSMarius Strobl } 509119051cbSMarius Strobl 510119051cbSMarius Strobl static void 511119051cbSMarius Strobl cas_suspend(struct cas_softc *sc) 512119051cbSMarius Strobl { 51344e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 514119051cbSMarius Strobl 515119051cbSMarius Strobl CAS_LOCK(sc); 516119051cbSMarius Strobl cas_stop(ifp); 517119051cbSMarius Strobl CAS_UNLOCK(sc); 518119051cbSMarius Strobl } 519119051cbSMarius Strobl 520119051cbSMarius Strobl static void 521119051cbSMarius Strobl cas_resume(struct cas_softc *sc) 522119051cbSMarius Strobl { 52344e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 524119051cbSMarius Strobl 525119051cbSMarius Strobl CAS_LOCK(sc); 526119051cbSMarius Strobl /* 527119051cbSMarius Strobl * On resume all registers have to be initialized again like 528119051cbSMarius Strobl * after power-on. 529119051cbSMarius Strobl */ 530119051cbSMarius Strobl sc->sc_flags &= ~CAS_INITED; 53144e65355SJustin Hibbits if (if_getflags(ifp) & IFF_UP) 532119051cbSMarius Strobl cas_init_locked(sc); 533119051cbSMarius Strobl CAS_UNLOCK(sc); 534119051cbSMarius Strobl } 535119051cbSMarius Strobl 536119051cbSMarius Strobl static inline void 537119051cbSMarius Strobl cas_rxcksum(struct mbuf *m, uint16_t cksum) 538119051cbSMarius Strobl { 539119051cbSMarius Strobl struct ether_header *eh; 540119051cbSMarius Strobl struct ip *ip; 541119051cbSMarius Strobl struct udphdr *uh; 542119051cbSMarius Strobl uint16_t *opts; 543119051cbSMarius Strobl int32_t hlen, len, pktlen; 544119051cbSMarius Strobl uint32_t temp32; 545119051cbSMarius Strobl 546119051cbSMarius Strobl pktlen = m->m_pkthdr.len; 547119051cbSMarius Strobl if (pktlen < sizeof(struct ether_header) + sizeof(struct ip)) 548119051cbSMarius Strobl return; 549119051cbSMarius Strobl eh = mtod(m, struct ether_header *); 550119051cbSMarius Strobl if (eh->ether_type != htons(ETHERTYPE_IP)) 551119051cbSMarius Strobl return; 552119051cbSMarius Strobl ip = (struct ip *)(eh + 1); 553119051cbSMarius Strobl if (ip->ip_v != IPVERSION) 554119051cbSMarius Strobl return; 555119051cbSMarius Strobl 556119051cbSMarius Strobl hlen = ip->ip_hl << 2; 557119051cbSMarius Strobl pktlen -= sizeof(struct ether_header); 558119051cbSMarius Strobl if (hlen < sizeof(struct ip)) 559119051cbSMarius Strobl return; 560119051cbSMarius Strobl if (ntohs(ip->ip_len) < hlen) 561119051cbSMarius Strobl return; 562119051cbSMarius Strobl if (ntohs(ip->ip_len) != pktlen) 563119051cbSMarius Strobl return; 564119051cbSMarius Strobl if (ip->ip_off & htons(IP_MF | IP_OFFMASK)) 565119051cbSMarius Strobl return; /* Cannot handle fragmented packet. */ 566119051cbSMarius Strobl 567119051cbSMarius Strobl switch (ip->ip_p) { 568119051cbSMarius Strobl case IPPROTO_TCP: 569119051cbSMarius Strobl if (pktlen < (hlen + sizeof(struct tcphdr))) 570119051cbSMarius Strobl return; 571119051cbSMarius Strobl break; 572119051cbSMarius Strobl case IPPROTO_UDP: 573119051cbSMarius Strobl if (pktlen < (hlen + sizeof(struct udphdr))) 574119051cbSMarius Strobl return; 575119051cbSMarius Strobl uh = (struct udphdr *)((uint8_t *)ip + hlen); 576119051cbSMarius Strobl if (uh->uh_sum == 0) 577119051cbSMarius Strobl return; /* no checksum */ 578119051cbSMarius Strobl break; 579119051cbSMarius Strobl default: 580119051cbSMarius Strobl return; 581119051cbSMarius Strobl } 582119051cbSMarius Strobl 583119051cbSMarius Strobl cksum = ~cksum; 584119051cbSMarius Strobl /* checksum fixup for IP options */ 585119051cbSMarius Strobl len = hlen - sizeof(struct ip); 586119051cbSMarius Strobl if (len > 0) { 587119051cbSMarius Strobl opts = (uint16_t *)(ip + 1); 588119051cbSMarius Strobl for (; len > 0; len -= sizeof(uint16_t), opts++) { 589119051cbSMarius Strobl temp32 = cksum - *opts; 590119051cbSMarius Strobl temp32 = (temp32 >> 16) + (temp32 & 65535); 591119051cbSMarius Strobl cksum = temp32 & 65535; 592119051cbSMarius Strobl } 593119051cbSMarius Strobl } 594119051cbSMarius Strobl m->m_pkthdr.csum_flags |= CSUM_DATA_VALID; 595119051cbSMarius Strobl m->m_pkthdr.csum_data = cksum; 596119051cbSMarius Strobl } 597119051cbSMarius Strobl 598119051cbSMarius Strobl static void 599119051cbSMarius Strobl cas_cddma_callback(void *xsc, bus_dma_segment_t *segs, int nsegs, int error) 600119051cbSMarius Strobl { 601119051cbSMarius Strobl struct cas_softc *sc = xsc; 602119051cbSMarius Strobl 603119051cbSMarius Strobl if (error != 0) 604119051cbSMarius Strobl return; 605119051cbSMarius Strobl if (nsegs != 1) 606119051cbSMarius Strobl panic("%s: bad control buffer segment count", __func__); 607119051cbSMarius Strobl sc->sc_cddma = segs[0].ds_addr; 608119051cbSMarius Strobl } 609119051cbSMarius Strobl 610119051cbSMarius Strobl static void 611119051cbSMarius Strobl cas_rxdma_callback(void *xsc, bus_dma_segment_t *segs, int nsegs, int error) 612119051cbSMarius Strobl { 613119051cbSMarius Strobl struct cas_softc *sc = xsc; 614119051cbSMarius Strobl 615119051cbSMarius Strobl if (error != 0) 616119051cbSMarius Strobl return; 617119051cbSMarius Strobl if (nsegs != 1) 618119051cbSMarius Strobl panic("%s: bad RX buffer segment count", __func__); 619119051cbSMarius Strobl sc->sc_rxdsoft[sc->sc_rxdptr].rxds_paddr = segs[0].ds_addr; 620119051cbSMarius Strobl } 621119051cbSMarius Strobl 622119051cbSMarius Strobl static void 623119051cbSMarius Strobl cas_tick(void *arg) 624119051cbSMarius Strobl { 625119051cbSMarius Strobl struct cas_softc *sc = arg; 62644e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 627119051cbSMarius Strobl uint32_t v; 628119051cbSMarius Strobl 629119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 630119051cbSMarius Strobl 631119051cbSMarius Strobl /* 632119051cbSMarius Strobl * Unload collision and error counters. 633119051cbSMarius Strobl */ 634884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_COLLISIONS, 635119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_NORM_COLL_CNT) + 636884c5b45SGleb Smirnoff CAS_READ_4(sc, CAS_MAC_FIRST_COLL_CNT)); 637119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MAC_EXCESS_COLL_CNT) + 638119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_LATE_COLL_CNT); 639884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_COLLISIONS, v); 640884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_OERRORS, v); 641884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IERRORS, 642119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_LEN_ERR_CNT) + 643119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_ALIGN_ERR) + 644119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_CRC_ERR_CNT) + 645884c5b45SGleb Smirnoff CAS_READ_4(sc, CAS_MAC_RX_CODE_VIOL)); 646119051cbSMarius Strobl 647119051cbSMarius Strobl /* 648119051cbSMarius Strobl * Then clear the hardware counters. 649119051cbSMarius Strobl */ 650119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_NORM_COLL_CNT, 0); 651119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_FIRST_COLL_CNT, 0); 652119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_EXCESS_COLL_CNT, 0); 653119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_LATE_COLL_CNT, 0); 654119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_LEN_ERR_CNT, 0); 655119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_ALIGN_ERR, 0); 656119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CRC_ERR_CNT, 0); 657119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CODE_VIOL, 0); 658119051cbSMarius Strobl 659119051cbSMarius Strobl mii_tick(sc->sc_mii); 660119051cbSMarius Strobl 66173591c46SMarius Strobl if (sc->sc_txfree != CAS_MAXTXFREE) 66273591c46SMarius Strobl cas_tint(sc); 66373591c46SMarius Strobl 66473591c46SMarius Strobl cas_watchdog(sc); 665119051cbSMarius Strobl 666119051cbSMarius Strobl callout_reset(&sc->sc_tick_ch, hz, cas_tick, sc); 667119051cbSMarius Strobl } 668119051cbSMarius Strobl 669119051cbSMarius Strobl static int 670119051cbSMarius Strobl cas_bitwait(struct cas_softc *sc, bus_addr_t r, uint32_t clr, uint32_t set) 671119051cbSMarius Strobl { 672119051cbSMarius Strobl int i; 673119051cbSMarius Strobl uint32_t reg; 674119051cbSMarius Strobl 675119051cbSMarius Strobl for (i = CAS_TRIES; i--; DELAY(100)) { 676119051cbSMarius Strobl reg = CAS_READ_4(sc, r); 677119051cbSMarius Strobl if ((reg & clr) == 0 && (reg & set) == set) 678119051cbSMarius Strobl return (1); 679119051cbSMarius Strobl } 680119051cbSMarius Strobl return (0); 681119051cbSMarius Strobl } 682119051cbSMarius Strobl 683119051cbSMarius Strobl static void 684119051cbSMarius Strobl cas_reset(struct cas_softc *sc) 685119051cbSMarius Strobl { 686119051cbSMarius Strobl 687119051cbSMarius Strobl #ifdef CAS_DEBUG 688119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 689119051cbSMarius Strobl #endif 690119051cbSMarius Strobl /* Disable all interrupts in order to avoid spurious ones. */ 691119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 692119051cbSMarius Strobl 693119051cbSMarius Strobl cas_reset_rx(sc); 694119051cbSMarius Strobl cas_reset_tx(sc); 695119051cbSMarius Strobl 696119051cbSMarius Strobl /* 697119051cbSMarius Strobl * Do a full reset modulo the result of the last auto-negotiation 698119051cbSMarius Strobl * when using the SERDES. 699119051cbSMarius Strobl */ 700119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_RX | CAS_RESET_TX | 701119051cbSMarius Strobl ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 702119051cbSMarius Strobl CAS_BARRIER(sc, CAS_RESET, 4, 703119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 704119051cbSMarius Strobl DELAY(3000); 705119051cbSMarius Strobl if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_RX | CAS_RESET_TX, 0)) 706119051cbSMarius Strobl device_printf(sc->sc_dev, "cannot reset device\n"); 707119051cbSMarius Strobl } 708119051cbSMarius Strobl 709119051cbSMarius Strobl static void 71044e65355SJustin Hibbits cas_stop(if_t ifp) 711119051cbSMarius Strobl { 71244e65355SJustin Hibbits struct cas_softc *sc = if_getsoftc(ifp); 713119051cbSMarius Strobl struct cas_txsoft *txs; 714119051cbSMarius Strobl 715119051cbSMarius Strobl #ifdef CAS_DEBUG 716119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 717119051cbSMarius Strobl #endif 718119051cbSMarius Strobl 719119051cbSMarius Strobl callout_stop(&sc->sc_tick_ch); 720119051cbSMarius Strobl callout_stop(&sc->sc_rx_ch); 721119051cbSMarius Strobl 722119051cbSMarius Strobl /* Disable all interrupts in order to avoid spurious ones. */ 723119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 724119051cbSMarius Strobl 725119051cbSMarius Strobl cas_reset_tx(sc); 726119051cbSMarius Strobl cas_reset_rx(sc); 727119051cbSMarius Strobl 728119051cbSMarius Strobl /* 729119051cbSMarius Strobl * Release any queued transmit buffers. 730119051cbSMarius Strobl */ 731119051cbSMarius Strobl while ((txs = STAILQ_FIRST(&sc->sc_txdirtyq)) != NULL) { 732119051cbSMarius Strobl STAILQ_REMOVE_HEAD(&sc->sc_txdirtyq, txs_q); 733119051cbSMarius Strobl if (txs->txs_ndescs != 0) { 734119051cbSMarius Strobl bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 735119051cbSMarius Strobl BUS_DMASYNC_POSTWRITE); 736119051cbSMarius Strobl bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 737119051cbSMarius Strobl if (txs->txs_mbuf != NULL) { 738119051cbSMarius Strobl m_freem(txs->txs_mbuf); 739119051cbSMarius Strobl txs->txs_mbuf = NULL; 740119051cbSMarius Strobl } 741119051cbSMarius Strobl } 742119051cbSMarius Strobl STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 743119051cbSMarius Strobl } 744119051cbSMarius Strobl 745119051cbSMarius Strobl /* 746119051cbSMarius Strobl * Mark the interface down and cancel the watchdog timer. 747119051cbSMarius Strobl */ 74844e65355SJustin Hibbits if_setdrvflagbits(ifp, 0, (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)); 749119051cbSMarius Strobl sc->sc_flags &= ~CAS_LINK; 750119051cbSMarius Strobl sc->sc_wdog_timer = 0; 751119051cbSMarius Strobl } 752119051cbSMarius Strobl 753119051cbSMarius Strobl static int 754119051cbSMarius Strobl cas_reset_rx(struct cas_softc *sc) 755119051cbSMarius Strobl { 756119051cbSMarius Strobl 757119051cbSMarius Strobl /* 758119051cbSMarius Strobl * Resetting while DMA is in progress can cause a bus hang, so we 759119051cbSMarius Strobl * disable DMA first. 760119051cbSMarius Strobl */ 7615f8a9ae4SMarius Strobl (void)cas_disable_rx(sc); 762119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_CONF, 0); 763119051cbSMarius Strobl CAS_BARRIER(sc, CAS_RX_CONF, 4, 764119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 765119051cbSMarius Strobl if (!cas_bitwait(sc, CAS_RX_CONF, CAS_RX_CONF_RXDMA_EN, 0)) 766119051cbSMarius Strobl device_printf(sc->sc_dev, "cannot disable RX DMA\n"); 767119051cbSMarius Strobl 768119051cbSMarius Strobl /* Finally, reset the ERX. */ 769119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_RX | 770119051cbSMarius Strobl ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 771119051cbSMarius Strobl CAS_BARRIER(sc, CAS_RESET, 4, 772119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 7735f8a9ae4SMarius Strobl if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_RX, 0)) { 774119051cbSMarius Strobl device_printf(sc->sc_dev, "cannot reset receiver\n"); 775119051cbSMarius Strobl return (1); 776119051cbSMarius Strobl } 777119051cbSMarius Strobl return (0); 778119051cbSMarius Strobl } 779119051cbSMarius Strobl 780119051cbSMarius Strobl static int 781119051cbSMarius Strobl cas_reset_tx(struct cas_softc *sc) 782119051cbSMarius Strobl { 783119051cbSMarius Strobl 784119051cbSMarius Strobl /* 785119051cbSMarius Strobl * Resetting while DMA is in progress can cause a bus hang, so we 786119051cbSMarius Strobl * disable DMA first. 787119051cbSMarius Strobl */ 7885f8a9ae4SMarius Strobl (void)cas_disable_tx(sc); 789119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_TX_CONF, 0); 790119051cbSMarius Strobl CAS_BARRIER(sc, CAS_TX_CONF, 4, 791119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 792119051cbSMarius Strobl if (!cas_bitwait(sc, CAS_TX_CONF, CAS_TX_CONF_TXDMA_EN, 0)) 793119051cbSMarius Strobl device_printf(sc->sc_dev, "cannot disable TX DMA\n"); 794119051cbSMarius Strobl 795119051cbSMarius Strobl /* Finally, reset the ETX. */ 796119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_TX | 797119051cbSMarius Strobl ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 798119051cbSMarius Strobl CAS_BARRIER(sc, CAS_RESET, 4, 799119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 8005f8a9ae4SMarius Strobl if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_TX, 0)) { 801119051cbSMarius Strobl device_printf(sc->sc_dev, "cannot reset transmitter\n"); 802119051cbSMarius Strobl return (1); 803119051cbSMarius Strobl } 804119051cbSMarius Strobl return (0); 805119051cbSMarius Strobl } 806119051cbSMarius Strobl 807119051cbSMarius Strobl static int 808119051cbSMarius Strobl cas_disable_rx(struct cas_softc *sc) 809119051cbSMarius Strobl { 810119051cbSMarius Strobl 811119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CONF, 812119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_CONF) & ~CAS_MAC_RX_CONF_EN); 813119051cbSMarius Strobl CAS_BARRIER(sc, CAS_MAC_RX_CONF, 4, 814119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 8155f8a9ae4SMarius Strobl if (cas_bitwait(sc, CAS_MAC_RX_CONF, CAS_MAC_RX_CONF_EN, 0)) 8165f8a9ae4SMarius Strobl return (1); 817a587c657SMarius Strobl if (bootverbose) 8185f8a9ae4SMarius Strobl device_printf(sc->sc_dev, "cannot disable RX MAC\n"); 8195f8a9ae4SMarius Strobl return (0); 820119051cbSMarius Strobl } 821119051cbSMarius Strobl 822119051cbSMarius Strobl static int 823119051cbSMarius Strobl cas_disable_tx(struct cas_softc *sc) 824119051cbSMarius Strobl { 825119051cbSMarius Strobl 826119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_TX_CONF, 827119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_TX_CONF) & ~CAS_MAC_TX_CONF_EN); 828119051cbSMarius Strobl CAS_BARRIER(sc, CAS_MAC_TX_CONF, 4, 829119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 8305f8a9ae4SMarius Strobl if (cas_bitwait(sc, CAS_MAC_TX_CONF, CAS_MAC_TX_CONF_EN, 0)) 8315f8a9ae4SMarius Strobl return (1); 832a587c657SMarius Strobl if (bootverbose) 8335f8a9ae4SMarius Strobl device_printf(sc->sc_dev, "cannot disable TX MAC\n"); 8345f8a9ae4SMarius Strobl return (0); 835119051cbSMarius Strobl } 836119051cbSMarius Strobl 837119051cbSMarius Strobl static inline void 838119051cbSMarius Strobl cas_rxcompinit(struct cas_rx_comp *rxcomp) 839119051cbSMarius Strobl { 840119051cbSMarius Strobl 841119051cbSMarius Strobl rxcomp->crc_word1 = 0; 842119051cbSMarius Strobl rxcomp->crc_word2 = 0; 843119051cbSMarius Strobl rxcomp->crc_word3 = 844119051cbSMarius Strobl htole64(CAS_SET(ETHER_HDR_LEN + sizeof(struct ip), CAS_RC3_CSO)); 845119051cbSMarius Strobl rxcomp->crc_word4 = htole64(CAS_RC4_ZERO); 846119051cbSMarius Strobl } 847119051cbSMarius Strobl 848119051cbSMarius Strobl static void 849119051cbSMarius Strobl cas_meminit(struct cas_softc *sc) 850119051cbSMarius Strobl { 851119051cbSMarius Strobl int i; 852119051cbSMarius Strobl 853119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 854119051cbSMarius Strobl 855119051cbSMarius Strobl /* 856119051cbSMarius Strobl * Initialize the transmit descriptor ring. 857119051cbSMarius Strobl */ 858119051cbSMarius Strobl for (i = 0; i < CAS_NTXDESC; i++) { 859119051cbSMarius Strobl sc->sc_txdescs[i].cd_flags = 0; 860119051cbSMarius Strobl sc->sc_txdescs[i].cd_buf_ptr = 0; 861119051cbSMarius Strobl } 862119051cbSMarius Strobl sc->sc_txfree = CAS_MAXTXFREE; 863119051cbSMarius Strobl sc->sc_txnext = 0; 864119051cbSMarius Strobl sc->sc_txwin = 0; 865119051cbSMarius Strobl 866119051cbSMarius Strobl /* 867119051cbSMarius Strobl * Initialize the receive completion ring. 868119051cbSMarius Strobl */ 869119051cbSMarius Strobl for (i = 0; i < CAS_NRXCOMP; i++) 870119051cbSMarius Strobl cas_rxcompinit(&sc->sc_rxcomps[i]); 871119051cbSMarius Strobl sc->sc_rxcptr = 0; 872119051cbSMarius Strobl 873119051cbSMarius Strobl /* 874119051cbSMarius Strobl * Initialize the first receive descriptor ring. We leave 875119051cbSMarius Strobl * the second one zeroed as we don't actually use it. 876119051cbSMarius Strobl */ 877119051cbSMarius Strobl for (i = 0; i < CAS_NRXDESC; i++) 878119051cbSMarius Strobl CAS_INIT_RXDESC(sc, i, i); 879119051cbSMarius Strobl sc->sc_rxdptr = 0; 880119051cbSMarius Strobl 881119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 882119051cbSMarius Strobl } 883119051cbSMarius Strobl 884119051cbSMarius Strobl static u_int 885119051cbSMarius Strobl cas_descsize(u_int sz) 886119051cbSMarius Strobl { 887119051cbSMarius Strobl 888119051cbSMarius Strobl switch (sz) { 889119051cbSMarius Strobl case 32: 890119051cbSMarius Strobl return (CAS_DESC_32); 891119051cbSMarius Strobl case 64: 892119051cbSMarius Strobl return (CAS_DESC_64); 893119051cbSMarius Strobl case 128: 894119051cbSMarius Strobl return (CAS_DESC_128); 895119051cbSMarius Strobl case 256: 896119051cbSMarius Strobl return (CAS_DESC_256); 897119051cbSMarius Strobl case 512: 898119051cbSMarius Strobl return (CAS_DESC_512); 899119051cbSMarius Strobl case 1024: 900119051cbSMarius Strobl return (CAS_DESC_1K); 901119051cbSMarius Strobl case 2048: 902119051cbSMarius Strobl return (CAS_DESC_2K); 903119051cbSMarius Strobl case 4096: 904119051cbSMarius Strobl return (CAS_DESC_4K); 905119051cbSMarius Strobl case 8192: 906119051cbSMarius Strobl return (CAS_DESC_8K); 907119051cbSMarius Strobl default: 908119051cbSMarius Strobl printf("%s: invalid descriptor ring size %d\n", __func__, sz); 909119051cbSMarius Strobl return (CAS_DESC_32); 910119051cbSMarius Strobl } 911119051cbSMarius Strobl } 912119051cbSMarius Strobl 913119051cbSMarius Strobl static u_int 914119051cbSMarius Strobl cas_rxcompsize(u_int sz) 915119051cbSMarius Strobl { 916119051cbSMarius Strobl 917119051cbSMarius Strobl switch (sz) { 918119051cbSMarius Strobl case 128: 919119051cbSMarius Strobl return (CAS_RX_CONF_COMP_128); 920119051cbSMarius Strobl case 256: 921119051cbSMarius Strobl return (CAS_RX_CONF_COMP_256); 922119051cbSMarius Strobl case 512: 923119051cbSMarius Strobl return (CAS_RX_CONF_COMP_512); 924119051cbSMarius Strobl case 1024: 925119051cbSMarius Strobl return (CAS_RX_CONF_COMP_1K); 926119051cbSMarius Strobl case 2048: 927119051cbSMarius Strobl return (CAS_RX_CONF_COMP_2K); 928119051cbSMarius Strobl case 4096: 929119051cbSMarius Strobl return (CAS_RX_CONF_COMP_4K); 930119051cbSMarius Strobl case 8192: 931119051cbSMarius Strobl return (CAS_RX_CONF_COMP_8K); 932119051cbSMarius Strobl case 16384: 933119051cbSMarius Strobl return (CAS_RX_CONF_COMP_16K); 934119051cbSMarius Strobl case 32768: 935119051cbSMarius Strobl return (CAS_RX_CONF_COMP_32K); 936119051cbSMarius Strobl default: 937119051cbSMarius Strobl printf("%s: invalid dcompletion ring size %d\n", __func__, sz); 938119051cbSMarius Strobl return (CAS_RX_CONF_COMP_128); 939119051cbSMarius Strobl } 940119051cbSMarius Strobl } 941119051cbSMarius Strobl 942119051cbSMarius Strobl static void 943119051cbSMarius Strobl cas_init(void *xsc) 944119051cbSMarius Strobl { 945119051cbSMarius Strobl struct cas_softc *sc = xsc; 946119051cbSMarius Strobl 947119051cbSMarius Strobl CAS_LOCK(sc); 948119051cbSMarius Strobl cas_init_locked(sc); 949119051cbSMarius Strobl CAS_UNLOCK(sc); 950119051cbSMarius Strobl } 951119051cbSMarius Strobl 952119051cbSMarius Strobl /* 953119051cbSMarius Strobl * Initialization of interface; set up initialization block 954119051cbSMarius Strobl * and transmit/receive descriptor rings. 955119051cbSMarius Strobl */ 956119051cbSMarius Strobl static void 957119051cbSMarius Strobl cas_init_locked(struct cas_softc *sc) 958119051cbSMarius Strobl { 95944e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 960119051cbSMarius Strobl uint32_t v; 961119051cbSMarius Strobl 962119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 963119051cbSMarius Strobl 96444e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0) 96573591c46SMarius Strobl return; 96673591c46SMarius Strobl 967119051cbSMarius Strobl #ifdef CAS_DEBUG 968119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s: calling stop", device_get_name(sc->sc_dev), 969119051cbSMarius Strobl __func__); 970119051cbSMarius Strobl #endif 971119051cbSMarius Strobl /* 972119051cbSMarius Strobl * Initialization sequence. The numbered steps below correspond 973119051cbSMarius Strobl * to the sequence outlined in section 6.3.5.1 in the Ethernet 974119051cbSMarius Strobl * Channel Engine manual (part of the PCIO manual). 975119051cbSMarius Strobl * See also the STP2002-STQ document from Sun Microsystems. 976119051cbSMarius Strobl */ 977119051cbSMarius Strobl 978119051cbSMarius Strobl /* step 1 & 2. Reset the Ethernet Channel. */ 979119051cbSMarius Strobl cas_stop(ifp); 980119051cbSMarius Strobl cas_reset(sc); 981119051cbSMarius Strobl #ifdef CAS_DEBUG 982119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s: restarting", device_get_name(sc->sc_dev), 983119051cbSMarius Strobl __func__); 984119051cbSMarius Strobl #endif 985119051cbSMarius Strobl 986917825acSMarius Strobl if ((sc->sc_flags & CAS_SERDES) == 0) 987119051cbSMarius Strobl /* Re-initialize the MIF. */ 988119051cbSMarius Strobl cas_mifinit(sc); 989119051cbSMarius Strobl 990119051cbSMarius Strobl /* step 3. Setup data structures in host memory. */ 991119051cbSMarius Strobl cas_meminit(sc); 992119051cbSMarius Strobl 993119051cbSMarius Strobl /* step 4. TX MAC registers & counters */ 994119051cbSMarius Strobl cas_init_regs(sc); 995119051cbSMarius Strobl 996119051cbSMarius Strobl /* step 5. RX MAC registers & counters */ 997119051cbSMarius Strobl 998119051cbSMarius Strobl /* step 6 & 7. Program Ring Base Addresses. */ 999119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_TX_DESC3_BASE_HI, 1000119051cbSMarius Strobl (((uint64_t)CAS_CDTXDADDR(sc, 0)) >> 32)); 1001119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_TX_DESC3_BASE_LO, 1002119051cbSMarius Strobl CAS_CDTXDADDR(sc, 0) & 0xffffffff); 1003119051cbSMarius Strobl 1004119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_COMP_BASE_HI, 1005119051cbSMarius Strobl (((uint64_t)CAS_CDRXCADDR(sc, 0)) >> 32)); 1006119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_COMP_BASE_LO, 1007119051cbSMarius Strobl CAS_CDRXCADDR(sc, 0) & 0xffffffff); 1008119051cbSMarius Strobl 1009119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_DESC_BASE_HI, 1010119051cbSMarius Strobl (((uint64_t)CAS_CDRXDADDR(sc, 0)) >> 32)); 1011119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_DESC_BASE_LO, 1012119051cbSMarius Strobl CAS_CDRXDADDR(sc, 0) & 0xffffffff); 1013119051cbSMarius Strobl 1014119051cbSMarius Strobl if ((sc->sc_flags & CAS_REG_PLUS) != 0) { 1015119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_DESC2_BASE_HI, 1016119051cbSMarius Strobl (((uint64_t)CAS_CDRXD2ADDR(sc, 0)) >> 32)); 1017119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_DESC2_BASE_LO, 1018119051cbSMarius Strobl CAS_CDRXD2ADDR(sc, 0) & 0xffffffff); 1019119051cbSMarius Strobl } 1020119051cbSMarius Strobl 1021119051cbSMarius Strobl #ifdef CAS_DEBUG 1022119051cbSMarius Strobl CTR5(KTR_CAS, 1023119051cbSMarius Strobl "loading TXDR %lx, RXCR %lx, RXDR %lx, RXD2R %lx, cddma %lx", 1024119051cbSMarius Strobl CAS_CDTXDADDR(sc, 0), CAS_CDRXCADDR(sc, 0), CAS_CDRXDADDR(sc, 0), 1025119051cbSMarius Strobl CAS_CDRXD2ADDR(sc, 0), sc->sc_cddma); 1026119051cbSMarius Strobl #endif 1027119051cbSMarius Strobl 1028119051cbSMarius Strobl /* step 8. Global Configuration & Interrupt Masks */ 1029119051cbSMarius Strobl 1030119051cbSMarius Strobl /* Disable weighted round robin. */ 1031119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_CAW, CAS_CAW_RR_DIS); 1032119051cbSMarius Strobl 1033119051cbSMarius Strobl /* 1034119051cbSMarius Strobl * Enable infinite bursts for revisions without PCI issues if 103558aa35d4SWarner Losh * applicable. Doing so greatly improves the TX performance. 1036119051cbSMarius Strobl */ 1037119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_INF_BURST, 1038119051cbSMarius Strobl (sc->sc_flags & CAS_TABORT) == 0 ? CAS_INF_BURST_EN : 1039119051cbSMarius Strobl 0); 1040119051cbSMarius Strobl 1041119051cbSMarius Strobl /* Set up interrupts. */ 1042119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_INTMASK, 104373591c46SMarius Strobl ~(CAS_INTR_TX_INT_ME | CAS_INTR_TX_TAG_ERR | 1044119051cbSMarius Strobl CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_TAG_ERR | 1045119051cbSMarius Strobl CAS_INTR_RX_COMP_FULL | CAS_INTR_RX_BUF_AEMPTY | 1046119051cbSMarius Strobl CAS_INTR_RX_COMP_AFULL | CAS_INTR_RX_LEN_MMATCH | 1047119051cbSMarius Strobl CAS_INTR_PCI_ERROR_INT 1048119051cbSMarius Strobl #ifdef CAS_DEBUG 1049119051cbSMarius Strobl | CAS_INTR_PCS_INT | CAS_INTR_MIF 1050119051cbSMarius Strobl #endif 1051119051cbSMarius Strobl )); 105273591c46SMarius Strobl /* Don't clear top level interrupts when CAS_STATUS_ALIAS is read. */ 105373591c46SMarius Strobl CAS_WRITE_4(sc, CAS_CLEAR_ALIAS, 0); 1054119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_MASK, ~CAS_MAC_RX_OVERFLOW); 1055119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_TX_MASK, 1056119051cbSMarius Strobl ~(CAS_MAC_TX_UNDERRUN | CAS_MAC_TX_MAX_PKT_ERR)); 1057119051cbSMarius Strobl #ifdef CAS_DEBUG 1058119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_CTRL_MASK, 1059119051cbSMarius Strobl ~(CAS_MAC_CTRL_PAUSE_RCVD | CAS_MAC_CTRL_PAUSE | 1060119051cbSMarius Strobl CAS_MAC_CTRL_NON_PAUSE)); 1061119051cbSMarius Strobl #else 1062119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_CTRL_MASK, 1063119051cbSMarius Strobl CAS_MAC_CTRL_PAUSE_RCVD | CAS_MAC_CTRL_PAUSE | 1064119051cbSMarius Strobl CAS_MAC_CTRL_NON_PAUSE); 1065119051cbSMarius Strobl #endif 1066119051cbSMarius Strobl 1067119051cbSMarius Strobl /* Enable PCI error interrupts. */ 1068119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_ERROR_MASK, 1069119051cbSMarius Strobl ~(CAS_ERROR_DTRTO | CAS_ERROR_OTHER | CAS_ERROR_DMAW_ZERO | 1070119051cbSMarius Strobl CAS_ERROR_DMAR_ZERO | CAS_ERROR_RTRTO)); 1071119051cbSMarius Strobl 1072119051cbSMarius Strobl /* Enable PCI error interrupts in BIM configuration. */ 1073119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_BIM_CONF, 1074119051cbSMarius Strobl CAS_BIM_CONF_DPAR_EN | CAS_BIM_CONF_RMA_EN | CAS_BIM_CONF_RTA_EN); 1075119051cbSMarius Strobl 1076119051cbSMarius Strobl /* 1077119051cbSMarius Strobl * step 9. ETX Configuration: encode receive descriptor ring size, 1078119051cbSMarius Strobl * enable DMA and disable pre-interrupt writeback completion. 1079119051cbSMarius Strobl */ 1080119051cbSMarius Strobl v = cas_descsize(CAS_NTXDESC) << CAS_TX_CONF_DESC3_SHFT; 1081119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_TX_CONF, v | CAS_TX_CONF_TXDMA_EN | 1082119051cbSMarius Strobl CAS_TX_CONF_RDPP_DIS | CAS_TX_CONF_PICWB_DIS); 1083119051cbSMarius Strobl 1084119051cbSMarius Strobl /* step 10. ERX Configuration */ 1085119051cbSMarius Strobl 1086119051cbSMarius Strobl /* 1087119051cbSMarius Strobl * Encode receive completion and descriptor ring sizes, set the 1088119051cbSMarius Strobl * swivel offset. 1089119051cbSMarius Strobl */ 1090119051cbSMarius Strobl v = cas_rxcompsize(CAS_NRXCOMP) << CAS_RX_CONF_COMP_SHFT; 1091119051cbSMarius Strobl v |= cas_descsize(CAS_NRXDESC) << CAS_RX_CONF_DESC_SHFT; 1092119051cbSMarius Strobl if ((sc->sc_flags & CAS_REG_PLUS) != 0) 1093119051cbSMarius Strobl v |= cas_descsize(CAS_NRXDESC2) << CAS_RX_CONF_DESC2_SHFT; 1094119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_CONF, 1095119051cbSMarius Strobl v | (ETHER_ALIGN << CAS_RX_CONF_SOFF_SHFT)); 1096119051cbSMarius Strobl 1097119051cbSMarius Strobl /* Set the PAUSE thresholds. We use the maximum OFF threshold. */ 1098119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_PTHRS, 109936334c57SMarius Strobl (111 << CAS_RX_PTHRS_XOFF_SHFT) | (15 << CAS_RX_PTHRS_XON_SHFT)); 1100119051cbSMarius Strobl 1101119051cbSMarius Strobl /* RX blanking */ 1102119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_BLANK, 1103119051cbSMarius Strobl (15 << CAS_RX_BLANK_TIME_SHFT) | (5 << CAS_RX_BLANK_PKTS_SHFT)); 1104119051cbSMarius Strobl 1105119051cbSMarius Strobl /* Set RX_COMP_AFULL threshold to half of the RX completions. */ 1106119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_AEMPTY_THRS, 1107119051cbSMarius Strobl (CAS_NRXCOMP / 2) << CAS_RX_AEMPTY_COMP_SHFT); 1108119051cbSMarius Strobl 1109119051cbSMarius Strobl /* Initialize the RX page size register as appropriate for 8k. */ 1110119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_PSZ, 1111119051cbSMarius Strobl (CAS_RX_PSZ_8K << CAS_RX_PSZ_SHFT) | 1112119051cbSMarius Strobl (4 << CAS_RX_PSZ_MB_CNT_SHFT) | 1113119051cbSMarius Strobl (CAS_RX_PSZ_MB_STRD_2K << CAS_RX_PSZ_MB_STRD_SHFT) | 1114119051cbSMarius Strobl (CAS_RX_PSZ_MB_OFF_64 << CAS_RX_PSZ_MB_OFF_SHFT)); 1115119051cbSMarius Strobl 1116119051cbSMarius Strobl /* Disable RX random early detection. */ 1117119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_RED, 0); 1118119051cbSMarius Strobl 1119119051cbSMarius Strobl /* Zero the RX reassembly DMA table. */ 1120119051cbSMarius Strobl for (v = 0; v <= CAS_RX_REAS_DMA_ADDR_LC; v++) { 1121119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_REAS_DMA_ADDR, v); 1122119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_LO, 0); 1123119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_MD, 0); 1124119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_HI, 0); 1125119051cbSMarius Strobl } 1126119051cbSMarius Strobl 1127119051cbSMarius Strobl /* Ensure the RX control FIFO and RX IPP FIFO addresses are zero. */ 1128119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_CTRL_FIFO, 0); 1129119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_IPP_ADDR, 0); 1130119051cbSMarius Strobl 1131119051cbSMarius Strobl /* Finally, enable RX DMA. */ 1132119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_CONF, 1133119051cbSMarius Strobl CAS_READ_4(sc, CAS_RX_CONF) | CAS_RX_CONF_RXDMA_EN); 1134119051cbSMarius Strobl 1135119051cbSMarius Strobl /* step 11. Configure Media. */ 1136119051cbSMarius Strobl 1137119051cbSMarius Strobl /* step 12. RX_MAC Configuration Register */ 11385f8a9ae4SMarius Strobl v = CAS_READ_4(sc, CAS_MAC_RX_CONF); 11395f8a9ae4SMarius Strobl v &= ~(CAS_MAC_RX_CONF_STRPPAD | CAS_MAC_RX_CONF_EN); 11405f8a9ae4SMarius Strobl v |= CAS_MAC_RX_CONF_STRPFCS; 11415f8a9ae4SMarius Strobl sc->sc_mac_rxcfg = v; 11425f8a9ae4SMarius Strobl /* 11435f8a9ae4SMarius Strobl * Clear the RX filter and reprogram it. This will also set the 11445f8a9ae4SMarius Strobl * current RX MAC configuration and enable it. 11455f8a9ae4SMarius Strobl */ 11465f8a9ae4SMarius Strobl cas_setladrf(sc); 1147119051cbSMarius Strobl 1148119051cbSMarius Strobl /* step 13. TX_MAC Configuration Register */ 1149119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MAC_TX_CONF); 1150119051cbSMarius Strobl v |= CAS_MAC_TX_CONF_EN; 11515f8a9ae4SMarius Strobl (void)cas_disable_tx(sc); 1152119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_TX_CONF, v); 1153119051cbSMarius Strobl 1154119051cbSMarius Strobl /* step 14. Issue Transmit Pending command. */ 1155119051cbSMarius Strobl 1156a34f0477SMarius Strobl /* step 15. Give the receiver a swift kick. */ 1157119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_KICK, CAS_NRXDESC - 4); 1158119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_COMP_TAIL, 0); 1159119051cbSMarius Strobl if ((sc->sc_flags & CAS_REG_PLUS) != 0) 1160119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_KICK2, CAS_NRXDESC2 - 4); 1161119051cbSMarius Strobl 116244e65355SJustin Hibbits if_setdrvflagbits(ifp, IFF_DRV_RUNNING, 0); 116344e65355SJustin Hibbits if_setdrvflagbits(ifp, 0, IFF_DRV_OACTIVE); 1164119051cbSMarius Strobl 1165119051cbSMarius Strobl mii_mediachg(sc->sc_mii); 1166119051cbSMarius Strobl 1167119051cbSMarius Strobl /* Start the one second timer. */ 1168119051cbSMarius Strobl sc->sc_wdog_timer = 0; 1169119051cbSMarius Strobl callout_reset(&sc->sc_tick_ch, hz, cas_tick, sc); 1170119051cbSMarius Strobl } 1171119051cbSMarius Strobl 1172119051cbSMarius Strobl static int 1173119051cbSMarius Strobl cas_load_txmbuf(struct cas_softc *sc, struct mbuf **m_head) 1174119051cbSMarius Strobl { 1175119051cbSMarius Strobl bus_dma_segment_t txsegs[CAS_NTXSEGS]; 1176119051cbSMarius Strobl struct cas_txsoft *txs; 1177119051cbSMarius Strobl struct ip *ip; 1178119051cbSMarius Strobl struct mbuf *m; 1179119051cbSMarius Strobl uint64_t cflags; 1180119051cbSMarius Strobl int error, nexttx, nsegs, offset, seg; 1181119051cbSMarius Strobl 1182119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1183119051cbSMarius Strobl 1184119051cbSMarius Strobl /* Get a work queue entry. */ 1185119051cbSMarius Strobl if ((txs = STAILQ_FIRST(&sc->sc_txfreeq)) == NULL) { 1186119051cbSMarius Strobl /* Ran out of descriptors. */ 1187119051cbSMarius Strobl return (ENOBUFS); 1188119051cbSMarius Strobl } 1189119051cbSMarius Strobl 1190119051cbSMarius Strobl cflags = 0; 1191119051cbSMarius Strobl if (((*m_head)->m_pkthdr.csum_flags & CAS_CSUM_FEATURES) != 0) { 1192119051cbSMarius Strobl if (M_WRITABLE(*m_head) == 0) { 1193c6499eccSGleb Smirnoff m = m_dup(*m_head, M_NOWAIT); 1194119051cbSMarius Strobl m_freem(*m_head); 1195119051cbSMarius Strobl *m_head = m; 1196119051cbSMarius Strobl if (m == NULL) 1197119051cbSMarius Strobl return (ENOBUFS); 1198119051cbSMarius Strobl } 1199119051cbSMarius Strobl offset = sizeof(struct ether_header); 1200119051cbSMarius Strobl m = m_pullup(*m_head, offset + sizeof(struct ip)); 1201119051cbSMarius Strobl if (m == NULL) { 1202119051cbSMarius Strobl *m_head = NULL; 1203119051cbSMarius Strobl return (ENOBUFS); 1204119051cbSMarius Strobl } 1205119051cbSMarius Strobl ip = (struct ip *)(mtod(m, caddr_t) + offset); 1206119051cbSMarius Strobl offset += (ip->ip_hl << 2); 1207119051cbSMarius Strobl cflags = (offset << CAS_TD_CKSUM_START_SHFT) | 1208119051cbSMarius Strobl ((offset + m->m_pkthdr.csum_data) << 1209119051cbSMarius Strobl CAS_TD_CKSUM_STUFF_SHFT) | CAS_TD_CKSUM_EN; 1210119051cbSMarius Strobl *m_head = m; 1211119051cbSMarius Strobl } 1212119051cbSMarius Strobl 1213119051cbSMarius Strobl error = bus_dmamap_load_mbuf_sg(sc->sc_tdmatag, txs->txs_dmamap, 1214119051cbSMarius Strobl *m_head, txsegs, &nsegs, BUS_DMA_NOWAIT); 1215119051cbSMarius Strobl if (error == EFBIG) { 1216c6499eccSGleb Smirnoff m = m_collapse(*m_head, M_NOWAIT, CAS_NTXSEGS); 1217119051cbSMarius Strobl if (m == NULL) { 1218119051cbSMarius Strobl m_freem(*m_head); 1219119051cbSMarius Strobl *m_head = NULL; 1220119051cbSMarius Strobl return (ENOBUFS); 1221119051cbSMarius Strobl } 1222119051cbSMarius Strobl *m_head = m; 1223119051cbSMarius Strobl error = bus_dmamap_load_mbuf_sg(sc->sc_tdmatag, 1224119051cbSMarius Strobl txs->txs_dmamap, *m_head, txsegs, &nsegs, 1225119051cbSMarius Strobl BUS_DMA_NOWAIT); 1226119051cbSMarius Strobl if (error != 0) { 1227119051cbSMarius Strobl m_freem(*m_head); 1228119051cbSMarius Strobl *m_head = NULL; 1229119051cbSMarius Strobl return (error); 1230119051cbSMarius Strobl } 1231119051cbSMarius Strobl } else if (error != 0) 1232119051cbSMarius Strobl return (error); 1233119051cbSMarius Strobl /* If nsegs is wrong then the stack is corrupt. */ 1234119051cbSMarius Strobl KASSERT(nsegs <= CAS_NTXSEGS, 1235119051cbSMarius Strobl ("%s: too many DMA segments (%d)", __func__, nsegs)); 1236119051cbSMarius Strobl if (nsegs == 0) { 1237119051cbSMarius Strobl m_freem(*m_head); 1238119051cbSMarius Strobl *m_head = NULL; 1239119051cbSMarius Strobl return (EIO); 1240119051cbSMarius Strobl } 1241119051cbSMarius Strobl 1242119051cbSMarius Strobl /* 1243119051cbSMarius Strobl * Ensure we have enough descriptors free to describe 1244119051cbSMarius Strobl * the packet. Note, we always reserve one descriptor 1245119051cbSMarius Strobl * at the end of the ring as a termination point, in 1246119051cbSMarius Strobl * order to prevent wrap-around. 1247119051cbSMarius Strobl */ 1248119051cbSMarius Strobl if (nsegs > sc->sc_txfree - 1) { 1249119051cbSMarius Strobl txs->txs_ndescs = 0; 1250119051cbSMarius Strobl bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 1251119051cbSMarius Strobl return (ENOBUFS); 1252119051cbSMarius Strobl } 1253119051cbSMarius Strobl 1254119051cbSMarius Strobl txs->txs_ndescs = nsegs; 1255119051cbSMarius Strobl txs->txs_firstdesc = sc->sc_txnext; 1256119051cbSMarius Strobl nexttx = txs->txs_firstdesc; 1257119051cbSMarius Strobl for (seg = 0; seg < nsegs; seg++, nexttx = CAS_NEXTTX(nexttx)) { 1258119051cbSMarius Strobl #ifdef CAS_DEBUG 1259119051cbSMarius Strobl CTR6(KTR_CAS, 1260119051cbSMarius Strobl "%s: mapping seg %d (txd %d), len %lx, addr %#lx (%#lx)", 1261119051cbSMarius Strobl __func__, seg, nexttx, txsegs[seg].ds_len, 1262119051cbSMarius Strobl txsegs[seg].ds_addr, htole64(txsegs[seg].ds_addr)); 1263119051cbSMarius Strobl #endif 1264119051cbSMarius Strobl sc->sc_txdescs[nexttx].cd_buf_ptr = 1265119051cbSMarius Strobl htole64(txsegs[seg].ds_addr); 1266119051cbSMarius Strobl KASSERT(txsegs[seg].ds_len < 1267119051cbSMarius Strobl CAS_TD_BUF_LEN_MASK >> CAS_TD_BUF_LEN_SHFT, 1268119051cbSMarius Strobl ("%s: segment size too large!", __func__)); 1269119051cbSMarius Strobl sc->sc_txdescs[nexttx].cd_flags = 1270119051cbSMarius Strobl htole64(txsegs[seg].ds_len << CAS_TD_BUF_LEN_SHFT); 1271119051cbSMarius Strobl txs->txs_lastdesc = nexttx; 1272119051cbSMarius Strobl } 1273119051cbSMarius Strobl 1274119051cbSMarius Strobl /* Set EOF on the last descriptor. */ 1275119051cbSMarius Strobl #ifdef CAS_DEBUG 1276119051cbSMarius Strobl CTR3(KTR_CAS, "%s: end of frame at segment %d, TX %d", 1277119051cbSMarius Strobl __func__, seg, nexttx); 1278119051cbSMarius Strobl #endif 1279119051cbSMarius Strobl sc->sc_txdescs[txs->txs_lastdesc].cd_flags |= 1280119051cbSMarius Strobl htole64(CAS_TD_END_OF_FRAME); 1281119051cbSMarius Strobl 1282119051cbSMarius Strobl /* Lastly set SOF on the first descriptor. */ 1283119051cbSMarius Strobl #ifdef CAS_DEBUG 1284119051cbSMarius Strobl CTR3(KTR_CAS, "%s: start of frame at segment %d, TX %d", 1285119051cbSMarius Strobl __func__, seg, nexttx); 1286119051cbSMarius Strobl #endif 128773591c46SMarius Strobl if (sc->sc_txwin += nsegs > CAS_MAXTXFREE * 2 / 3) { 1288119051cbSMarius Strobl sc->sc_txwin = 0; 1289119051cbSMarius Strobl sc->sc_txdescs[txs->txs_firstdesc].cd_flags |= 1290119051cbSMarius Strobl htole64(cflags | CAS_TD_START_OF_FRAME | CAS_TD_INT_ME); 1291119051cbSMarius Strobl } else 1292119051cbSMarius Strobl sc->sc_txdescs[txs->txs_firstdesc].cd_flags |= 1293119051cbSMarius Strobl htole64(cflags | CAS_TD_START_OF_FRAME); 1294119051cbSMarius Strobl 1295119051cbSMarius Strobl /* Sync the DMA map. */ 1296119051cbSMarius Strobl bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 1297119051cbSMarius Strobl BUS_DMASYNC_PREWRITE); 1298119051cbSMarius Strobl 1299119051cbSMarius Strobl #ifdef CAS_DEBUG 1300119051cbSMarius Strobl CTR4(KTR_CAS, "%s: setting firstdesc=%d, lastdesc=%d, ndescs=%d", 1301119051cbSMarius Strobl __func__, txs->txs_firstdesc, txs->txs_lastdesc, 1302119051cbSMarius Strobl txs->txs_ndescs); 1303119051cbSMarius Strobl #endif 1304119051cbSMarius Strobl STAILQ_REMOVE_HEAD(&sc->sc_txfreeq, txs_q); 1305119051cbSMarius Strobl STAILQ_INSERT_TAIL(&sc->sc_txdirtyq, txs, txs_q); 1306119051cbSMarius Strobl txs->txs_mbuf = *m_head; 1307119051cbSMarius Strobl 1308119051cbSMarius Strobl sc->sc_txnext = CAS_NEXTTX(txs->txs_lastdesc); 1309119051cbSMarius Strobl sc->sc_txfree -= txs->txs_ndescs; 1310119051cbSMarius Strobl 1311119051cbSMarius Strobl return (0); 1312119051cbSMarius Strobl } 1313119051cbSMarius Strobl 1314119051cbSMarius Strobl static void 1315119051cbSMarius Strobl cas_init_regs(struct cas_softc *sc) 1316119051cbSMarius Strobl { 1317119051cbSMarius Strobl int i; 131844e65355SJustin Hibbits const u_char *laddr = if_getlladdr(sc->sc_ifp); 1319119051cbSMarius Strobl 1320119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1321119051cbSMarius Strobl 1322119051cbSMarius Strobl /* These registers are not cleared on reset. */ 1323119051cbSMarius Strobl if ((sc->sc_flags & CAS_INITED) == 0) { 1324119051cbSMarius Strobl /* magic values */ 1325119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_IPG0, 0); 1326119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_IPG1, 8); 1327119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_IPG2, 4); 1328119051cbSMarius Strobl 1329119051cbSMarius Strobl /* min frame length */ 1330119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_MIN_FRAME, ETHER_MIN_LEN); 1331119051cbSMarius Strobl /* max frame length and max burst size */ 1332119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_MAX_BF, 1333119051cbSMarius Strobl ((ETHER_MAX_LEN_JUMBO + ETHER_VLAN_ENCAP_LEN) << 1334119051cbSMarius Strobl CAS_MAC_MAX_BF_FRM_SHFT) | 1335119051cbSMarius Strobl (0x2000 << CAS_MAC_MAX_BF_BST_SHFT)); 1336119051cbSMarius Strobl 1337119051cbSMarius Strobl /* more magic values */ 1338119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x7); 1339119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_JAM_SIZE, 0x4); 1340119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ATTEMPT_LIMIT, 0x10); 134136334c57SMarius Strobl CAS_WRITE_4(sc, CAS_MAC_CTRL_TYPE, 0x8808); 1342119051cbSMarius Strobl 1343119051cbSMarius Strobl /* random number seed */ 1344119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RANDOM_SEED, 1345119051cbSMarius Strobl ((laddr[5] << 8) | laddr[4]) & 0x3ff); 1346119051cbSMarius Strobl 1347119051cbSMarius Strobl /* secondary MAC addresses: 0:0:0:0:0:0 */ 1348119051cbSMarius Strobl for (i = CAS_MAC_ADDR3; i <= CAS_MAC_ADDR41; 1349119051cbSMarius Strobl i += CAS_MAC_ADDR4 - CAS_MAC_ADDR3) 1350119051cbSMarius Strobl CAS_WRITE_4(sc, i, 0); 1351119051cbSMarius Strobl 1352119051cbSMarius Strobl /* MAC control address: 01:80:c2:00:00:01 */ 1353119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR42, 0x0001); 1354119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR43, 0xc200); 1355119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR44, 0x0180); 1356119051cbSMarius Strobl 1357119051cbSMarius Strobl /* MAC filter address: 0:0:0:0:0:0 */ 1358119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_AFILTER0, 0); 1359119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_AFILTER1, 0); 1360119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_AFILTER2, 0); 1361119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_AFILTER_MASK1_2, 0); 1362119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_AFILTER_MASK0, 0); 1363119051cbSMarius Strobl 1364119051cbSMarius Strobl /* Zero the hash table. */ 1365119051cbSMarius Strobl for (i = CAS_MAC_HASH0; i <= CAS_MAC_HASH15; 1366119051cbSMarius Strobl i += CAS_MAC_HASH1 - CAS_MAC_HASH0) 1367119051cbSMarius Strobl CAS_WRITE_4(sc, i, 0); 1368119051cbSMarius Strobl 1369119051cbSMarius Strobl sc->sc_flags |= CAS_INITED; 1370119051cbSMarius Strobl } 1371119051cbSMarius Strobl 1372119051cbSMarius Strobl /* Counters need to be zeroed. */ 1373119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_NORM_COLL_CNT, 0); 1374119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_FIRST_COLL_CNT, 0); 1375119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_EXCESS_COLL_CNT, 0); 1376119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_LATE_COLL_CNT, 0); 1377119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_DEFER_TMR_CNT, 0); 1378119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_PEAK_ATTEMPTS, 0); 1379119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_FRAME_COUNT, 0); 1380119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_LEN_ERR_CNT, 0); 1381119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_ALIGN_ERR, 0); 1382119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CRC_ERR_CNT, 0); 1383119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CODE_VIOL, 0); 1384119051cbSMarius Strobl 1385119051cbSMarius Strobl /* Set XOFF PAUSE time. */ 1386119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_SPC, 0x1BF0 << CAS_MAC_SPC_TIME_SHFT); 1387119051cbSMarius Strobl 1388119051cbSMarius Strobl /* Set the station address. */ 1389119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR0, (laddr[4] << 8) | laddr[5]); 1390119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR1, (laddr[2] << 8) | laddr[3]); 1391119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_ADDR2, (laddr[0] << 8) | laddr[1]); 1392119051cbSMarius Strobl 1393119051cbSMarius Strobl /* Enable MII outputs. */ 1394119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_XIF_CONF, CAS_MAC_XIF_CONF_TX_OE); 1395119051cbSMarius Strobl } 1396119051cbSMarius Strobl 1397119051cbSMarius Strobl static void 139873591c46SMarius Strobl cas_tx_task(void *arg, int pending __unused) 1399119051cbSMarius Strobl { 140044e65355SJustin Hibbits if_t ifp; 1401119051cbSMarius Strobl 140244e65355SJustin Hibbits ifp = (if_t)arg; 140373591c46SMarius Strobl cas_start(ifp); 1404119051cbSMarius Strobl } 1405119051cbSMarius Strobl 1406119051cbSMarius Strobl static inline void 1407119051cbSMarius Strobl cas_txkick(struct cas_softc *sc) 1408119051cbSMarius Strobl { 1409119051cbSMarius Strobl 1410119051cbSMarius Strobl /* 1411119051cbSMarius Strobl * Update the TX kick register. This register has to point to the 1412119051cbSMarius Strobl * descriptor after the last valid one and for optimum performance 1413119051cbSMarius Strobl * should be incremented in multiples of 4 (the DMA engine fetches/ 1414119051cbSMarius Strobl * updates descriptors in batches of 4). 1415119051cbSMarius Strobl */ 1416119051cbSMarius Strobl #ifdef CAS_DEBUG 1417119051cbSMarius Strobl CTR3(KTR_CAS, "%s: %s: kicking TX %d", 1418119051cbSMarius Strobl device_get_name(sc->sc_dev), __func__, sc->sc_txnext); 1419119051cbSMarius Strobl #endif 1420119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1421119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_TX_KICK3, sc->sc_txnext); 1422119051cbSMarius Strobl } 1423119051cbSMarius Strobl 1424119051cbSMarius Strobl static void 142544e65355SJustin Hibbits cas_start(if_t ifp) 1426119051cbSMarius Strobl { 142744e65355SJustin Hibbits struct cas_softc *sc = if_getsoftc(ifp); 1428119051cbSMarius Strobl struct mbuf *m; 1429119051cbSMarius Strobl int kicked, ntx; 1430119051cbSMarius Strobl 143173591c46SMarius Strobl CAS_LOCK(sc); 1432119051cbSMarius Strobl 143344e65355SJustin Hibbits if ((if_getdrvflags(ifp) & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) != 143473591c46SMarius Strobl IFF_DRV_RUNNING || (sc->sc_flags & CAS_LINK) == 0) { 143573591c46SMarius Strobl CAS_UNLOCK(sc); 1436119051cbSMarius Strobl return; 143773591c46SMarius Strobl } 143873591c46SMarius Strobl 143973591c46SMarius Strobl if (sc->sc_txfree < CAS_MAXTXFREE / 4) 144073591c46SMarius Strobl cas_tint(sc); 1441119051cbSMarius Strobl 1442119051cbSMarius Strobl #ifdef CAS_DEBUG 1443119051cbSMarius Strobl CTR4(KTR_CAS, "%s: %s: txfree %d, txnext %d", 1444119051cbSMarius Strobl device_get_name(sc->sc_dev), __func__, sc->sc_txfree, 1445119051cbSMarius Strobl sc->sc_txnext); 1446119051cbSMarius Strobl #endif 1447119051cbSMarius Strobl ntx = 0; 1448119051cbSMarius Strobl kicked = 0; 144944e65355SJustin Hibbits for (; !if_sendq_empty(ifp) && sc->sc_txfree > 1;) { 145044e65355SJustin Hibbits m = if_dequeue(ifp); 1451119051cbSMarius Strobl if (m == NULL) 1452119051cbSMarius Strobl break; 1453119051cbSMarius Strobl if (cas_load_txmbuf(sc, &m) != 0) { 1454119051cbSMarius Strobl if (m == NULL) 1455119051cbSMarius Strobl break; 145644e65355SJustin Hibbits if_setdrvflagbits(ifp, IFF_DRV_OACTIVE, 0); 145744e65355SJustin Hibbits if_sendq_prepend(ifp, m); 1458119051cbSMarius Strobl break; 1459119051cbSMarius Strobl } 1460119051cbSMarius Strobl if ((sc->sc_txnext % 4) == 0) { 1461119051cbSMarius Strobl cas_txkick(sc); 1462119051cbSMarius Strobl kicked = 1; 1463119051cbSMarius Strobl } else 1464119051cbSMarius Strobl kicked = 0; 1465119051cbSMarius Strobl ntx++; 1466119051cbSMarius Strobl BPF_MTAP(ifp, m); 1467119051cbSMarius Strobl } 1468119051cbSMarius Strobl 1469119051cbSMarius Strobl if (ntx > 0) { 1470119051cbSMarius Strobl if (kicked == 0) 1471119051cbSMarius Strobl cas_txkick(sc); 1472119051cbSMarius Strobl #ifdef CAS_DEBUG 1473119051cbSMarius Strobl CTR2(KTR_CAS, "%s: packets enqueued, OWN on %d", 1474119051cbSMarius Strobl device_get_name(sc->sc_dev), sc->sc_txnext); 1475119051cbSMarius Strobl #endif 1476119051cbSMarius Strobl 1477119051cbSMarius Strobl /* Set a watchdog timer in case the chip flakes out. */ 1478119051cbSMarius Strobl sc->sc_wdog_timer = 5; 1479119051cbSMarius Strobl #ifdef CAS_DEBUG 1480119051cbSMarius Strobl CTR3(KTR_CAS, "%s: %s: watchdog %d", 1481119051cbSMarius Strobl device_get_name(sc->sc_dev), __func__, 1482119051cbSMarius Strobl sc->sc_wdog_timer); 1483119051cbSMarius Strobl #endif 1484119051cbSMarius Strobl } 148573591c46SMarius Strobl 148673591c46SMarius Strobl CAS_UNLOCK(sc); 1487119051cbSMarius Strobl } 1488119051cbSMarius Strobl 1489119051cbSMarius Strobl static void 1490119051cbSMarius Strobl cas_tint(struct cas_softc *sc) 1491119051cbSMarius Strobl { 149244e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 1493119051cbSMarius Strobl struct cas_txsoft *txs; 1494119051cbSMarius Strobl int progress; 1495119051cbSMarius Strobl uint32_t txlast; 1496119051cbSMarius Strobl #ifdef CAS_DEBUG 1497119051cbSMarius Strobl int i; 1498119051cbSMarius Strobl 1499119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1500119051cbSMarius Strobl 1501119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 1502119051cbSMarius Strobl #endif 1503119051cbSMarius Strobl 1504119051cbSMarius Strobl /* 1505119051cbSMarius Strobl * Go through our TX list and free mbufs for those 1506119051cbSMarius Strobl * frames that have been transmitted. 1507119051cbSMarius Strobl */ 1508119051cbSMarius Strobl progress = 0; 1509119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD); 1510119051cbSMarius Strobl while ((txs = STAILQ_FIRST(&sc->sc_txdirtyq)) != NULL) { 1511119051cbSMarius Strobl #ifdef CAS_DEBUG 151244e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_DEBUG) != 0) { 1513119051cbSMarius Strobl printf(" txsoft %p transmit chain:\n", txs); 1514119051cbSMarius Strobl for (i = txs->txs_firstdesc;; i = CAS_NEXTTX(i)) { 1515119051cbSMarius Strobl printf("descriptor %d: ", i); 1516119051cbSMarius Strobl printf("cd_flags: 0x%016llx\t", 1517119051cbSMarius Strobl (long long)le64toh( 1518119051cbSMarius Strobl sc->sc_txdescs[i].cd_flags)); 1519119051cbSMarius Strobl printf("cd_buf_ptr: 0x%016llx\n", 1520119051cbSMarius Strobl (long long)le64toh( 1521119051cbSMarius Strobl sc->sc_txdescs[i].cd_buf_ptr)); 1522119051cbSMarius Strobl if (i == txs->txs_lastdesc) 1523119051cbSMarius Strobl break; 1524119051cbSMarius Strobl } 1525119051cbSMarius Strobl } 1526119051cbSMarius Strobl #endif 1527119051cbSMarius Strobl 1528119051cbSMarius Strobl /* 1529119051cbSMarius Strobl * In theory, we could harvest some descriptors before 1530119051cbSMarius Strobl * the ring is empty, but that's a bit complicated. 1531119051cbSMarius Strobl * 1532119051cbSMarius Strobl * CAS_TX_COMPn points to the last descriptor 1533119051cbSMarius Strobl * processed + 1. 1534119051cbSMarius Strobl */ 1535119051cbSMarius Strobl txlast = CAS_READ_4(sc, CAS_TX_COMP3); 1536119051cbSMarius Strobl #ifdef CAS_DEBUG 1537119051cbSMarius Strobl CTR4(KTR_CAS, "%s: txs->txs_firstdesc = %d, " 1538119051cbSMarius Strobl "txs->txs_lastdesc = %d, txlast = %d", 1539119051cbSMarius Strobl __func__, txs->txs_firstdesc, txs->txs_lastdesc, txlast); 1540119051cbSMarius Strobl #endif 1541119051cbSMarius Strobl if (txs->txs_firstdesc <= txs->txs_lastdesc) { 1542119051cbSMarius Strobl if ((txlast >= txs->txs_firstdesc) && 1543119051cbSMarius Strobl (txlast <= txs->txs_lastdesc)) 1544119051cbSMarius Strobl break; 1545119051cbSMarius Strobl } else { 1546119051cbSMarius Strobl /* Ick -- this command wraps. */ 1547119051cbSMarius Strobl if ((txlast >= txs->txs_firstdesc) || 1548119051cbSMarius Strobl (txlast <= txs->txs_lastdesc)) 1549119051cbSMarius Strobl break; 1550119051cbSMarius Strobl } 1551119051cbSMarius Strobl 1552119051cbSMarius Strobl #ifdef CAS_DEBUG 1553119051cbSMarius Strobl CTR1(KTR_CAS, "%s: releasing a descriptor", __func__); 1554119051cbSMarius Strobl #endif 1555119051cbSMarius Strobl STAILQ_REMOVE_HEAD(&sc->sc_txdirtyq, txs_q); 1556119051cbSMarius Strobl 1557119051cbSMarius Strobl sc->sc_txfree += txs->txs_ndescs; 1558119051cbSMarius Strobl 1559119051cbSMarius Strobl bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 1560119051cbSMarius Strobl BUS_DMASYNC_POSTWRITE); 1561119051cbSMarius Strobl bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 1562119051cbSMarius Strobl if (txs->txs_mbuf != NULL) { 1563119051cbSMarius Strobl m_freem(txs->txs_mbuf); 1564119051cbSMarius Strobl txs->txs_mbuf = NULL; 1565119051cbSMarius Strobl } 1566119051cbSMarius Strobl 1567119051cbSMarius Strobl STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 1568119051cbSMarius Strobl 1569884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_OPACKETS, 1); 1570119051cbSMarius Strobl progress = 1; 1571119051cbSMarius Strobl } 1572119051cbSMarius Strobl 1573119051cbSMarius Strobl #ifdef CAS_DEBUG 157436334c57SMarius Strobl CTR5(KTR_CAS, "%s: CAS_TX_SM1 %x CAS_TX_SM2 %x CAS_TX_DESC_BASE %llx " 1575119051cbSMarius Strobl "CAS_TX_COMP3 %x", 157636334c57SMarius Strobl __func__, CAS_READ_4(sc, CAS_TX_SM1), CAS_READ_4(sc, CAS_TX_SM2), 157736334c57SMarius Strobl ((long long)CAS_READ_4(sc, CAS_TX_DESC3_BASE_HI) << 32) | 157836334c57SMarius Strobl CAS_READ_4(sc, CAS_TX_DESC3_BASE_LO), 1579119051cbSMarius Strobl CAS_READ_4(sc, CAS_TX_COMP3)); 1580119051cbSMarius Strobl #endif 1581119051cbSMarius Strobl 1582119051cbSMarius Strobl if (progress) { 158373591c46SMarius Strobl /* We freed some descriptors, so reset IFF_DRV_OACTIVE. */ 158444e65355SJustin Hibbits if_setdrvflagbits(ifp, 0, IFF_DRV_OACTIVE); 1585119051cbSMarius Strobl if (STAILQ_EMPTY(&sc->sc_txdirtyq)) 1586119051cbSMarius Strobl sc->sc_wdog_timer = 0; 1587119051cbSMarius Strobl } 1588119051cbSMarius Strobl 1589119051cbSMarius Strobl #ifdef CAS_DEBUG 1590119051cbSMarius Strobl CTR3(KTR_CAS, "%s: %s: watchdog %d", 1591119051cbSMarius Strobl device_get_name(sc->sc_dev), __func__, sc->sc_wdog_timer); 1592119051cbSMarius Strobl #endif 1593119051cbSMarius Strobl } 1594119051cbSMarius Strobl 1595119051cbSMarius Strobl static void 1596119051cbSMarius Strobl cas_rint_timeout(void *arg) 1597119051cbSMarius Strobl { 15986c3e93cbSGleb Smirnoff struct epoch_tracker et; 1599119051cbSMarius Strobl struct cas_softc *sc = arg; 1600119051cbSMarius Strobl 1601fcdbb666SMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1602119051cbSMarius Strobl 16036c3e93cbSGleb Smirnoff NET_EPOCH_ENTER(et); 1604119051cbSMarius Strobl cas_rint(sc); 16056c3e93cbSGleb Smirnoff NET_EPOCH_EXIT(et); 1606119051cbSMarius Strobl } 1607119051cbSMarius Strobl 1608119051cbSMarius Strobl static void 1609119051cbSMarius Strobl cas_rint(struct cas_softc *sc) 1610119051cbSMarius Strobl { 1611119051cbSMarius Strobl struct cas_rxdsoft *rxds, *rxds2; 161244e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 1613119051cbSMarius Strobl struct mbuf *m, *m2; 16145f598665SWarner Losh uint64_t word1, word2, word3 __unused, word4; 1615119051cbSMarius Strobl uint32_t rxhead; 1616119051cbSMarius Strobl u_int idx, idx2, len, off, skip; 1617119051cbSMarius Strobl 1618fcdbb666SMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1619119051cbSMarius Strobl 1620119051cbSMarius Strobl callout_stop(&sc->sc_rx_ch); 1621119051cbSMarius Strobl 1622119051cbSMarius Strobl #ifdef CAS_DEBUG 1623119051cbSMarius Strobl CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 1624119051cbSMarius Strobl #endif 1625119051cbSMarius Strobl 1626119051cbSMarius Strobl #define PRINTWORD(n, delimiter) \ 1627119051cbSMarius Strobl printf("word ## n: 0x%016llx%c", (long long)word ## n, delimiter) 1628119051cbSMarius Strobl 1629119051cbSMarius Strobl #define SKIPASSERT(n) \ 1630119051cbSMarius Strobl KASSERT(sc->sc_rxcomps[sc->sc_rxcptr].crc_word ## n == 0, \ 1631119051cbSMarius Strobl ("%s: word ## n not 0", __func__)) 1632119051cbSMarius Strobl 1633119051cbSMarius Strobl #define WORDTOH(n) \ 1634119051cbSMarius Strobl word ## n = le64toh(sc->sc_rxcomps[sc->sc_rxcptr].crc_word ## n) 1635119051cbSMarius Strobl 1636119051cbSMarius Strobl /* 1637119051cbSMarius Strobl * Read the completion head register once. This limits 1638119051cbSMarius Strobl * how long the following loop can execute. 1639119051cbSMarius Strobl */ 1640119051cbSMarius Strobl rxhead = CAS_READ_4(sc, CAS_RX_COMP_HEAD); 1641119051cbSMarius Strobl #ifdef CAS_DEBUG 1642119051cbSMarius Strobl CTR4(KTR_CAS, "%s: sc->sc_rxcptr %d, sc->sc_rxdptr %d, head %d", 164336334c57SMarius Strobl __func__, sc->sc_rxcptr, sc->sc_rxdptr, rxhead); 1644119051cbSMarius Strobl #endif 1645119051cbSMarius Strobl skip = 0; 1646119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 1647119051cbSMarius Strobl for (; sc->sc_rxcptr != rxhead; 1648119051cbSMarius Strobl sc->sc_rxcptr = CAS_NEXTRXCOMP(sc->sc_rxcptr)) { 1649119051cbSMarius Strobl if (skip != 0) { 1650119051cbSMarius Strobl SKIPASSERT(1); 1651119051cbSMarius Strobl SKIPASSERT(2); 1652119051cbSMarius Strobl SKIPASSERT(3); 1653119051cbSMarius Strobl 1654119051cbSMarius Strobl --skip; 1655119051cbSMarius Strobl goto skip; 1656119051cbSMarius Strobl } 1657119051cbSMarius Strobl 1658119051cbSMarius Strobl WORDTOH(1); 1659119051cbSMarius Strobl WORDTOH(2); 1660119051cbSMarius Strobl WORDTOH(3); 1661119051cbSMarius Strobl WORDTOH(4); 1662119051cbSMarius Strobl 1663119051cbSMarius Strobl #ifdef CAS_DEBUG 166444e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_DEBUG) != 0) { 1665119051cbSMarius Strobl printf(" completion %d: ", sc->sc_rxcptr); 1666119051cbSMarius Strobl PRINTWORD(1, '\t'); 1667119051cbSMarius Strobl PRINTWORD(2, '\t'); 1668119051cbSMarius Strobl PRINTWORD(3, '\t'); 1669119051cbSMarius Strobl PRINTWORD(4, '\n'); 1670119051cbSMarius Strobl } 1671119051cbSMarius Strobl #endif 1672119051cbSMarius Strobl 1673119051cbSMarius Strobl if (__predict_false( 1674119051cbSMarius Strobl (word1 & CAS_RC1_TYPE_MASK) == CAS_RC1_TYPE_HW || 1675119051cbSMarius Strobl (word4 & CAS_RC4_ZERO) != 0)) { 1676119051cbSMarius Strobl /* 1677119051cbSMarius Strobl * The descriptor is still marked as owned, although 1678119051cbSMarius Strobl * it is supposed to have completed. This has been 1679119051cbSMarius Strobl * observed on some machines. Just exiting here 1680119051cbSMarius Strobl * might leave the packet sitting around until another 1681119051cbSMarius Strobl * one arrives to trigger a new interrupt, which is 1682119051cbSMarius Strobl * generally undesirable, so set up a timeout. 1683119051cbSMarius Strobl */ 1684119051cbSMarius Strobl callout_reset(&sc->sc_rx_ch, CAS_RXOWN_TICKS, 1685119051cbSMarius Strobl cas_rint_timeout, sc); 1686119051cbSMarius Strobl break; 1687119051cbSMarius Strobl } 1688119051cbSMarius Strobl 1689119051cbSMarius Strobl if (__predict_false( 1690119051cbSMarius Strobl (word4 & (CAS_RC4_BAD | CAS_RC4_LEN_MMATCH)) != 0)) { 1691884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IERRORS, 1); 1692119051cbSMarius Strobl device_printf(sc->sc_dev, 1693119051cbSMarius Strobl "receive error: CRC error\n"); 1694119051cbSMarius Strobl continue; 1695119051cbSMarius Strobl } 1696119051cbSMarius Strobl 1697119051cbSMarius Strobl KASSERT(CAS_GET(word1, CAS_RC1_DATA_SIZE) == 0 || 1698119051cbSMarius Strobl CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0, 1699119051cbSMarius Strobl ("%s: data and header present", __func__)); 1700119051cbSMarius Strobl KASSERT((word1 & CAS_RC1_SPLIT_PKT) == 0 || 1701119051cbSMarius Strobl CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0, 1702119051cbSMarius Strobl ("%s: split and header present", __func__)); 1703119051cbSMarius Strobl KASSERT(CAS_GET(word1, CAS_RC1_DATA_SIZE) == 0 || 1704119051cbSMarius Strobl (word1 & CAS_RC1_RELEASE_HDR) == 0, 1705119051cbSMarius Strobl ("%s: data present but header release", __func__)); 1706119051cbSMarius Strobl KASSERT(CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0 || 1707119051cbSMarius Strobl (word1 & CAS_RC1_RELEASE_DATA) == 0, 1708119051cbSMarius Strobl ("%s: header present but data release", __func__)); 1709119051cbSMarius Strobl 1710119051cbSMarius Strobl if ((len = CAS_GET(word2, CAS_RC2_HDR_SIZE)) != 0) { 1711119051cbSMarius Strobl idx = CAS_GET(word2, CAS_RC2_HDR_INDEX); 1712119051cbSMarius Strobl off = CAS_GET(word2, CAS_RC2_HDR_OFF); 1713119051cbSMarius Strobl #ifdef CAS_DEBUG 1714119051cbSMarius Strobl CTR4(KTR_CAS, "%s: hdr at idx %d, off %d, len %d", 1715119051cbSMarius Strobl __func__, idx, off, len); 1716119051cbSMarius Strobl #endif 1717119051cbSMarius Strobl rxds = &sc->sc_rxdsoft[idx]; 1718c6499eccSGleb Smirnoff MGETHDR(m, M_NOWAIT, MT_DATA); 1719119051cbSMarius Strobl if (m != NULL) { 1720119051cbSMarius Strobl refcount_acquire(&rxds->rxds_refcount); 1721119051cbSMarius Strobl bus_dmamap_sync(sc->sc_rdmatag, 1722119051cbSMarius Strobl rxds->rxds_dmamap, BUS_DMASYNC_POSTREAD); 1723e8fd18f3SGleb Smirnoff m_extadd(m, (char *)rxds->rxds_buf + 1724fe2e5c4dSMarius Strobl off * 256 + ETHER_ALIGN, len, cas_free, 1725119051cbSMarius Strobl sc, (void *)(uintptr_t)idx, 1726119051cbSMarius Strobl M_RDONLY, EXT_NET_DRV); 1727119051cbSMarius Strobl if ((m->m_flags & M_EXT) == 0) { 1728119051cbSMarius Strobl m_freem(m); 1729119051cbSMarius Strobl m = NULL; 1730119051cbSMarius Strobl } 1731119051cbSMarius Strobl } 1732119051cbSMarius Strobl if (m != NULL) { 1733119051cbSMarius Strobl m->m_pkthdr.rcvif = ifp; 1734119051cbSMarius Strobl m->m_pkthdr.len = m->m_len = len; 1735884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IPACKETS, 1); 173644e65355SJustin Hibbits if ((if_getcapenable(ifp) & IFCAP_RXCSUM) != 0) 1737119051cbSMarius Strobl cas_rxcksum(m, CAS_GET(word4, 1738119051cbSMarius Strobl CAS_RC4_TCP_CSUM)); 1739119051cbSMarius Strobl /* Pass it on. */ 1740fcdbb666SMarius Strobl CAS_UNLOCK(sc); 174144e65355SJustin Hibbits if_input(ifp, m); 1742fcdbb666SMarius Strobl CAS_LOCK(sc); 1743119051cbSMarius Strobl } else 1744884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IQDROPS, 1); 1745119051cbSMarius Strobl 1746119051cbSMarius Strobl if ((word1 & CAS_RC1_RELEASE_HDR) != 0 && 1747119051cbSMarius Strobl refcount_release(&rxds->rxds_refcount) != 0) 1748119051cbSMarius Strobl cas_add_rxdesc(sc, idx); 1749119051cbSMarius Strobl } else if ((len = CAS_GET(word1, CAS_RC1_DATA_SIZE)) != 0) { 1750119051cbSMarius Strobl idx = CAS_GET(word1, CAS_RC1_DATA_INDEX); 1751119051cbSMarius Strobl off = CAS_GET(word1, CAS_RC1_DATA_OFF); 1752119051cbSMarius Strobl #ifdef CAS_DEBUG 1753119051cbSMarius Strobl CTR4(KTR_CAS, "%s: data at idx %d, off %d, len %d", 1754119051cbSMarius Strobl __func__, idx, off, len); 1755119051cbSMarius Strobl #endif 1756119051cbSMarius Strobl rxds = &sc->sc_rxdsoft[idx]; 1757c6499eccSGleb Smirnoff MGETHDR(m, M_NOWAIT, MT_DATA); 1758119051cbSMarius Strobl if (m != NULL) { 1759119051cbSMarius Strobl refcount_acquire(&rxds->rxds_refcount); 1760119051cbSMarius Strobl off += ETHER_ALIGN; 1761119051cbSMarius Strobl m->m_len = min(CAS_PAGE_SIZE - off, len); 1762119051cbSMarius Strobl bus_dmamap_sync(sc->sc_rdmatag, 1763119051cbSMarius Strobl rxds->rxds_dmamap, BUS_DMASYNC_POSTREAD); 1764e8fd18f3SGleb Smirnoff m_extadd(m, (char *)rxds->rxds_buf + off, 1765fe2e5c4dSMarius Strobl m->m_len, cas_free, sc, 1766fe2e5c4dSMarius Strobl (void *)(uintptr_t)idx, M_RDONLY, 1767fe2e5c4dSMarius Strobl EXT_NET_DRV); 1768119051cbSMarius Strobl if ((m->m_flags & M_EXT) == 0) { 1769119051cbSMarius Strobl m_freem(m); 1770119051cbSMarius Strobl m = NULL; 1771119051cbSMarius Strobl } 1772119051cbSMarius Strobl } 1773119051cbSMarius Strobl idx2 = 0; 1774b1abfe6cSMarius Strobl m2 = NULL; 1775119051cbSMarius Strobl rxds2 = NULL; 1776119051cbSMarius Strobl if ((word1 & CAS_RC1_SPLIT_PKT) != 0) { 1777119051cbSMarius Strobl KASSERT((word1 & CAS_RC1_RELEASE_NEXT) != 0, 1778119051cbSMarius Strobl ("%s: split but no release next", 1779119051cbSMarius Strobl __func__)); 1780119051cbSMarius Strobl 1781119051cbSMarius Strobl idx2 = CAS_GET(word2, CAS_RC2_NEXT_INDEX); 1782119051cbSMarius Strobl #ifdef CAS_DEBUG 1783119051cbSMarius Strobl CTR2(KTR_CAS, "%s: split at idx %d", 1784119051cbSMarius Strobl __func__, idx2); 1785119051cbSMarius Strobl #endif 1786119051cbSMarius Strobl rxds2 = &sc->sc_rxdsoft[idx2]; 1787b1abfe6cSMarius Strobl if (m != NULL) { 1788c6499eccSGleb Smirnoff MGET(m2, M_NOWAIT, MT_DATA); 1789119051cbSMarius Strobl if (m2 != NULL) { 1790119051cbSMarius Strobl refcount_acquire( 1791119051cbSMarius Strobl &rxds2->rxds_refcount); 1792119051cbSMarius Strobl m2->m_len = len - m->m_len; 1793b1abfe6cSMarius Strobl bus_dmamap_sync( 1794b1abfe6cSMarius Strobl sc->sc_rdmatag, 1795119051cbSMarius Strobl rxds2->rxds_dmamap, 1796119051cbSMarius Strobl BUS_DMASYNC_POSTREAD); 1797e8fd18f3SGleb Smirnoff m_extadd(m2, 1798e8fd18f3SGleb Smirnoff (char *)rxds2->rxds_buf, 1799b1abfe6cSMarius Strobl m2->m_len, cas_free, sc, 1800b1abfe6cSMarius Strobl (void *)(uintptr_t)idx2, 1801119051cbSMarius Strobl M_RDONLY, EXT_NET_DRV); 1802b1abfe6cSMarius Strobl if ((m2->m_flags & M_EXT) == 1803b1abfe6cSMarius Strobl 0) { 1804119051cbSMarius Strobl m_freem(m2); 1805119051cbSMarius Strobl m2 = NULL; 1806119051cbSMarius Strobl } 1807119051cbSMarius Strobl } 1808b1abfe6cSMarius Strobl } 1809119051cbSMarius Strobl if (m2 != NULL) 1810119051cbSMarius Strobl m->m_next = m2; 1811b1abfe6cSMarius Strobl else if (m != NULL) { 1812119051cbSMarius Strobl m_freem(m); 1813119051cbSMarius Strobl m = NULL; 1814119051cbSMarius Strobl } 1815119051cbSMarius Strobl } 1816119051cbSMarius Strobl if (m != NULL) { 1817119051cbSMarius Strobl m->m_pkthdr.rcvif = ifp; 1818119051cbSMarius Strobl m->m_pkthdr.len = len; 1819884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IPACKETS, 1); 182044e65355SJustin Hibbits if ((if_getcapenable(ifp) & IFCAP_RXCSUM) != 0) 1821119051cbSMarius Strobl cas_rxcksum(m, CAS_GET(word4, 1822119051cbSMarius Strobl CAS_RC4_TCP_CSUM)); 1823119051cbSMarius Strobl /* Pass it on. */ 1824fcdbb666SMarius Strobl CAS_UNLOCK(sc); 182544e65355SJustin Hibbits if_input(ifp, m); 1826fcdbb666SMarius Strobl CAS_LOCK(sc); 1827119051cbSMarius Strobl } else 1828884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IQDROPS, 1); 1829119051cbSMarius Strobl 1830119051cbSMarius Strobl if ((word1 & CAS_RC1_RELEASE_DATA) != 0 && 1831119051cbSMarius Strobl refcount_release(&rxds->rxds_refcount) != 0) 1832119051cbSMarius Strobl cas_add_rxdesc(sc, idx); 1833119051cbSMarius Strobl if ((word1 & CAS_RC1_SPLIT_PKT) != 0 && 1834119051cbSMarius Strobl refcount_release(&rxds2->rxds_refcount) != 0) 1835119051cbSMarius Strobl cas_add_rxdesc(sc, idx2); 1836119051cbSMarius Strobl } 1837119051cbSMarius Strobl 1838119051cbSMarius Strobl skip = CAS_GET(word1, CAS_RC1_SKIP); 1839119051cbSMarius Strobl 1840119051cbSMarius Strobl skip: 1841119051cbSMarius Strobl cas_rxcompinit(&sc->sc_rxcomps[sc->sc_rxcptr]); 184244e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) == 0) 184373591c46SMarius Strobl break; 1844119051cbSMarius Strobl } 1845119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1846119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_COMP_TAIL, sc->sc_rxcptr); 1847119051cbSMarius Strobl 1848119051cbSMarius Strobl #undef PRINTWORD 1849119051cbSMarius Strobl #undef SKIPASSERT 1850119051cbSMarius Strobl #undef WORDTOH 1851119051cbSMarius Strobl 1852119051cbSMarius Strobl #ifdef CAS_DEBUG 1853119051cbSMarius Strobl CTR4(KTR_CAS, "%s: done sc->sc_rxcptr %d, sc->sc_rxdptr %d, head %d", 185436334c57SMarius Strobl __func__, sc->sc_rxcptr, sc->sc_rxdptr, 1855119051cbSMarius Strobl CAS_READ_4(sc, CAS_RX_COMP_HEAD)); 1856119051cbSMarius Strobl #endif 1857119051cbSMarius Strobl } 1858119051cbSMarius Strobl 185915c28f87SGleb Smirnoff static void 1860e8fd18f3SGleb Smirnoff cas_free(struct mbuf *m) 1861119051cbSMarius Strobl { 1862119051cbSMarius Strobl struct cas_rxdsoft *rxds; 1863119051cbSMarius Strobl struct cas_softc *sc; 1864fcdbb666SMarius Strobl u_int idx, locked; 1865119051cbSMarius Strobl 1866e8fd18f3SGleb Smirnoff sc = m->m_ext.ext_arg1; 1867e8fd18f3SGleb Smirnoff idx = (uintptr_t)m->m_ext.ext_arg2; 1868119051cbSMarius Strobl rxds = &sc->sc_rxdsoft[idx]; 1869119051cbSMarius Strobl if (refcount_release(&rxds->rxds_refcount) == 0) 187015c28f87SGleb Smirnoff return; 1871119051cbSMarius Strobl 1872119051cbSMarius Strobl /* 1873119051cbSMarius Strobl * NB: this function can be called via m_freem(9) within 1874119051cbSMarius Strobl * this driver! 1875119051cbSMarius Strobl */ 1876fcdbb666SMarius Strobl if ((locked = CAS_LOCK_OWNED(sc)) == 0) 1877fcdbb666SMarius Strobl CAS_LOCK(sc); 1878119051cbSMarius Strobl cas_add_rxdesc(sc, idx); 1879fcdbb666SMarius Strobl if (locked == 0) 1880fcdbb666SMarius Strobl CAS_UNLOCK(sc); 1881119051cbSMarius Strobl } 1882119051cbSMarius Strobl 1883119051cbSMarius Strobl static inline void 1884119051cbSMarius Strobl cas_add_rxdesc(struct cas_softc *sc, u_int idx) 1885119051cbSMarius Strobl { 1886119051cbSMarius Strobl 1887fcdbb666SMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 1888119051cbSMarius Strobl 1889119051cbSMarius Strobl bus_dmamap_sync(sc->sc_rdmatag, sc->sc_rxdsoft[idx].rxds_dmamap, 1890119051cbSMarius Strobl BUS_DMASYNC_PREREAD); 1891119051cbSMarius Strobl CAS_UPDATE_RXDESC(sc, sc->sc_rxdptr, idx); 1892119051cbSMarius Strobl sc->sc_rxdptr = CAS_NEXTRXDESC(sc->sc_rxdptr); 1893119051cbSMarius Strobl 1894119051cbSMarius Strobl /* 1895119051cbSMarius Strobl * Update the RX kick register. This register has to point to the 1896119051cbSMarius Strobl * descriptor after the last valid one (before the current batch) 1897119051cbSMarius Strobl * and for optimum performance should be incremented in multiples 1898119051cbSMarius Strobl * of 4 (the DMA engine fetches/updates descriptors in batches of 4). 1899119051cbSMarius Strobl */ 1900119051cbSMarius Strobl if ((sc->sc_rxdptr % 4) == 0) { 1901119051cbSMarius Strobl CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1902119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_RX_KICK, 1903119051cbSMarius Strobl (sc->sc_rxdptr + CAS_NRXDESC - 4) & CAS_NRXDESC_MASK); 1904119051cbSMarius Strobl } 1905119051cbSMarius Strobl } 1906119051cbSMarius Strobl 1907119051cbSMarius Strobl static void 1908119051cbSMarius Strobl cas_eint(struct cas_softc *sc, u_int status) 1909119051cbSMarius Strobl { 191044e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 1911119051cbSMarius Strobl 1912fcdbb666SMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 191373591c46SMarius Strobl 1914884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IERRORS, 1); 1915119051cbSMarius Strobl 1916119051cbSMarius Strobl device_printf(sc->sc_dev, "%s: status 0x%x", __func__, status); 1917119051cbSMarius Strobl if ((status & CAS_INTR_PCI_ERROR_INT) != 0) { 1918119051cbSMarius Strobl status = CAS_READ_4(sc, CAS_ERROR_STATUS); 1919119051cbSMarius Strobl printf(", PCI bus error 0x%x", status); 1920119051cbSMarius Strobl if ((status & CAS_ERROR_OTHER) != 0) { 1921119051cbSMarius Strobl status = pci_read_config(sc->sc_dev, PCIR_STATUS, 2); 1922119051cbSMarius Strobl printf(", PCI status 0x%x", status); 1923119051cbSMarius Strobl pci_write_config(sc->sc_dev, PCIR_STATUS, status, 2); 1924119051cbSMarius Strobl } 1925119051cbSMarius Strobl } 1926119051cbSMarius Strobl printf("\n"); 1927119051cbSMarius Strobl 192844e65355SJustin Hibbits if_setdrvflagbits(ifp, 0, IFF_DRV_RUNNING); 1929fcdbb666SMarius Strobl cas_init_locked(sc); 193044e65355SJustin Hibbits if (!if_sendq_empty(ifp)) 193173591c46SMarius Strobl taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 1932119051cbSMarius Strobl } 1933119051cbSMarius Strobl 193473591c46SMarius Strobl static int 1935119051cbSMarius Strobl cas_intr(void *v) 1936119051cbSMarius Strobl { 1937119051cbSMarius Strobl struct cas_softc *sc = v; 193873591c46SMarius Strobl 193973591c46SMarius Strobl if (__predict_false((CAS_READ_4(sc, CAS_STATUS_ALIAS) & 194073591c46SMarius Strobl CAS_INTR_SUMMARY) == 0)) 194173591c46SMarius Strobl return (FILTER_STRAY); 194273591c46SMarius Strobl 194373591c46SMarius Strobl /* Disable interrupts. */ 194473591c46SMarius Strobl CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 194573591c46SMarius Strobl taskqueue_enqueue(sc->sc_tq, &sc->sc_intr_task); 194673591c46SMarius Strobl 194773591c46SMarius Strobl return (FILTER_HANDLED); 194873591c46SMarius Strobl } 194973591c46SMarius Strobl 195073591c46SMarius Strobl static void 195173591c46SMarius Strobl cas_intr_task(void *arg, int pending __unused) 195273591c46SMarius Strobl { 195373591c46SMarius Strobl struct cas_softc *sc = arg; 195444e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 1955119051cbSMarius Strobl uint32_t status, status2; 1956119051cbSMarius Strobl 195773591c46SMarius Strobl CAS_LOCK_ASSERT(sc, MA_NOTOWNED); 195873591c46SMarius Strobl 195944e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) == 0) 196073591c46SMarius Strobl return; 196173591c46SMarius Strobl 1962119051cbSMarius Strobl status = CAS_READ_4(sc, CAS_STATUS); 1963119051cbSMarius Strobl if (__predict_false((status & CAS_INTR_SUMMARY) == 0)) 196473591c46SMarius Strobl goto done; 1965119051cbSMarius Strobl 1966fcdbb666SMarius Strobl CAS_LOCK(sc); 1967119051cbSMarius Strobl #ifdef CAS_DEBUG 1968119051cbSMarius Strobl CTR4(KTR_CAS, "%s: %s: cplt %x, status %x", 1969119051cbSMarius Strobl device_get_name(sc->sc_dev), __func__, 197036334c57SMarius Strobl (status >> CAS_STATUS_TX_COMP3_SHFT), (u_int)status); 1971119051cbSMarius Strobl 1972119051cbSMarius Strobl /* 1973119051cbSMarius Strobl * PCS interrupts must be cleared, otherwise no traffic is passed! 1974119051cbSMarius Strobl */ 1975119051cbSMarius Strobl if ((status & CAS_INTR_PCS_INT) != 0) { 1976119051cbSMarius Strobl status2 = 1977119051cbSMarius Strobl CAS_READ_4(sc, CAS_PCS_INTR_STATUS) | 1978119051cbSMarius Strobl CAS_READ_4(sc, CAS_PCS_INTR_STATUS); 1979119051cbSMarius Strobl if ((status2 & CAS_PCS_INTR_LINK) != 0) 1980119051cbSMarius Strobl device_printf(sc->sc_dev, 1981119051cbSMarius Strobl "%s: PCS link status changed\n", __func__); 1982119051cbSMarius Strobl } 1983119051cbSMarius Strobl if ((status & CAS_MAC_CTRL_STATUS) != 0) { 1984119051cbSMarius Strobl status2 = CAS_READ_4(sc, CAS_MAC_CTRL_STATUS); 1985119051cbSMarius Strobl if ((status2 & CAS_MAC_CTRL_PAUSE) != 0) 1986119051cbSMarius Strobl device_printf(sc->sc_dev, 1987119051cbSMarius Strobl "%s: PAUSE received (PAUSE time %d slots)\n", 1988119051cbSMarius Strobl __func__, 1989119051cbSMarius Strobl (status2 & CAS_MAC_CTRL_STATUS_PT_MASK) >> 1990119051cbSMarius Strobl CAS_MAC_CTRL_STATUS_PT_SHFT); 1991119051cbSMarius Strobl if ((status2 & CAS_MAC_CTRL_PAUSE) != 0) 1992119051cbSMarius Strobl device_printf(sc->sc_dev, 1993119051cbSMarius Strobl "%s: transited to PAUSE state\n", __func__); 1994119051cbSMarius Strobl if ((status2 & CAS_MAC_CTRL_NON_PAUSE) != 0) 1995119051cbSMarius Strobl device_printf(sc->sc_dev, 1996119051cbSMarius Strobl "%s: transited to non-PAUSE state\n", __func__); 1997119051cbSMarius Strobl } 1998119051cbSMarius Strobl if ((status & CAS_INTR_MIF) != 0) 1999119051cbSMarius Strobl device_printf(sc->sc_dev, "%s: MIF interrupt\n", __func__); 2000119051cbSMarius Strobl #endif 2001119051cbSMarius Strobl 2002119051cbSMarius Strobl if (__predict_false((status & 2003119051cbSMarius Strobl (CAS_INTR_TX_TAG_ERR | CAS_INTR_RX_TAG_ERR | 2004119051cbSMarius Strobl CAS_INTR_RX_LEN_MMATCH | CAS_INTR_PCI_ERROR_INT)) != 0)) { 2005119051cbSMarius Strobl cas_eint(sc, status); 2006fcdbb666SMarius Strobl CAS_UNLOCK(sc); 2007119051cbSMarius Strobl return; 2008119051cbSMarius Strobl } 2009119051cbSMarius Strobl 2010119051cbSMarius Strobl if (__predict_false(status & CAS_INTR_TX_MAC_INT)) { 2011119051cbSMarius Strobl status2 = CAS_READ_4(sc, CAS_MAC_TX_STATUS); 2012119051cbSMarius Strobl if ((status2 & 2013119051cbSMarius Strobl (CAS_MAC_TX_UNDERRUN | CAS_MAC_TX_MAX_PKT_ERR)) != 0) 2014884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_OERRORS, 1); 2015119051cbSMarius Strobl else if ((status2 & ~CAS_MAC_TX_FRAME_XMTD) != 0) 2016119051cbSMarius Strobl device_printf(sc->sc_dev, 2017119051cbSMarius Strobl "MAC TX fault, status %x\n", status2); 2018119051cbSMarius Strobl } 2019119051cbSMarius Strobl 2020119051cbSMarius Strobl if (__predict_false(status & CAS_INTR_RX_MAC_INT)) { 2021119051cbSMarius Strobl status2 = CAS_READ_4(sc, CAS_MAC_RX_STATUS); 2022119051cbSMarius Strobl if ((status2 & CAS_MAC_RX_OVERFLOW) != 0) 2023884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_IERRORS, 1); 2024119051cbSMarius Strobl else if ((status2 & ~CAS_MAC_RX_FRAME_RCVD) != 0) 2025119051cbSMarius Strobl device_printf(sc->sc_dev, 2026119051cbSMarius Strobl "MAC RX fault, status %x\n", status2); 2027119051cbSMarius Strobl } 2028119051cbSMarius Strobl 2029119051cbSMarius Strobl if ((status & 2030119051cbSMarius Strobl (CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_COMP_FULL | 2031119051cbSMarius Strobl CAS_INTR_RX_BUF_AEMPTY | CAS_INTR_RX_COMP_AFULL)) != 0) { 2032119051cbSMarius Strobl cas_rint(sc); 203373591c46SMarius Strobl #ifdef CAS_DEBUG 2034119051cbSMarius Strobl if (__predict_false((status & 2035119051cbSMarius Strobl (CAS_INTR_RX_BUF_NA | CAS_INTR_RX_COMP_FULL | 2036119051cbSMarius Strobl CAS_INTR_RX_BUF_AEMPTY | CAS_INTR_RX_COMP_AFULL)) != 0)) 2037119051cbSMarius Strobl device_printf(sc->sc_dev, 2038119051cbSMarius Strobl "RX fault, status %x\n", status); 203973591c46SMarius Strobl #endif 2040119051cbSMarius Strobl } 2041119051cbSMarius Strobl 2042119051cbSMarius Strobl if ((status & 2043fcdbb666SMarius Strobl (CAS_INTR_TX_INT_ME | CAS_INTR_TX_ALL | CAS_INTR_TX_DONE)) != 0) 2044119051cbSMarius Strobl cas_tint(sc); 2045119051cbSMarius Strobl 204644e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) == 0) { 2047fcdbb666SMarius Strobl CAS_UNLOCK(sc); 204873591c46SMarius Strobl return; 204944e65355SJustin Hibbits } else if (!if_sendq_empty(ifp)) 205073591c46SMarius Strobl taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 2051fcdbb666SMarius Strobl CAS_UNLOCK(sc); 205273591c46SMarius Strobl 205373591c46SMarius Strobl status = CAS_READ_4(sc, CAS_STATUS_ALIAS); 205473591c46SMarius Strobl if (__predict_false((status & CAS_INTR_SUMMARY) != 0)) { 205573591c46SMarius Strobl taskqueue_enqueue(sc->sc_tq, &sc->sc_intr_task); 205673591c46SMarius Strobl return; 205773591c46SMarius Strobl } 205873591c46SMarius Strobl 205973591c46SMarius Strobl done: 206073591c46SMarius Strobl /* Re-enable interrupts. */ 206173591c46SMarius Strobl CAS_WRITE_4(sc, CAS_INTMASK, 206273591c46SMarius Strobl ~(CAS_INTR_TX_INT_ME | CAS_INTR_TX_TAG_ERR | 206373591c46SMarius Strobl CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_TAG_ERR | 206473591c46SMarius Strobl CAS_INTR_RX_COMP_FULL | CAS_INTR_RX_BUF_AEMPTY | 206573591c46SMarius Strobl CAS_INTR_RX_COMP_AFULL | CAS_INTR_RX_LEN_MMATCH | 206673591c46SMarius Strobl CAS_INTR_PCI_ERROR_INT 206773591c46SMarius Strobl #ifdef CAS_DEBUG 206873591c46SMarius Strobl | CAS_INTR_PCS_INT | CAS_INTR_MIF 206973591c46SMarius Strobl #endif 207073591c46SMarius Strobl )); 207173591c46SMarius Strobl } 207273591c46SMarius Strobl 207373591c46SMarius Strobl static void 2074119051cbSMarius Strobl cas_watchdog(struct cas_softc *sc) 2075119051cbSMarius Strobl { 207644e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 2077119051cbSMarius Strobl 2078119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 2079119051cbSMarius Strobl 2080119051cbSMarius Strobl #ifdef CAS_DEBUG 2081119051cbSMarius Strobl CTR4(KTR_CAS, 208236334c57SMarius Strobl "%s: CAS_RX_CONF %x CAS_MAC_RX_STATUS %x CAS_MAC_RX_CONF %x", 208336334c57SMarius Strobl __func__, CAS_READ_4(sc, CAS_RX_CONF), 2084119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_STATUS), 208536334c57SMarius Strobl CAS_READ_4(sc, CAS_MAC_RX_CONF)); 2086119051cbSMarius Strobl CTR4(KTR_CAS, 208736334c57SMarius Strobl "%s: CAS_TX_CONF %x CAS_MAC_TX_STATUS %x CAS_MAC_TX_CONF %x", 208836334c57SMarius Strobl __func__, CAS_READ_4(sc, CAS_TX_CONF), 2089119051cbSMarius Strobl CAS_READ_4(sc, CAS_MAC_TX_STATUS), 209036334c57SMarius Strobl CAS_READ_4(sc, CAS_MAC_TX_CONF)); 2091119051cbSMarius Strobl #endif 2092119051cbSMarius Strobl 2093119051cbSMarius Strobl if (sc->sc_wdog_timer == 0 || --sc->sc_wdog_timer != 0) 209473591c46SMarius Strobl return; 2095119051cbSMarius Strobl 2096119051cbSMarius Strobl if ((sc->sc_flags & CAS_LINK) != 0) 2097119051cbSMarius Strobl device_printf(sc->sc_dev, "device timeout\n"); 2098119051cbSMarius Strobl else if (bootverbose) 2099119051cbSMarius Strobl device_printf(sc->sc_dev, "device timeout (no link)\n"); 2100884c5b45SGleb Smirnoff if_inc_counter(ifp, IFCOUNTER_OERRORS, 1); 2101119051cbSMarius Strobl 2102119051cbSMarius Strobl /* Try to get more packets going. */ 210344e65355SJustin Hibbits if_setdrvflagbits(ifp, 0, IFF_DRV_RUNNING); 2104119051cbSMarius Strobl cas_init_locked(sc); 210544e65355SJustin Hibbits if (!if_sendq_empty(ifp)) 210673591c46SMarius Strobl taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 2107119051cbSMarius Strobl } 2108119051cbSMarius Strobl 2109119051cbSMarius Strobl static void 2110119051cbSMarius Strobl cas_mifinit(struct cas_softc *sc) 2111119051cbSMarius Strobl { 2112119051cbSMarius Strobl 2113119051cbSMarius Strobl /* Configure the MIF in frame mode. */ 2114119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MIF_CONF, 2115119051cbSMarius Strobl CAS_READ_4(sc, CAS_MIF_CONF) & ~CAS_MIF_CONF_BB_MODE); 2116917825acSMarius Strobl CAS_BARRIER(sc, CAS_MIF_CONF, 4, 2117917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2118119051cbSMarius Strobl } 2119119051cbSMarius Strobl 2120119051cbSMarius Strobl /* 2121119051cbSMarius Strobl * MII interface 2122119051cbSMarius Strobl * 2123119051cbSMarius Strobl * The MII interface supports at least three different operating modes: 2124119051cbSMarius Strobl * 2125119051cbSMarius Strobl * Bitbang mode is implemented using data, clock and output enable registers. 2126119051cbSMarius Strobl * 2127119051cbSMarius Strobl * Frame mode is implemented by loading a complete frame into the frame 2128119051cbSMarius Strobl * register and polling the valid bit for completion. 2129119051cbSMarius Strobl * 2130119051cbSMarius Strobl * Polling mode uses the frame register but completion is indicated by 2131119051cbSMarius Strobl * an interrupt. 2132119051cbSMarius Strobl * 2133119051cbSMarius Strobl */ 2134119051cbSMarius Strobl static int 2135119051cbSMarius Strobl cas_mii_readreg(device_t dev, int phy, int reg) 2136119051cbSMarius Strobl { 2137119051cbSMarius Strobl struct cas_softc *sc; 2138119051cbSMarius Strobl int n; 2139119051cbSMarius Strobl uint32_t v; 2140119051cbSMarius Strobl 2141119051cbSMarius Strobl #ifdef CAS_DEBUG_PHY 2142119051cbSMarius Strobl printf("%s: phy %d reg %d\n", __func__, phy, reg); 2143119051cbSMarius Strobl #endif 2144119051cbSMarius Strobl 2145119051cbSMarius Strobl sc = device_get_softc(dev); 2146119051cbSMarius Strobl if ((sc->sc_flags & CAS_SERDES) != 0) { 2147119051cbSMarius Strobl switch (reg) { 2148119051cbSMarius Strobl case MII_BMCR: 2149119051cbSMarius Strobl reg = CAS_PCS_CTRL; 2150119051cbSMarius Strobl break; 2151119051cbSMarius Strobl case MII_BMSR: 2152119051cbSMarius Strobl reg = CAS_PCS_STATUS; 2153119051cbSMarius Strobl break; 2154119051cbSMarius Strobl case MII_PHYIDR1: 2155119051cbSMarius Strobl case MII_PHYIDR2: 2156119051cbSMarius Strobl return (0); 2157119051cbSMarius Strobl case MII_ANAR: 2158119051cbSMarius Strobl reg = CAS_PCS_ANAR; 2159119051cbSMarius Strobl break; 2160119051cbSMarius Strobl case MII_ANLPAR: 2161119051cbSMarius Strobl reg = CAS_PCS_ANLPAR; 2162119051cbSMarius Strobl break; 2163119051cbSMarius Strobl case MII_EXTSR: 2164119051cbSMarius Strobl return (EXTSR_1000XFDX | EXTSR_1000XHDX); 2165119051cbSMarius Strobl default: 2166119051cbSMarius Strobl device_printf(sc->sc_dev, 2167119051cbSMarius Strobl "%s: unhandled register %d\n", __func__, reg); 2168119051cbSMarius Strobl return (0); 2169119051cbSMarius Strobl } 2170119051cbSMarius Strobl return (CAS_READ_4(sc, reg)); 2171119051cbSMarius Strobl } 2172119051cbSMarius Strobl 2173119051cbSMarius Strobl /* Construct the frame command. */ 2174119051cbSMarius Strobl v = CAS_MIF_FRAME_READ | 2175119051cbSMarius Strobl (phy << CAS_MIF_FRAME_PHY_SHFT) | 2176119051cbSMarius Strobl (reg << CAS_MIF_FRAME_REG_SHFT); 2177119051cbSMarius Strobl 2178119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MIF_FRAME, v); 2179119051cbSMarius Strobl CAS_BARRIER(sc, CAS_MIF_FRAME, 4, 2180119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2181119051cbSMarius Strobl for (n = 0; n < 100; n++) { 2182119051cbSMarius Strobl DELAY(1); 2183119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MIF_FRAME); 2184119051cbSMarius Strobl if (v & CAS_MIF_FRAME_TA_LSB) 2185119051cbSMarius Strobl return (v & CAS_MIF_FRAME_DATA); 2186119051cbSMarius Strobl } 2187119051cbSMarius Strobl 2188119051cbSMarius Strobl device_printf(sc->sc_dev, "%s: timed out\n", __func__); 2189119051cbSMarius Strobl return (0); 2190119051cbSMarius Strobl } 2191119051cbSMarius Strobl 2192119051cbSMarius Strobl static int 2193119051cbSMarius Strobl cas_mii_writereg(device_t dev, int phy, int reg, int val) 2194119051cbSMarius Strobl { 2195119051cbSMarius Strobl struct cas_softc *sc; 2196119051cbSMarius Strobl int n; 2197119051cbSMarius Strobl uint32_t v; 2198119051cbSMarius Strobl 2199119051cbSMarius Strobl #ifdef CAS_DEBUG_PHY 2200119051cbSMarius Strobl printf("%s: phy %d reg %d val %x\n", phy, reg, val, __func__); 2201119051cbSMarius Strobl #endif 2202119051cbSMarius Strobl 2203119051cbSMarius Strobl sc = device_get_softc(dev); 2204119051cbSMarius Strobl if ((sc->sc_flags & CAS_SERDES) != 0) { 2205119051cbSMarius Strobl switch (reg) { 2206119051cbSMarius Strobl case MII_BMSR: 2207119051cbSMarius Strobl reg = CAS_PCS_STATUS; 2208119051cbSMarius Strobl break; 2209119051cbSMarius Strobl case MII_BMCR: 2210119051cbSMarius Strobl reg = CAS_PCS_CTRL; 2211119051cbSMarius Strobl if ((val & CAS_PCS_CTRL_RESET) == 0) 2212119051cbSMarius Strobl break; 2213119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_CTRL, val); 2214119051cbSMarius Strobl CAS_BARRIER(sc, CAS_PCS_CTRL, 4, 2215119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2216119051cbSMarius Strobl if (!cas_bitwait(sc, CAS_PCS_CTRL, 2217119051cbSMarius Strobl CAS_PCS_CTRL_RESET, 0)) 2218119051cbSMarius Strobl device_printf(sc->sc_dev, 2219119051cbSMarius Strobl "cannot reset PCS\n"); 2220119051cbSMarius Strobl /* FALLTHROUGH */ 2221119051cbSMarius Strobl case MII_ANAR: 2222119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_CONF, 0); 2223119051cbSMarius Strobl CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2224119051cbSMarius Strobl BUS_SPACE_BARRIER_WRITE); 2225119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_ANAR, val); 2226917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_ANAR, 4, 2227917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 2228119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_SERDES_CTRL, 2229119051cbSMarius Strobl CAS_PCS_SERDES_CTRL_ESD); 2230917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2231917825acSMarius Strobl BUS_SPACE_BARRIER_WRITE); 2232119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_PCS_CONF, 2233119051cbSMarius Strobl CAS_PCS_CONF_EN); 2234917825acSMarius Strobl CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2235917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2236119051cbSMarius Strobl return (0); 2237119051cbSMarius Strobl case MII_ANLPAR: 2238119051cbSMarius Strobl reg = CAS_PCS_ANLPAR; 2239119051cbSMarius Strobl break; 2240119051cbSMarius Strobl default: 2241119051cbSMarius Strobl device_printf(sc->sc_dev, 2242119051cbSMarius Strobl "%s: unhandled register %d\n", __func__, reg); 2243119051cbSMarius Strobl return (0); 2244119051cbSMarius Strobl } 2245119051cbSMarius Strobl CAS_WRITE_4(sc, reg, val); 2246917825acSMarius Strobl CAS_BARRIER(sc, reg, 4, 2247917825acSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2248119051cbSMarius Strobl return (0); 2249119051cbSMarius Strobl } 2250119051cbSMarius Strobl 2251119051cbSMarius Strobl /* Construct the frame command. */ 2252119051cbSMarius Strobl v = CAS_MIF_FRAME_WRITE | 2253119051cbSMarius Strobl (phy << CAS_MIF_FRAME_PHY_SHFT) | 2254119051cbSMarius Strobl (reg << CAS_MIF_FRAME_REG_SHFT) | 2255119051cbSMarius Strobl (val & CAS_MIF_FRAME_DATA); 2256119051cbSMarius Strobl 2257119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MIF_FRAME, v); 2258119051cbSMarius Strobl CAS_BARRIER(sc, CAS_MIF_FRAME, 4, 2259119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2260119051cbSMarius Strobl for (n = 0; n < 100; n++) { 2261119051cbSMarius Strobl DELAY(1); 2262119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MIF_FRAME); 2263119051cbSMarius Strobl if (v & CAS_MIF_FRAME_TA_LSB) 2264119051cbSMarius Strobl return (1); 2265119051cbSMarius Strobl } 2266119051cbSMarius Strobl 2267119051cbSMarius Strobl device_printf(sc->sc_dev, "%s: timed out\n", __func__); 2268119051cbSMarius Strobl return (0); 2269119051cbSMarius Strobl } 2270119051cbSMarius Strobl 2271119051cbSMarius Strobl static void 2272119051cbSMarius Strobl cas_mii_statchg(device_t dev) 2273119051cbSMarius Strobl { 2274119051cbSMarius Strobl struct cas_softc *sc; 227544e65355SJustin Hibbits if_t ifp; 2276119051cbSMarius Strobl int gigabit; 2277119051cbSMarius Strobl uint32_t rxcfg, txcfg, v; 2278119051cbSMarius Strobl 2279119051cbSMarius Strobl sc = device_get_softc(dev); 2280119051cbSMarius Strobl ifp = sc->sc_ifp; 2281119051cbSMarius Strobl 2282119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 2283119051cbSMarius Strobl 2284119051cbSMarius Strobl #ifdef CAS_DEBUG 228544e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_DEBUG) != 0) 22868e5d93dbSMarius Strobl device_printf(sc->sc_dev, "%s: status changen", __func__); 2287119051cbSMarius Strobl #endif 2288119051cbSMarius Strobl 2289119051cbSMarius Strobl if ((sc->sc_mii->mii_media_status & IFM_ACTIVE) != 0 && 2290119051cbSMarius Strobl IFM_SUBTYPE(sc->sc_mii->mii_media_active) != IFM_NONE) 2291119051cbSMarius Strobl sc->sc_flags |= CAS_LINK; 2292119051cbSMarius Strobl else 2293119051cbSMarius Strobl sc->sc_flags &= ~CAS_LINK; 2294119051cbSMarius Strobl 2295119051cbSMarius Strobl switch (IFM_SUBTYPE(sc->sc_mii->mii_media_active)) { 2296119051cbSMarius Strobl case IFM_1000_SX: 2297119051cbSMarius Strobl case IFM_1000_LX: 2298119051cbSMarius Strobl case IFM_1000_CX: 2299119051cbSMarius Strobl case IFM_1000_T: 2300119051cbSMarius Strobl gigabit = 1; 2301119051cbSMarius Strobl break; 2302119051cbSMarius Strobl default: 2303119051cbSMarius Strobl gigabit = 0; 2304119051cbSMarius Strobl } 2305119051cbSMarius Strobl 2306119051cbSMarius Strobl /* 2307119051cbSMarius Strobl * The configuration done here corresponds to the steps F) and 2308119051cbSMarius Strobl * G) and as far as enabling of RX and TX MAC goes also step H) 2309119051cbSMarius Strobl * of the initialization sequence outlined in section 11.2.1 of 2310119051cbSMarius Strobl * the Cassini+ ASIC Specification. 2311119051cbSMarius Strobl */ 2312119051cbSMarius Strobl 23135f8a9ae4SMarius Strobl rxcfg = sc->sc_mac_rxcfg; 23145f8a9ae4SMarius Strobl rxcfg &= ~CAS_MAC_RX_CONF_CARR; 2315119051cbSMarius Strobl txcfg = CAS_MAC_TX_CONF_EN_IPG0 | CAS_MAC_TX_CONF_NGU | 2316119051cbSMarius Strobl CAS_MAC_TX_CONF_NGUL; 2317119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) != 0) 2318119051cbSMarius Strobl txcfg |= CAS_MAC_TX_CONF_ICARR | CAS_MAC_TX_CONF_ICOLLIS; 2319119051cbSMarius Strobl else if (gigabit != 0) { 2320119051cbSMarius Strobl rxcfg |= CAS_MAC_RX_CONF_CARR; 2321119051cbSMarius Strobl txcfg |= CAS_MAC_TX_CONF_CARR; 2322119051cbSMarius Strobl } 23235f8a9ae4SMarius Strobl (void)cas_disable_tx(sc); 2324119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_TX_CONF, txcfg); 23255f8a9ae4SMarius Strobl (void)cas_disable_rx(sc); 2326119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CONF, rxcfg); 2327119051cbSMarius Strobl 2328119051cbSMarius Strobl v = CAS_READ_4(sc, CAS_MAC_CTRL_CONF) & 2329119051cbSMarius Strobl ~(CAS_MAC_CTRL_CONF_TXP | CAS_MAC_CTRL_CONF_RXP); 2330119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & 2331119051cbSMarius Strobl IFM_ETH_RXPAUSE) != 0) 2332119051cbSMarius Strobl v |= CAS_MAC_CTRL_CONF_RXP; 2333119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & 2334119051cbSMarius Strobl IFM_ETH_TXPAUSE) != 0) 2335119051cbSMarius Strobl v |= CAS_MAC_CTRL_CONF_TXP; 2336119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_CTRL_CONF, v); 2337119051cbSMarius Strobl 2338119051cbSMarius Strobl /* 2339119051cbSMarius Strobl * All supported chips have a bug causing incorrect checksum 2340119051cbSMarius Strobl * to be calculated when letting them strip the FCS in half- 2341119051cbSMarius Strobl * duplex mode. In theory we could disable FCS stripping and 2342119051cbSMarius Strobl * manually adjust the checksum accordingly. It seems to make 2343119051cbSMarius Strobl * more sense to optimze for the common case and just disable 2344119051cbSMarius Strobl * hardware checksumming in half-duplex mode though. 2345119051cbSMarius Strobl */ 2346119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) { 234744e65355SJustin Hibbits if_setcapenablebit(ifp, 0, IFCAP_HWCSUM); 234844e65355SJustin Hibbits if_sethwassist(ifp, 0); 2349119051cbSMarius Strobl } else if ((sc->sc_flags & CAS_NO_CSUM) == 0) { 235044e65355SJustin Hibbits if_setcapenable(ifp, if_getcapabilities(ifp)); 235144e65355SJustin Hibbits if_sethwassist(ifp, CAS_CSUM_FEATURES); 2352119051cbSMarius Strobl } 2353119051cbSMarius Strobl 2354119051cbSMarius Strobl if (sc->sc_variant == CAS_SATURN) { 2355119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) 2356119051cbSMarius Strobl /* silicon bug workaround */ 2357119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x41); 2358119051cbSMarius Strobl else 2359119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x7); 2360119051cbSMarius Strobl } 2361119051cbSMarius Strobl 2362119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0 && 2363119051cbSMarius Strobl gigabit != 0) 2364119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_SLOT_TIME, 2365119051cbSMarius Strobl CAS_MAC_SLOT_TIME_CARR); 2366119051cbSMarius Strobl else 2367119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_SLOT_TIME, 2368119051cbSMarius Strobl CAS_MAC_SLOT_TIME_NORM); 2369119051cbSMarius Strobl 2370119051cbSMarius Strobl /* XIF Configuration */ 2371119051cbSMarius Strobl v = CAS_MAC_XIF_CONF_TX_OE | CAS_MAC_XIF_CONF_LNKLED; 2372119051cbSMarius Strobl if ((sc->sc_flags & CAS_SERDES) == 0) { 2373119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) 2374119051cbSMarius Strobl v |= CAS_MAC_XIF_CONF_NOECHO; 2375119051cbSMarius Strobl v |= CAS_MAC_XIF_CONF_BUF_OE; 2376119051cbSMarius Strobl } 2377119051cbSMarius Strobl if (gigabit != 0) 2378119051cbSMarius Strobl v |= CAS_MAC_XIF_CONF_GMII; 2379119051cbSMarius Strobl if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) != 0) 2380119051cbSMarius Strobl v |= CAS_MAC_XIF_CONF_FDXLED; 2381119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_XIF_CONF, v); 2382119051cbSMarius Strobl 23835f8a9ae4SMarius Strobl sc->sc_mac_rxcfg = rxcfg; 238444e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0 && 2385119051cbSMarius Strobl (sc->sc_flags & CAS_LINK) != 0) { 2386119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_TX_CONF, 2387119051cbSMarius Strobl txcfg | CAS_MAC_TX_CONF_EN); 2388119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CONF, 2389119051cbSMarius Strobl rxcfg | CAS_MAC_RX_CONF_EN); 2390119051cbSMarius Strobl } 2391119051cbSMarius Strobl } 2392119051cbSMarius Strobl 2393119051cbSMarius Strobl static int 239444e65355SJustin Hibbits cas_mediachange(if_t ifp) 2395119051cbSMarius Strobl { 239644e65355SJustin Hibbits struct cas_softc *sc = if_getsoftc(ifp); 2397119051cbSMarius Strobl int error; 2398119051cbSMarius Strobl 2399119051cbSMarius Strobl /* XXX add support for serial media. */ 2400119051cbSMarius Strobl 2401119051cbSMarius Strobl CAS_LOCK(sc); 2402119051cbSMarius Strobl error = mii_mediachg(sc->sc_mii); 2403119051cbSMarius Strobl CAS_UNLOCK(sc); 2404119051cbSMarius Strobl return (error); 2405119051cbSMarius Strobl } 2406119051cbSMarius Strobl 2407119051cbSMarius Strobl static void 240844e65355SJustin Hibbits cas_mediastatus(if_t ifp, struct ifmediareq *ifmr) 2409119051cbSMarius Strobl { 241044e65355SJustin Hibbits struct cas_softc *sc = if_getsoftc(ifp); 2411119051cbSMarius Strobl 2412119051cbSMarius Strobl CAS_LOCK(sc); 241344e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_UP) == 0) { 2414119051cbSMarius Strobl CAS_UNLOCK(sc); 2415119051cbSMarius Strobl return; 2416119051cbSMarius Strobl } 2417119051cbSMarius Strobl 2418119051cbSMarius Strobl mii_pollstat(sc->sc_mii); 2419119051cbSMarius Strobl ifmr->ifm_active = sc->sc_mii->mii_media_active; 2420119051cbSMarius Strobl ifmr->ifm_status = sc->sc_mii->mii_media_status; 2421119051cbSMarius Strobl CAS_UNLOCK(sc); 2422119051cbSMarius Strobl } 2423119051cbSMarius Strobl 2424119051cbSMarius Strobl static int 242544e65355SJustin Hibbits cas_ioctl(if_t ifp, u_long cmd, caddr_t data) 2426119051cbSMarius Strobl { 242744e65355SJustin Hibbits struct cas_softc *sc = if_getsoftc(ifp); 2428119051cbSMarius Strobl struct ifreq *ifr = (struct ifreq *)data; 2429119051cbSMarius Strobl int error; 2430119051cbSMarius Strobl 2431119051cbSMarius Strobl error = 0; 2432119051cbSMarius Strobl switch (cmd) { 2433119051cbSMarius Strobl case SIOCSIFFLAGS: 2434119051cbSMarius Strobl CAS_LOCK(sc); 243544e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_UP) != 0) { 243644e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0 && 243744e65355SJustin Hibbits ((if_getflags(ifp) ^ sc->sc_ifflags) & 2438119051cbSMarius Strobl (IFF_ALLMULTI | IFF_PROMISC)) != 0) 2439119051cbSMarius Strobl cas_setladrf(sc); 2440119051cbSMarius Strobl else 2441119051cbSMarius Strobl cas_init_locked(sc); 244244e65355SJustin Hibbits } else if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0) 2443119051cbSMarius Strobl cas_stop(ifp); 244444e65355SJustin Hibbits sc->sc_ifflags = if_getflags(ifp); 2445119051cbSMarius Strobl CAS_UNLOCK(sc); 2446119051cbSMarius Strobl break; 2447119051cbSMarius Strobl case SIOCSIFCAP: 2448119051cbSMarius Strobl CAS_LOCK(sc); 2449119051cbSMarius Strobl if ((sc->sc_flags & CAS_NO_CSUM) != 0) { 2450119051cbSMarius Strobl error = EINVAL; 2451119051cbSMarius Strobl CAS_UNLOCK(sc); 2452119051cbSMarius Strobl break; 2453119051cbSMarius Strobl } 245444e65355SJustin Hibbits if_setcapenable(ifp, ifr->ifr_reqcap); 245544e65355SJustin Hibbits if ((if_getcapenable(ifp) & IFCAP_TXCSUM) != 0) 245644e65355SJustin Hibbits if_sethwassist(ifp, CAS_CSUM_FEATURES); 2457119051cbSMarius Strobl else 245844e65355SJustin Hibbits if_sethwassist(ifp, 0); 2459119051cbSMarius Strobl CAS_UNLOCK(sc); 2460119051cbSMarius Strobl break; 2461119051cbSMarius Strobl case SIOCADDMULTI: 2462119051cbSMarius Strobl case SIOCDELMULTI: 2463119051cbSMarius Strobl CAS_LOCK(sc); 246444e65355SJustin Hibbits if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0) 2465119051cbSMarius Strobl cas_setladrf(sc); 2466119051cbSMarius Strobl CAS_UNLOCK(sc); 2467119051cbSMarius Strobl break; 2468119051cbSMarius Strobl case SIOCSIFMTU: 2469119051cbSMarius Strobl if ((ifr->ifr_mtu < ETHERMIN) || 2470119051cbSMarius Strobl (ifr->ifr_mtu > ETHERMTU_JUMBO)) 2471119051cbSMarius Strobl error = EINVAL; 2472119051cbSMarius Strobl else 247344e65355SJustin Hibbits if_setmtu(ifp, ifr->ifr_mtu); 2474119051cbSMarius Strobl break; 2475119051cbSMarius Strobl case SIOCGIFMEDIA: 2476119051cbSMarius Strobl case SIOCSIFMEDIA: 2477119051cbSMarius Strobl error = ifmedia_ioctl(ifp, ifr, &sc->sc_mii->mii_media, cmd); 2478119051cbSMarius Strobl break; 2479119051cbSMarius Strobl default: 2480119051cbSMarius Strobl error = ether_ioctl(ifp, cmd, data); 2481119051cbSMarius Strobl break; 2482119051cbSMarius Strobl } 2483119051cbSMarius Strobl 2484119051cbSMarius Strobl return (error); 2485119051cbSMarius Strobl } 2486119051cbSMarius Strobl 248795f4453eSGleb Smirnoff static u_int 248895f4453eSGleb Smirnoff cas_hash_maddr(void *arg, struct sockaddr_dl *sdl, u_int cnt) 248995f4453eSGleb Smirnoff { 249095f4453eSGleb Smirnoff uint32_t crc, *hash = arg; 249195f4453eSGleb Smirnoff 249295f4453eSGleb Smirnoff crc = ether_crc32_le(LLADDR(sdl), ETHER_ADDR_LEN); 249395f4453eSGleb Smirnoff /* We just want the 8 most significant bits. */ 249495f4453eSGleb Smirnoff crc >>= 24; 249595f4453eSGleb Smirnoff /* Set the corresponding bit in the filter. */ 249695f4453eSGleb Smirnoff hash[crc >> 4] |= 1 << (15 - (crc & 15)); 249795f4453eSGleb Smirnoff 249895f4453eSGleb Smirnoff return (1); 249995f4453eSGleb Smirnoff } 250095f4453eSGleb Smirnoff 2501119051cbSMarius Strobl static void 2502119051cbSMarius Strobl cas_setladrf(struct cas_softc *sc) 2503119051cbSMarius Strobl { 250444e65355SJustin Hibbits if_t ifp = sc->sc_ifp; 2505119051cbSMarius Strobl int i; 2506119051cbSMarius Strobl uint32_t hash[16]; 250795f4453eSGleb Smirnoff uint32_t v; 2508119051cbSMarius Strobl 2509119051cbSMarius Strobl CAS_LOCK_ASSERT(sc, MA_OWNED); 2510119051cbSMarius Strobl 2511119051cbSMarius Strobl /* 25125f8a9ae4SMarius Strobl * Turn off the RX MAC and the hash filter as required by the Sun 25135f8a9ae4SMarius Strobl * Cassini programming restrictions. 2514119051cbSMarius Strobl */ 25155f8a9ae4SMarius Strobl v = sc->sc_mac_rxcfg & ~(CAS_MAC_RX_CONF_HFILTER | 25165f8a9ae4SMarius Strobl CAS_MAC_RX_CONF_EN); 2517119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CONF, v); 2518119051cbSMarius Strobl CAS_BARRIER(sc, CAS_MAC_RX_CONF, 4, 2519119051cbSMarius Strobl BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 25205f8a9ae4SMarius Strobl if (!cas_bitwait(sc, CAS_MAC_RX_CONF, CAS_MAC_RX_CONF_HFILTER | 25215f8a9ae4SMarius Strobl CAS_MAC_RX_CONF_EN, 0)) 25225f8a9ae4SMarius Strobl device_printf(sc->sc_dev, 25235f8a9ae4SMarius Strobl "cannot disable RX MAC or hash filter\n"); 2524119051cbSMarius Strobl 25255f8a9ae4SMarius Strobl v &= ~(CAS_MAC_RX_CONF_PROMISC | CAS_MAC_RX_CONF_PGRP); 252644e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_PROMISC) != 0) { 2527119051cbSMarius Strobl v |= CAS_MAC_RX_CONF_PROMISC; 2528119051cbSMarius Strobl goto chipit; 2529119051cbSMarius Strobl } 253044e65355SJustin Hibbits if ((if_getflags(ifp) & IFF_ALLMULTI) != 0) { 2531119051cbSMarius Strobl v |= CAS_MAC_RX_CONF_PGRP; 2532119051cbSMarius Strobl goto chipit; 2533119051cbSMarius Strobl } 2534119051cbSMarius Strobl 2535119051cbSMarius Strobl /* 2536119051cbSMarius Strobl * Set up multicast address filter by passing all multicast 2537119051cbSMarius Strobl * addresses through a crc generator, and then using the high 2538119051cbSMarius Strobl * order 8 bits as an index into the 256 bit logical address 2539119051cbSMarius Strobl * filter. The high order 4 bits selects the word, while the 2540119051cbSMarius Strobl * other 4 bits select the bit within the word (where bit 0 2541119051cbSMarius Strobl * is the MSB). 2542119051cbSMarius Strobl */ 2543119051cbSMarius Strobl 2544119051cbSMarius Strobl memset(hash, 0, sizeof(hash)); 254595f4453eSGleb Smirnoff if_foreach_llmaddr(ifp, cas_hash_maddr, &hash); 2546119051cbSMarius Strobl 2547119051cbSMarius Strobl v |= CAS_MAC_RX_CONF_HFILTER; 2548119051cbSMarius Strobl 2549119051cbSMarius Strobl /* Now load the hash table into the chip (if we are using it). */ 2550119051cbSMarius Strobl for (i = 0; i < 16; i++) 2551119051cbSMarius Strobl CAS_WRITE_4(sc, 2552119051cbSMarius Strobl CAS_MAC_HASH0 + i * (CAS_MAC_HASH1 - CAS_MAC_HASH0), 2553119051cbSMarius Strobl hash[i]); 2554119051cbSMarius Strobl 2555119051cbSMarius Strobl chipit: 25565f8a9ae4SMarius Strobl sc->sc_mac_rxcfg = v; 25575f8a9ae4SMarius Strobl CAS_WRITE_4(sc, CAS_MAC_RX_CONF, v | CAS_MAC_RX_CONF_EN); 2558119051cbSMarius Strobl } 2559119051cbSMarius Strobl 2560119051cbSMarius Strobl static int cas_pci_attach(device_t dev); 2561119051cbSMarius Strobl static int cas_pci_detach(device_t dev); 2562119051cbSMarius Strobl static int cas_pci_probe(device_t dev); 2563119051cbSMarius Strobl static int cas_pci_resume(device_t dev); 2564119051cbSMarius Strobl static int cas_pci_suspend(device_t dev); 2565119051cbSMarius Strobl 2566119051cbSMarius Strobl static device_method_t cas_pci_methods[] = { 2567119051cbSMarius Strobl /* Device interface */ 2568119051cbSMarius Strobl DEVMETHOD(device_probe, cas_pci_probe), 2569119051cbSMarius Strobl DEVMETHOD(device_attach, cas_pci_attach), 2570119051cbSMarius Strobl DEVMETHOD(device_detach, cas_pci_detach), 2571119051cbSMarius Strobl DEVMETHOD(device_suspend, cas_pci_suspend), 2572119051cbSMarius Strobl DEVMETHOD(device_resume, cas_pci_resume), 2573119051cbSMarius Strobl /* Use the suspend handler here, it is all that is required. */ 2574119051cbSMarius Strobl DEVMETHOD(device_shutdown, cas_pci_suspend), 2575119051cbSMarius Strobl 2576119051cbSMarius Strobl /* MII interface */ 2577119051cbSMarius Strobl DEVMETHOD(miibus_readreg, cas_mii_readreg), 2578119051cbSMarius Strobl DEVMETHOD(miibus_writereg, cas_mii_writereg), 2579119051cbSMarius Strobl DEVMETHOD(miibus_statchg, cas_mii_statchg), 2580119051cbSMarius Strobl 25814b7ec270SMarius Strobl DEVMETHOD_END 2582119051cbSMarius Strobl }; 2583119051cbSMarius Strobl 2584119051cbSMarius Strobl static driver_t cas_pci_driver = { 2585119051cbSMarius Strobl "cas", 2586119051cbSMarius Strobl cas_pci_methods, 2587119051cbSMarius Strobl sizeof(struct cas_softc) 2588119051cbSMarius Strobl }; 2589119051cbSMarius Strobl 2590119051cbSMarius Strobl static const struct cas_pci_dev { 2591119051cbSMarius Strobl uint32_t cpd_devid; 2592119051cbSMarius Strobl uint8_t cpd_revid; 2593119051cbSMarius Strobl int cpd_variant; 2594119051cbSMarius Strobl const char *cpd_desc; 259529658c96SDimitry Andric } cas_pci_devlist[] = { 2596119051cbSMarius Strobl { 0x0035100b, 0x0, CAS_SATURN, "NS DP83065 Saturn Gigabit Ethernet" }, 2597119051cbSMarius Strobl { 0xabba108e, 0x10, CAS_CASPLUS, "Sun Cassini+ Gigabit Ethernet" }, 2598119051cbSMarius Strobl { 0xabba108e, 0x0, CAS_CAS, "Sun Cassini Gigabit Ethernet" }, 2599119051cbSMarius Strobl { 0, 0, 0, NULL } 2600119051cbSMarius Strobl }; 2601119051cbSMarius Strobl 260238729079SJohn Baldwin DRIVER_MODULE(cas, pci, cas_pci_driver, 0, 0); 260363c017daSWarner Losh MODULE_PNP_INFO("W32:vendor/device", pci, cas, cas_pci_devlist, 2604329e817fSWarner Losh nitems(cas_pci_devlist) - 1); 26053e38757dSJohn Baldwin DRIVER_MODULE(miibus, cas, miibus_driver, 0, 0); 260663c017daSWarner Losh MODULE_DEPEND(cas, pci, 1, 1, 1); 260763c017daSWarner Losh 2608119051cbSMarius Strobl static int 2609119051cbSMarius Strobl cas_pci_probe(device_t dev) 2610119051cbSMarius Strobl { 2611119051cbSMarius Strobl int i; 2612119051cbSMarius Strobl 2613119051cbSMarius Strobl for (i = 0; cas_pci_devlist[i].cpd_desc != NULL; i++) { 2614119051cbSMarius Strobl if (pci_get_devid(dev) == cas_pci_devlist[i].cpd_devid && 2615119051cbSMarius Strobl pci_get_revid(dev) >= cas_pci_devlist[i].cpd_revid) { 2616119051cbSMarius Strobl device_set_desc(dev, cas_pci_devlist[i].cpd_desc); 2617119051cbSMarius Strobl return (BUS_PROBE_DEFAULT); 2618119051cbSMarius Strobl } 2619119051cbSMarius Strobl } 2620119051cbSMarius Strobl 2621119051cbSMarius Strobl return (ENXIO); 2622119051cbSMarius Strobl } 2623119051cbSMarius Strobl 2624119051cbSMarius Strobl static struct resource_spec cas_pci_res_spec[] = { 2625119051cbSMarius Strobl { SYS_RES_IRQ, 0, RF_SHAREABLE | RF_ACTIVE }, /* CAS_RES_INTR */ 2626119051cbSMarius Strobl { SYS_RES_MEMORY, PCIR_BAR(0), RF_ACTIVE }, /* CAS_RES_MEM */ 2627119051cbSMarius Strobl { -1, 0 } 2628119051cbSMarius Strobl }; 2629119051cbSMarius Strobl 2630917825acSMarius Strobl #define CAS_LOCAL_MAC_ADDRESS "local-mac-address" 2631917825acSMarius Strobl #define CAS_PHY_INTERFACE "phy-interface" 2632917825acSMarius Strobl #define CAS_PHY_TYPE "phy-type" 2633917825acSMarius Strobl #define CAS_PHY_TYPE_PCS "pcs" 2634917825acSMarius Strobl 2635119051cbSMarius Strobl static int 2636119051cbSMarius Strobl cas_pci_attach(device_t dev) 2637119051cbSMarius Strobl { 2638917825acSMarius Strobl char buf[sizeof(CAS_LOCAL_MAC_ADDRESS)]; 2639119051cbSMarius Strobl struct cas_softc *sc; 2640119051cbSMarius Strobl int i; 264158aa35d4SWarner Losh #if !defined(__powerpc__) 2642119051cbSMarius Strobl u_char enaddr[4][ETHER_ADDR_LEN]; 2643917825acSMarius Strobl u_int j, k, lma, pcs[4], phy; 2644119051cbSMarius Strobl #endif 2645119051cbSMarius Strobl 2646119051cbSMarius Strobl sc = device_get_softc(dev); 2647119051cbSMarius Strobl sc->sc_variant = CAS_UNKNOWN; 2648119051cbSMarius Strobl for (i = 0; cas_pci_devlist[i].cpd_desc != NULL; i++) { 2649119051cbSMarius Strobl if (pci_get_devid(dev) == cas_pci_devlist[i].cpd_devid && 2650119051cbSMarius Strobl pci_get_revid(dev) >= cas_pci_devlist[i].cpd_revid) { 2651119051cbSMarius Strobl sc->sc_variant = cas_pci_devlist[i].cpd_variant; 2652119051cbSMarius Strobl break; 2653119051cbSMarius Strobl } 2654119051cbSMarius Strobl } 2655119051cbSMarius Strobl if (sc->sc_variant == CAS_UNKNOWN) { 2656119051cbSMarius Strobl device_printf(dev, "unknown adaptor\n"); 2657119051cbSMarius Strobl return (ENXIO); 2658119051cbSMarius Strobl } 2659119051cbSMarius Strobl 2660a587c657SMarius Strobl /* PCI configuration */ 2661a587c657SMarius Strobl pci_write_config(dev, PCIR_COMMAND, 2662a587c657SMarius Strobl pci_read_config(dev, PCIR_COMMAND, 2) | PCIM_CMD_BUSMASTEREN | 2663a587c657SMarius Strobl PCIM_CMD_MWRICEN | PCIM_CMD_PERRESPEN | PCIM_CMD_SERRESPEN, 2); 2664119051cbSMarius Strobl 2665119051cbSMarius Strobl sc->sc_dev = dev; 2666119051cbSMarius Strobl if (sc->sc_variant == CAS_CAS && pci_get_devid(dev) < 0x02) 2667119051cbSMarius Strobl /* Hardware checksumming may hang TX. */ 2668119051cbSMarius Strobl sc->sc_flags |= CAS_NO_CSUM; 2669119051cbSMarius Strobl if (sc->sc_variant == CAS_CASPLUS || sc->sc_variant == CAS_SATURN) 2670119051cbSMarius Strobl sc->sc_flags |= CAS_REG_PLUS; 2671119051cbSMarius Strobl if (sc->sc_variant == CAS_CAS || 2672119051cbSMarius Strobl (sc->sc_variant == CAS_CASPLUS && pci_get_revid(dev) < 0x11)) 2673119051cbSMarius Strobl sc->sc_flags |= CAS_TABORT; 2674119051cbSMarius Strobl if (bootverbose) 2675119051cbSMarius Strobl device_printf(dev, "flags=0x%x\n", sc->sc_flags); 2676119051cbSMarius Strobl 2677119051cbSMarius Strobl if (bus_alloc_resources(dev, cas_pci_res_spec, sc->sc_res)) { 2678119051cbSMarius Strobl device_printf(dev, "failed to allocate resources\n"); 2679119051cbSMarius Strobl bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2680119051cbSMarius Strobl return (ENXIO); 2681119051cbSMarius Strobl } 2682119051cbSMarius Strobl 2683119051cbSMarius Strobl CAS_LOCK_INIT(sc, device_get_nameunit(dev)); 2684119051cbSMarius Strobl 268558aa35d4SWarner Losh #if defined(__powerpc__) 2686119051cbSMarius Strobl OF_getetheraddr(dev, sc->sc_enaddr); 2687917825acSMarius Strobl if (OF_getprop(ofw_bus_get_node(dev), CAS_PHY_INTERFACE, buf, 2688917825acSMarius Strobl sizeof(buf)) > 0 || OF_getprop(ofw_bus_get_node(dev), 2689917825acSMarius Strobl CAS_PHY_TYPE, buf, sizeof(buf)) > 0) { 2690917825acSMarius Strobl buf[sizeof(buf) - 1] = '\0'; 2691917825acSMarius Strobl if (strcmp(buf, CAS_PHY_TYPE_PCS) == 0) 2692917825acSMarius Strobl sc->sc_flags |= CAS_SERDES; 2693917825acSMarius Strobl } 2694119051cbSMarius Strobl #else 2695119051cbSMarius Strobl /* 2696917825acSMarius Strobl * Dig out VPD (vital product data) and read the MAC address as well 2697917825acSMarius Strobl * as the PHY type. The VPD resides in the PCI Expansion ROM (PCI 2698917825acSMarius Strobl * FCode) and can't be accessed via the PCI capability pointer. 2699917825acSMarius Strobl * SUNW,pci-ce and SUNW,pci-qge use the Enhanced VPD format described 2700917825acSMarius Strobl * in the free US Patent 7149820. 2701119051cbSMarius Strobl */ 2702119051cbSMarius Strobl 2703119051cbSMarius Strobl #define PCI_ROMHDR_SIZE 0x1c 2704119051cbSMarius Strobl #define PCI_ROMHDR_SIG 0x00 2705119051cbSMarius Strobl #define PCI_ROMHDR_SIG_MAGIC 0xaa55 /* little endian */ 2706119051cbSMarius Strobl #define PCI_ROMHDR_PTR_DATA 0x18 2707119051cbSMarius Strobl #define PCI_ROM_SIZE 0x18 2708119051cbSMarius Strobl #define PCI_ROM_SIG 0x00 2709119051cbSMarius Strobl #define PCI_ROM_SIG_MAGIC 0x52494350 /* "PCIR", endian */ 2710119051cbSMarius Strobl /* reversed */ 2711119051cbSMarius Strobl #define PCI_ROM_VENDOR 0x04 2712119051cbSMarius Strobl #define PCI_ROM_DEVICE 0x06 2713119051cbSMarius Strobl #define PCI_ROM_PTR_VPD 0x08 2714119051cbSMarius Strobl #define PCI_VPDRES_BYTE0 0x00 2715119051cbSMarius Strobl #define PCI_VPDRES_ISLARGE(x) ((x) & 0x80) 2716119051cbSMarius Strobl #define PCI_VPDRES_LARGE_NAME(x) ((x) & 0x7f) 2717119051cbSMarius Strobl #define PCI_VPDRES_LARGE_LEN_LSB 0x01 2718119051cbSMarius Strobl #define PCI_VPDRES_LARGE_LEN_MSB 0x02 2719119051cbSMarius Strobl #define PCI_VPDRES_LARGE_SIZE 0x03 2720119051cbSMarius Strobl #define PCI_VPDRES_TYPE_ID_STRING 0x02 /* large */ 2721119051cbSMarius Strobl #define PCI_VPDRES_TYPE_VPD 0x10 /* large */ 2722119051cbSMarius Strobl #define PCI_VPD_KEY0 0x00 2723119051cbSMarius Strobl #define PCI_VPD_KEY1 0x01 2724119051cbSMarius Strobl #define PCI_VPD_LEN 0x02 2725119051cbSMarius Strobl #define PCI_VPD_SIZE 0x03 2726119051cbSMarius Strobl 2727119051cbSMarius Strobl #define CAS_ROM_READ_1(sc, offs) \ 2728119051cbSMarius Strobl CAS_READ_1((sc), CAS_PCI_ROM_OFFSET + (offs)) 2729119051cbSMarius Strobl #define CAS_ROM_READ_2(sc, offs) \ 2730119051cbSMarius Strobl CAS_READ_2((sc), CAS_PCI_ROM_OFFSET + (offs)) 2731119051cbSMarius Strobl #define CAS_ROM_READ_4(sc, offs) \ 2732119051cbSMarius Strobl CAS_READ_4((sc), CAS_PCI_ROM_OFFSET + (offs)) 2733119051cbSMarius Strobl 2734917825acSMarius Strobl lma = phy = 0; 2735917825acSMarius Strobl memset(enaddr, 0, sizeof(enaddr)); 2736917825acSMarius Strobl memset(pcs, 0, sizeof(pcs)); 2737917825acSMarius Strobl 2738119051cbSMarius Strobl /* Enable PCI Expansion ROM access. */ 2739119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_BIM_LDEV_OEN, 2740119051cbSMarius Strobl CAS_BIM_LDEV_OEN_PAD | CAS_BIM_LDEV_OEN_PROM); 2741119051cbSMarius Strobl 2742119051cbSMarius Strobl /* Read PCI Expansion ROM header. */ 2743119051cbSMarius Strobl if (CAS_ROM_READ_2(sc, PCI_ROMHDR_SIG) != PCI_ROMHDR_SIG_MAGIC || 2744119051cbSMarius Strobl (i = CAS_ROM_READ_2(sc, PCI_ROMHDR_PTR_DATA)) < 2745119051cbSMarius Strobl PCI_ROMHDR_SIZE) { 2746119051cbSMarius Strobl device_printf(dev, "unexpected PCI Expansion ROM header\n"); 2747119051cbSMarius Strobl goto fail_prom; 2748119051cbSMarius Strobl } 2749119051cbSMarius Strobl 2750119051cbSMarius Strobl /* Read PCI Expansion ROM data. */ 2751119051cbSMarius Strobl if (CAS_ROM_READ_4(sc, i + PCI_ROM_SIG) != PCI_ROM_SIG_MAGIC || 2752119051cbSMarius Strobl CAS_ROM_READ_2(sc, i + PCI_ROM_VENDOR) != pci_get_vendor(dev) || 2753119051cbSMarius Strobl CAS_ROM_READ_2(sc, i + PCI_ROM_DEVICE) != pci_get_device(dev) || 2754119051cbSMarius Strobl (j = CAS_ROM_READ_2(sc, i + PCI_ROM_PTR_VPD)) < 2755119051cbSMarius Strobl i + PCI_ROM_SIZE) { 2756119051cbSMarius Strobl device_printf(dev, "unexpected PCI Expansion ROM data\n"); 2757119051cbSMarius Strobl goto fail_prom; 2758119051cbSMarius Strobl } 2759119051cbSMarius Strobl 2760119051cbSMarius Strobl /* Read PCI VPD. */ 2761119051cbSMarius Strobl next: 2762119051cbSMarius Strobl if (PCI_VPDRES_ISLARGE(CAS_ROM_READ_1(sc, 2763119051cbSMarius Strobl j + PCI_VPDRES_BYTE0)) == 0) { 2764119051cbSMarius Strobl device_printf(dev, "no large PCI VPD\n"); 2765119051cbSMarius Strobl goto fail_prom; 2766119051cbSMarius Strobl } 2767119051cbSMarius Strobl 2768119051cbSMarius Strobl i = (CAS_ROM_READ_1(sc, j + PCI_VPDRES_LARGE_LEN_MSB) << 8) | 2769119051cbSMarius Strobl CAS_ROM_READ_1(sc, j + PCI_VPDRES_LARGE_LEN_LSB); 2770119051cbSMarius Strobl switch (PCI_VPDRES_LARGE_NAME(CAS_ROM_READ_1(sc, 2771119051cbSMarius Strobl j + PCI_VPDRES_BYTE0))) { 2772119051cbSMarius Strobl case PCI_VPDRES_TYPE_ID_STRING: 2773119051cbSMarius Strobl /* Skip identifier string. */ 2774119051cbSMarius Strobl j += PCI_VPDRES_LARGE_SIZE + i; 2775119051cbSMarius Strobl goto next; 2776119051cbSMarius Strobl case PCI_VPDRES_TYPE_VPD: 2777119051cbSMarius Strobl for (j += PCI_VPDRES_LARGE_SIZE; i > 0; 2778119051cbSMarius Strobl i -= PCI_VPD_SIZE + CAS_ROM_READ_1(sc, j + PCI_VPD_LEN), 2779119051cbSMarius Strobl j += PCI_VPD_SIZE + CAS_ROM_READ_1(sc, j + PCI_VPD_LEN)) { 2780119051cbSMarius Strobl if (CAS_ROM_READ_1(sc, j + PCI_VPD_KEY0) != 'Z') 2781119051cbSMarius Strobl /* no Enhanced VPD */ 2782119051cbSMarius Strobl continue; 2783119051cbSMarius Strobl if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE) != 'I') 2784119051cbSMarius Strobl /* no instance property */ 2785119051cbSMarius Strobl continue; 2786917825acSMarius Strobl if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE + 3) == 'B') { 2787917825acSMarius Strobl /* byte array */ 2788917825acSMarius Strobl if (CAS_ROM_READ_1(sc, 2789917825acSMarius Strobl j + PCI_VPD_SIZE + 4) != ETHER_ADDR_LEN) 2790119051cbSMarius Strobl continue; 2791119051cbSMarius Strobl bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2792119051cbSMarius Strobl CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 5, 2793917825acSMarius Strobl buf, sizeof(buf)); 2794917825acSMarius Strobl buf[sizeof(buf) - 1] = '\0'; 2795917825acSMarius Strobl if (strcmp(buf, CAS_LOCAL_MAC_ADDRESS) != 0) 2796119051cbSMarius Strobl continue; 2797119051cbSMarius Strobl bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2798917825acSMarius Strobl CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 2799917825acSMarius Strobl 5 + sizeof(CAS_LOCAL_MAC_ADDRESS), 2800917825acSMarius Strobl enaddr[lma], sizeof(enaddr[lma])); 2801917825acSMarius Strobl lma++; 2802917825acSMarius Strobl if (lma == 4 && phy == 4) 2803119051cbSMarius Strobl break; 2804917825acSMarius Strobl } else if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE + 3) == 2805917825acSMarius Strobl 'S') { 2806917825acSMarius Strobl /* string */ 2807917825acSMarius Strobl if (CAS_ROM_READ_1(sc, 2808917825acSMarius Strobl j + PCI_VPD_SIZE + 4) != 2809917825acSMarius Strobl sizeof(CAS_PHY_TYPE_PCS)) 2810917825acSMarius Strobl continue; 2811917825acSMarius Strobl bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2812917825acSMarius Strobl CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 5, 2813917825acSMarius Strobl buf, sizeof(buf)); 2814917825acSMarius Strobl buf[sizeof(buf) - 1] = '\0'; 2815917825acSMarius Strobl if (strcmp(buf, CAS_PHY_INTERFACE) == 0) 2816917825acSMarius Strobl k = sizeof(CAS_PHY_INTERFACE); 2817917825acSMarius Strobl else if (strcmp(buf, CAS_PHY_TYPE) == 0) 2818917825acSMarius Strobl k = sizeof(CAS_PHY_TYPE); 2819917825acSMarius Strobl else 2820917825acSMarius Strobl continue; 2821917825acSMarius Strobl bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2822917825acSMarius Strobl CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 2823917825acSMarius Strobl 5 + k, buf, sizeof(buf)); 2824917825acSMarius Strobl buf[sizeof(buf) - 1] = '\0'; 2825917825acSMarius Strobl if (strcmp(buf, CAS_PHY_TYPE_PCS) == 0) 2826917825acSMarius Strobl pcs[phy] = 1; 2827917825acSMarius Strobl phy++; 2828917825acSMarius Strobl if (lma == 4 && phy == 4) 2829917825acSMarius Strobl break; 2830917825acSMarius Strobl } 2831119051cbSMarius Strobl } 2832119051cbSMarius Strobl break; 2833119051cbSMarius Strobl default: 2834119051cbSMarius Strobl device_printf(dev, "unexpected PCI VPD\n"); 2835119051cbSMarius Strobl goto fail_prom; 2836119051cbSMarius Strobl } 2837119051cbSMarius Strobl 2838119051cbSMarius Strobl fail_prom: 2839119051cbSMarius Strobl CAS_WRITE_4(sc, CAS_BIM_LDEV_OEN, 0); 2840119051cbSMarius Strobl 2841917825acSMarius Strobl if (lma == 0) { 2842119051cbSMarius Strobl device_printf(dev, "could not determine Ethernet address\n"); 2843119051cbSMarius Strobl goto fail; 2844119051cbSMarius Strobl } 2845119051cbSMarius Strobl i = 0; 28467c2fdcacSMarius Strobl if (lma > 1 && pci_get_slot(dev) < nitems(enaddr)) 2847119051cbSMarius Strobl i = pci_get_slot(dev); 2848119051cbSMarius Strobl memcpy(sc->sc_enaddr, enaddr[i], ETHER_ADDR_LEN); 2849917825acSMarius Strobl 2850917825acSMarius Strobl if (phy == 0) { 2851917825acSMarius Strobl device_printf(dev, "could not determine PHY type\n"); 2852917825acSMarius Strobl goto fail; 2853917825acSMarius Strobl } 2854917825acSMarius Strobl i = 0; 28557c2fdcacSMarius Strobl if (phy > 1 && pci_get_slot(dev) < nitems(pcs)) 2856917825acSMarius Strobl i = pci_get_slot(dev); 2857917825acSMarius Strobl if (pcs[i] != 0) 2858917825acSMarius Strobl sc->sc_flags |= CAS_SERDES; 2859119051cbSMarius Strobl #endif 2860119051cbSMarius Strobl 2861119051cbSMarius Strobl if (cas_attach(sc) != 0) { 2862119051cbSMarius Strobl device_printf(dev, "could not be attached\n"); 2863119051cbSMarius Strobl goto fail; 2864119051cbSMarius Strobl } 2865119051cbSMarius Strobl 2866119051cbSMarius Strobl if (bus_setup_intr(dev, sc->sc_res[CAS_RES_INTR], INTR_TYPE_NET | 286773591c46SMarius Strobl INTR_MPSAFE, cas_intr, NULL, sc, &sc->sc_ih) != 0) { 2868119051cbSMarius Strobl device_printf(dev, "failed to set up interrupt\n"); 2869119051cbSMarius Strobl cas_detach(sc); 2870119051cbSMarius Strobl goto fail; 2871119051cbSMarius Strobl } 2872119051cbSMarius Strobl return (0); 2873119051cbSMarius Strobl 2874119051cbSMarius Strobl fail: 2875119051cbSMarius Strobl CAS_LOCK_DESTROY(sc); 2876119051cbSMarius Strobl bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2877119051cbSMarius Strobl return (ENXIO); 2878119051cbSMarius Strobl } 2879119051cbSMarius Strobl 2880119051cbSMarius Strobl static int 2881119051cbSMarius Strobl cas_pci_detach(device_t dev) 2882119051cbSMarius Strobl { 2883119051cbSMarius Strobl struct cas_softc *sc; 2884119051cbSMarius Strobl 2885119051cbSMarius Strobl sc = device_get_softc(dev); 2886119051cbSMarius Strobl bus_teardown_intr(dev, sc->sc_res[CAS_RES_INTR], sc->sc_ih); 2887119051cbSMarius Strobl cas_detach(sc); 2888119051cbSMarius Strobl CAS_LOCK_DESTROY(sc); 2889119051cbSMarius Strobl bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2890119051cbSMarius Strobl return (0); 2891119051cbSMarius Strobl } 2892119051cbSMarius Strobl 2893119051cbSMarius Strobl static int 2894119051cbSMarius Strobl cas_pci_suspend(device_t dev) 2895119051cbSMarius Strobl { 2896119051cbSMarius Strobl 2897119051cbSMarius Strobl cas_suspend(device_get_softc(dev)); 2898119051cbSMarius Strobl return (0); 2899119051cbSMarius Strobl } 2900119051cbSMarius Strobl 2901119051cbSMarius Strobl static int 2902119051cbSMarius Strobl cas_pci_resume(device_t dev) 2903119051cbSMarius Strobl { 2904119051cbSMarius Strobl 2905119051cbSMarius Strobl cas_resume(device_get_softc(dev)); 2906119051cbSMarius Strobl return (0); 2907119051cbSMarius Strobl } 2908