1194246Smarius/*- 2194246Smarius * Copyright (C) 2001 Eduardo Horvath. 3194246Smarius * Copyright (c) 2001-2003 Thomas Moestl 4194246Smarius * Copyright (c) 2007-2009 Marius Strobl <marius@FreeBSD.org> 5194246Smarius * All rights reserved. 6194246Smarius * 7194246Smarius * Redistribution and use in source and binary forms, with or without 8194246Smarius * modification, are permitted provided that the following conditions 9194246Smarius * are met: 10194246Smarius * 1. Redistributions of source code must retain the above copyright 11194246Smarius * notice, this list of conditions and the following disclaimer. 12194246Smarius * 2. Redistributions in binary form must reproduce the above copyright 13194246Smarius * notice, this list of conditions and the following disclaimer in the 14194246Smarius * documentation and/or other materials provided with the distribution. 15194246Smarius * 16194246Smarius * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND 17194246Smarius * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18194246Smarius * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19194246Smarius * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE 20194246Smarius * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21194246Smarius * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22194246Smarius * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23194246Smarius * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24194246Smarius * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25194246Smarius * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26194246Smarius * SUCH DAMAGE. 27194246Smarius * 28194246Smarius * from: NetBSD: gem.c,v 1.21 2002/06/01 23:50:58 lukem Exp 29194246Smarius * from: FreeBSD: if_gem.c 182060 2008-08-23 15:03:26Z marius 30194246Smarius */ 31194246Smarius 32194246Smarius#include <sys/cdefs.h> 33194246Smarius__FBSDID("$FreeBSD$"); 34194246Smarius 35194246Smarius/* 36194246Smarius * driver for Sun Cassini/Cassini+ and National Semiconductor DP83065 37194246Smarius * Saturn Gigabit Ethernet controllers 38194246Smarius */ 39194246Smarius 40194246Smarius#if 0 41194246Smarius#define CAS_DEBUG 42194246Smarius#endif 43194246Smarius 44194246Smarius#include <sys/param.h> 45194246Smarius#include <sys/systm.h> 46194246Smarius#include <sys/bus.h> 47194246Smarius#include <sys/callout.h> 48194246Smarius#include <sys/endian.h> 49194246Smarius#include <sys/mbuf.h> 50194246Smarius#include <sys/malloc.h> 51194246Smarius#include <sys/kernel.h> 52194246Smarius#include <sys/lock.h> 53194246Smarius#include <sys/module.h> 54194246Smarius#include <sys/mutex.h> 55194246Smarius#include <sys/refcount.h> 56194246Smarius#include <sys/resource.h> 57194246Smarius#include <sys/rman.h> 58194246Smarius#include <sys/socket.h> 59194246Smarius#include <sys/sockio.h> 60194904Smarius#include <sys/taskqueue.h> 61194246Smarius 62194246Smarius#include <net/bpf.h> 63194246Smarius#include <net/ethernet.h> 64194246Smarius#include <net/if.h> 65194246Smarius#include <net/if_arp.h> 66194246Smarius#include <net/if_dl.h> 67194246Smarius#include <net/if_media.h> 68194246Smarius#include <net/if_types.h> 69194246Smarius#include <net/if_vlan_var.h> 70194246Smarius 71194246Smarius#include <netinet/in.h> 72194246Smarius#include <netinet/in_systm.h> 73194246Smarius#include <netinet/ip.h> 74194246Smarius#include <netinet/tcp.h> 75194246Smarius#include <netinet/udp.h> 76194246Smarius 77194246Smarius#include <machine/bus.h> 78194246Smarius#if defined(__powerpc__) || defined(__sparc64__) 79207585Smarius#include <dev/ofw/ofw_bus.h> 80194246Smarius#include <dev/ofw/openfirm.h> 81194246Smarius#include <machine/ofw_machdep.h> 82194246Smarius#endif 83194246Smarius#include <machine/resource.h> 84194246Smarius 85194246Smarius#include <dev/mii/mii.h> 86194246Smarius#include <dev/mii/miivar.h> 87194246Smarius 88194246Smarius#include <dev/cas/if_casreg.h> 89194246Smarius#include <dev/cas/if_casvar.h> 90194246Smarius 91194246Smarius#include <dev/pci/pcireg.h> 92194246Smarius#include <dev/pci/pcivar.h> 93194246Smarius 94194246Smarius#include "miibus_if.h" 95194246Smarius 96194246Smarius#define RINGASSERT(n , min, max) \ 97194246Smarius CTASSERT(powerof2(n) && (n) >= (min) && (n) <= (max)) 98194246Smarius 99194246SmariusRINGASSERT(CAS_NRXCOMP, 128, 32768); 100194246SmariusRINGASSERT(CAS_NRXDESC, 32, 8192); 101194246SmariusRINGASSERT(CAS_NRXDESC2, 32, 8192); 102194246SmariusRINGASSERT(CAS_NTXDESC, 32, 8192); 103194246Smarius 104194246Smarius#undef RINGASSERT 105194246Smarius 106194246Smarius#define CCDASSERT(m, a) \ 107194246Smarius CTASSERT((offsetof(struct cas_control_data, m) & ((a) - 1)) == 0) 108194246Smarius 109194246SmariusCCDASSERT(ccd_rxcomps, CAS_RX_COMP_ALIGN); 110194246SmariusCCDASSERT(ccd_rxdescs, CAS_RX_DESC_ALIGN); 111194246SmariusCCDASSERT(ccd_rxdescs2, CAS_RX_DESC_ALIGN); 112194246Smarius 113194246Smarius#undef CCDASSERT 114194246Smarius 115194246Smarius#define CAS_TRIES 10000 116194246Smarius 117194246Smarius/* 118194246Smarius * According to documentation, the hardware has support for basic TCP 119194246Smarius * checksum offloading only, in practice this can be also used for UDP 120194246Smarius * however (i.e. the problem of previous Sun NICs that a checksum of 0x0 121194246Smarius * is not converted to 0xffff no longer exists). 122194246Smarius */ 123194246Smarius#define CAS_CSUM_FEATURES (CSUM_TCP | CSUM_UDP) 124194246Smarius 125194246Smariusstatic inline void cas_add_rxdesc(struct cas_softc *sc, u_int idx); 126194246Smariusstatic int cas_attach(struct cas_softc *sc); 127194246Smariusstatic int cas_bitwait(struct cas_softc *sc, bus_addr_t r, uint32_t clr, 128194246Smarius uint32_t set); 129194246Smariusstatic void cas_cddma_callback(void *xsc, bus_dma_segment_t *segs, 130194246Smarius int nsegs, int error); 131194246Smariusstatic void cas_detach(struct cas_softc *sc); 132194246Smariusstatic int cas_disable_rx(struct cas_softc *sc); 133194246Smariusstatic int cas_disable_tx(struct cas_softc *sc); 134194246Smariusstatic void cas_eint(struct cas_softc *sc, u_int status); 135254842Sandrestatic int cas_free(struct mbuf *m, void *arg1, void* arg2); 136194246Smariusstatic void cas_init(void *xsc); 137194246Smariusstatic void cas_init_locked(struct cas_softc *sc); 138194246Smariusstatic void cas_init_regs(struct cas_softc *sc); 139194904Smariusstatic int cas_intr(void *v); 140194904Smariusstatic void cas_intr_task(void *arg, int pending __unused); 141194246Smariusstatic int cas_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data); 142194246Smariusstatic int cas_load_txmbuf(struct cas_softc *sc, struct mbuf **m_head); 143194246Smariusstatic int cas_mediachange(struct ifnet *ifp); 144194246Smariusstatic void cas_mediastatus(struct ifnet *ifp, struct ifmediareq *ifmr); 145194246Smariusstatic void cas_meminit(struct cas_softc *sc); 146194246Smariusstatic void cas_mifinit(struct cas_softc *sc); 147194246Smariusstatic int cas_mii_readreg(device_t dev, int phy, int reg); 148194246Smariusstatic void cas_mii_statchg(device_t dev); 149194246Smariusstatic int cas_mii_writereg(device_t dev, int phy, int reg, int val); 150194246Smariusstatic void cas_reset(struct cas_softc *sc); 151194246Smariusstatic int cas_reset_rx(struct cas_softc *sc); 152194246Smariusstatic int cas_reset_tx(struct cas_softc *sc); 153194246Smariusstatic void cas_resume(struct cas_softc *sc); 154194246Smariusstatic u_int cas_descsize(u_int sz); 155194246Smariusstatic void cas_rint(struct cas_softc *sc); 156194246Smariusstatic void cas_rint_timeout(void *arg); 157194246Smariusstatic inline void cas_rxcksum(struct mbuf *m, uint16_t cksum); 158194246Smariusstatic inline void cas_rxcompinit(struct cas_rx_comp *rxcomp); 159194246Smariusstatic u_int cas_rxcompsize(u_int sz); 160194246Smariusstatic void cas_rxdma_callback(void *xsc, bus_dma_segment_t *segs, 161194246Smarius int nsegs, int error); 162194246Smariusstatic void cas_setladrf(struct cas_softc *sc); 163194246Smariusstatic void cas_start(struct ifnet *ifp); 164194246Smariusstatic void cas_stop(struct ifnet *ifp); 165194246Smariusstatic void cas_suspend(struct cas_softc *sc); 166194246Smariusstatic void cas_tick(void *arg); 167194246Smariusstatic void cas_tint(struct cas_softc *sc); 168194904Smariusstatic void cas_tx_task(void *arg, int pending __unused); 169194246Smariusstatic inline void cas_txkick(struct cas_softc *sc); 170194904Smariusstatic void cas_watchdog(struct cas_softc *sc); 171194246Smarius 172194246Smariusstatic devclass_t cas_devclass; 173194246Smarius 174194246SmariusMODULE_DEPEND(cas, ether, 1, 1, 1); 175194246SmariusMODULE_DEPEND(cas, miibus, 1, 1, 1); 176194246Smarius 177194246Smarius#ifdef CAS_DEBUG 178194246Smarius#include <sys/ktr.h> 179210334Sattilio#define KTR_CAS KTR_SPARE2 180194246Smarius#endif 181194246Smarius 182194246Smariusstatic int 183194246Smariuscas_attach(struct cas_softc *sc) 184194246Smarius{ 185194246Smarius struct cas_txsoft *txs; 186194246Smarius struct ifnet *ifp; 187194246Smarius int error, i; 188194246Smarius uint32_t v; 189194246Smarius 190194246Smarius /* Set up ifnet structure. */ 191194246Smarius ifp = sc->sc_ifp = if_alloc(IFT_ETHER); 192194246Smarius if (ifp == NULL) 193194246Smarius return (ENOSPC); 194194246Smarius ifp->if_softc = sc; 195194246Smarius if_initname(ifp, device_get_name(sc->sc_dev), 196194246Smarius device_get_unit(sc->sc_dev)); 197194246Smarius ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; 198194246Smarius ifp->if_start = cas_start; 199194246Smarius ifp->if_ioctl = cas_ioctl; 200194246Smarius ifp->if_init = cas_init; 201194246Smarius IFQ_SET_MAXLEN(&ifp->if_snd, CAS_TXQUEUELEN); 202194246Smarius ifp->if_snd.ifq_drv_maxlen = CAS_TXQUEUELEN; 203194246Smarius IFQ_SET_READY(&ifp->if_snd); 204194246Smarius 205194246Smarius callout_init_mtx(&sc->sc_tick_ch, &sc->sc_mtx, 0); 206223986Smarius callout_init_mtx(&sc->sc_rx_ch, &sc->sc_mtx, 0); 207194904Smarius /* Create local taskq. */ 208194904Smarius TASK_INIT(&sc->sc_intr_task, 0, cas_intr_task, sc); 209194904Smarius TASK_INIT(&sc->sc_tx_task, 1, cas_tx_task, ifp); 210194904Smarius sc->sc_tq = taskqueue_create_fast("cas_taskq", M_WAITOK, 211194904Smarius taskqueue_thread_enqueue, &sc->sc_tq); 212194904Smarius if (sc->sc_tq == NULL) { 213194904Smarius device_printf(sc->sc_dev, "could not create taskqueue\n"); 214194904Smarius error = ENXIO; 215194904Smarius goto fail_ifnet; 216194904Smarius } 217245923Smarius error = taskqueue_start_threads(&sc->sc_tq, 1, PI_NET, "%s taskq", 218194904Smarius device_get_nameunit(sc->sc_dev)); 219245923Smarius if (error != 0) { 220245923Smarius device_printf(sc->sc_dev, "could not start threads\n"); 221245923Smarius goto fail_taskq; 222245923Smarius } 223194246Smarius 224194246Smarius /* Make sure the chip is stopped. */ 225194246Smarius cas_reset(sc); 226194246Smarius 227194246Smarius error = bus_dma_tag_create(bus_get_dma_tag(sc->sc_dev), 1, 0, 228194246Smarius BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 229194246Smarius BUS_SPACE_MAXSIZE, 0, BUS_SPACE_MAXSIZE, 0, NULL, NULL, 230194246Smarius &sc->sc_pdmatag); 231194246Smarius if (error != 0) 232194904Smarius goto fail_taskq; 233194246Smarius 234194246Smarius error = bus_dma_tag_create(sc->sc_pdmatag, 1, 0, 235194246Smarius BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 236194246Smarius CAS_PAGE_SIZE, 1, CAS_PAGE_SIZE, 0, NULL, NULL, &sc->sc_rdmatag); 237194246Smarius if (error != 0) 238194246Smarius goto fail_ptag; 239194246Smarius 240194246Smarius error = bus_dma_tag_create(sc->sc_pdmatag, 1, 0, 241194246Smarius BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 242194246Smarius MCLBYTES * CAS_NTXSEGS, CAS_NTXSEGS, MCLBYTES, 243194246Smarius BUS_DMA_ALLOCNOW, NULL, NULL, &sc->sc_tdmatag); 244194246Smarius if (error != 0) 245194246Smarius goto fail_rtag; 246194246Smarius 247194246Smarius error = bus_dma_tag_create(sc->sc_pdmatag, CAS_TX_DESC_ALIGN, 0, 248194246Smarius BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 249194246Smarius sizeof(struct cas_control_data), 1, 250194246Smarius sizeof(struct cas_control_data), 0, 251194246Smarius NULL, NULL, &sc->sc_cdmatag); 252194246Smarius if (error != 0) 253194246Smarius goto fail_ttag; 254194246Smarius 255194246Smarius /* 256194246Smarius * Allocate the control data structures, create and load the 257194246Smarius * DMA map for it. 258194246Smarius */ 259194246Smarius if ((error = bus_dmamem_alloc(sc->sc_cdmatag, 260194246Smarius (void **)&sc->sc_control_data, 261194246Smarius BUS_DMA_WAITOK | BUS_DMA_COHERENT | BUS_DMA_ZERO, 262194246Smarius &sc->sc_cddmamap)) != 0) { 263194246Smarius device_printf(sc->sc_dev, 264194246Smarius "unable to allocate control data, error = %d\n", error); 265194246Smarius goto fail_ctag; 266194246Smarius } 267194246Smarius 268194246Smarius sc->sc_cddma = 0; 269194246Smarius if ((error = bus_dmamap_load(sc->sc_cdmatag, sc->sc_cddmamap, 270194246Smarius sc->sc_control_data, sizeof(struct cas_control_data), 271194246Smarius cas_cddma_callback, sc, 0)) != 0 || sc->sc_cddma == 0) { 272194246Smarius device_printf(sc->sc_dev, 273194246Smarius "unable to load control data DMA map, error = %d\n", 274194246Smarius error); 275194246Smarius goto fail_cmem; 276194246Smarius } 277194246Smarius 278194246Smarius /* 279194246Smarius * Initialize the transmit job descriptors. 280194246Smarius */ 281194246Smarius STAILQ_INIT(&sc->sc_txfreeq); 282194246Smarius STAILQ_INIT(&sc->sc_txdirtyq); 283194246Smarius 284194246Smarius /* 285194246Smarius * Create the transmit buffer DMA maps. 286194246Smarius */ 287194246Smarius error = ENOMEM; 288194246Smarius for (i = 0; i < CAS_TXQUEUELEN; i++) { 289194246Smarius txs = &sc->sc_txsoft[i]; 290194246Smarius txs->txs_mbuf = NULL; 291194246Smarius txs->txs_ndescs = 0; 292194246Smarius if ((error = bus_dmamap_create(sc->sc_tdmatag, 0, 293194246Smarius &txs->txs_dmamap)) != 0) { 294194246Smarius device_printf(sc->sc_dev, 295194246Smarius "unable to create TX DMA map %d, error = %d\n", 296194246Smarius i, error); 297194246Smarius goto fail_txd; 298194246Smarius } 299194246Smarius STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 300194246Smarius } 301194246Smarius 302194246Smarius /* 303194246Smarius * Allocate the receive buffers, create and load the DMA maps 304194246Smarius * for them. 305194246Smarius */ 306194246Smarius for (i = 0; i < CAS_NRXDESC; i++) { 307194246Smarius if ((error = bus_dmamem_alloc(sc->sc_rdmatag, 308194246Smarius &sc->sc_rxdsoft[i].rxds_buf, BUS_DMA_WAITOK, 309194246Smarius &sc->sc_rxdsoft[i].rxds_dmamap)) != 0) { 310194246Smarius device_printf(sc->sc_dev, 311194246Smarius "unable to allocate RX buffer %d, error = %d\n", 312194246Smarius i, error); 313194246Smarius goto fail_rxmem; 314194246Smarius } 315194246Smarius 316194246Smarius sc->sc_rxdptr = i; 317194246Smarius sc->sc_rxdsoft[i].rxds_paddr = 0; 318194246Smarius if ((error = bus_dmamap_load(sc->sc_rdmatag, 319194246Smarius sc->sc_rxdsoft[i].rxds_dmamap, sc->sc_rxdsoft[i].rxds_buf, 320194246Smarius CAS_PAGE_SIZE, cas_rxdma_callback, sc, 0)) != 0 || 321194246Smarius sc->sc_rxdsoft[i].rxds_paddr == 0) { 322194246Smarius device_printf(sc->sc_dev, 323194246Smarius "unable to load RX DMA map %d, error = %d\n", 324194246Smarius i, error); 325194246Smarius goto fail_rxmap; 326194246Smarius } 327194246Smarius } 328194246Smarius 329207585Smarius if ((sc->sc_flags & CAS_SERDES) == 0) { 330207585Smarius CAS_WRITE_4(sc, CAS_PCS_DATAPATH, CAS_PCS_DATAPATH_MII); 331207585Smarius CAS_BARRIER(sc, CAS_PCS_DATAPATH, 4, 332207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 333207585Smarius cas_mifinit(sc); 334207585Smarius /* 335207585Smarius * Look for an external PHY. 336207585Smarius */ 337207585Smarius error = ENXIO; 338207585Smarius v = CAS_READ_4(sc, CAS_MIF_CONF); 339207585Smarius if ((v & CAS_MIF_CONF_MDI1) != 0) { 340207585Smarius v |= CAS_MIF_CONF_PHY_SELECT; 341207585Smarius CAS_WRITE_4(sc, CAS_MIF_CONF, v); 342207585Smarius CAS_BARRIER(sc, CAS_MIF_CONF, 4, 343207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 344207585Smarius /* Enable/unfreeze the GMII pins of Saturn. */ 345207585Smarius if (sc->sc_variant == CAS_SATURN) { 346245923Smarius CAS_WRITE_4(sc, CAS_SATURN_PCFG, 347245923Smarius CAS_READ_4(sc, CAS_SATURN_PCFG) & 348245923Smarius ~CAS_SATURN_PCFG_FSI); 349207585Smarius CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 350207585Smarius BUS_SPACE_BARRIER_READ | 351207585Smarius BUS_SPACE_BARRIER_WRITE); 352245923Smarius DELAY(10000); 353207585Smarius } 354213893Smarius error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 355213893Smarius cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 356215721Smarius MII_PHY_ANY, MII_OFFSET_ANY, MIIF_DOPAUSE); 357194246Smarius } 358207585Smarius /* 359207585Smarius * Fall back on an internal PHY if no external PHY was found. 360207585Smarius */ 361207585Smarius if (error != 0 && (v & CAS_MIF_CONF_MDI0) != 0) { 362207585Smarius v &= ~CAS_MIF_CONF_PHY_SELECT; 363207585Smarius CAS_WRITE_4(sc, CAS_MIF_CONF, v); 364207585Smarius CAS_BARRIER(sc, CAS_MIF_CONF, 4, 365207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 366207585Smarius /* Freeze the GMII pins of Saturn for saving power. */ 367207585Smarius if (sc->sc_variant == CAS_SATURN) { 368207585Smarius CAS_WRITE_4(sc, CAS_SATURN_PCFG, 369245923Smarius CAS_READ_4(sc, CAS_SATURN_PCFG) | 370207585Smarius CAS_SATURN_PCFG_FSI); 371207585Smarius CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 372207585Smarius BUS_SPACE_BARRIER_READ | 373207585Smarius BUS_SPACE_BARRIER_WRITE); 374245923Smarius DELAY(10000); 375207585Smarius } 376213893Smarius error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 377213893Smarius cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 378215721Smarius MII_PHY_ANY, MII_OFFSET_ANY, MIIF_DOPAUSE); 379194246Smarius } 380207585Smarius } else { 381207585Smarius /* 382207585Smarius * Use the external PCS SERDES. 383207585Smarius */ 384194246Smarius CAS_WRITE_4(sc, CAS_PCS_DATAPATH, CAS_PCS_DATAPATH_SERDES); 385207585Smarius CAS_BARRIER(sc, CAS_PCS_DATAPATH, 4, BUS_SPACE_BARRIER_WRITE); 386207585Smarius /* Enable/unfreeze the SERDES pins of Saturn. */ 387207585Smarius if (sc->sc_variant == CAS_SATURN) { 388207585Smarius CAS_WRITE_4(sc, CAS_SATURN_PCFG, 0); 389207585Smarius CAS_BARRIER(sc, CAS_SATURN_PCFG, 4, 390207585Smarius BUS_SPACE_BARRIER_WRITE); 391207585Smarius } 392194246Smarius CAS_WRITE_4(sc, CAS_PCS_SERDES_CTRL, CAS_PCS_SERDES_CTRL_ESD); 393207585Smarius CAS_BARRIER(sc, CAS_PCS_SERDES_CTRL, 4, 394207585Smarius BUS_SPACE_BARRIER_WRITE); 395207585Smarius CAS_WRITE_4(sc, CAS_PCS_CONF, CAS_PCS_CONF_EN); 396207585Smarius CAS_BARRIER(sc, CAS_PCS_CONF, 4, 397207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 398213893Smarius error = mii_attach(sc->sc_dev, &sc->sc_miibus, ifp, 399213893Smarius cas_mediachange, cas_mediastatus, BMSR_DEFCAPMASK, 400215721Smarius CAS_PHYAD_EXTERNAL, MII_OFFSET_ANY, MIIF_DOPAUSE); 401194246Smarius } 402194246Smarius if (error != 0) { 403213893Smarius device_printf(sc->sc_dev, "attaching PHYs failed\n"); 404194246Smarius goto fail_rxmap; 405194246Smarius } 406194246Smarius sc->sc_mii = device_get_softc(sc->sc_miibus); 407194246Smarius 408194246Smarius /* 409194246Smarius * From this point forward, the attachment cannot fail. A failure 410194246Smarius * before this point releases all resources that may have been 411194246Smarius * allocated. 412194246Smarius */ 413194246Smarius 414194246Smarius /* Announce FIFO sizes. */ 415194246Smarius v = CAS_READ_4(sc, CAS_TX_FIFO_SIZE); 416194246Smarius device_printf(sc->sc_dev, "%ukB RX FIFO, %ukB TX FIFO\n", 417194246Smarius CAS_RX_FIFO_SIZE / 1024, v / 16); 418194246Smarius 419194246Smarius /* Attach the interface. */ 420194246Smarius ether_ifattach(ifp, sc->sc_enaddr); 421194246Smarius 422194246Smarius /* 423194246Smarius * Tell the upper layer(s) we support long frames/checksum offloads. 424194246Smarius */ 425194246Smarius ifp->if_data.ifi_hdrlen = sizeof(struct ether_vlan_header); 426194246Smarius ifp->if_capabilities = IFCAP_VLAN_MTU; 427194246Smarius if ((sc->sc_flags & CAS_NO_CSUM) == 0) { 428194246Smarius ifp->if_capabilities |= IFCAP_HWCSUM; 429194246Smarius ifp->if_hwassist = CAS_CSUM_FEATURES; 430194246Smarius } 431194246Smarius ifp->if_capenable = ifp->if_capabilities; 432194246Smarius 433194246Smarius return (0); 434194246Smarius 435194246Smarius /* 436194246Smarius * Free any resources we've allocated during the failed attach 437194246Smarius * attempt. Do this in reverse order and fall through. 438194246Smarius */ 439194246Smarius fail_rxmap: 440194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 441194246Smarius if (sc->sc_rxdsoft[i].rxds_paddr != 0) 442194246Smarius bus_dmamap_unload(sc->sc_rdmatag, 443194246Smarius sc->sc_rxdsoft[i].rxds_dmamap); 444194246Smarius fail_rxmem: 445194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 446194246Smarius if (sc->sc_rxdsoft[i].rxds_buf != NULL) 447194246Smarius bus_dmamem_free(sc->sc_rdmatag, 448194246Smarius sc->sc_rxdsoft[i].rxds_buf, 449194246Smarius sc->sc_rxdsoft[i].rxds_dmamap); 450194246Smarius fail_txd: 451194246Smarius for (i = 0; i < CAS_TXQUEUELEN; i++) 452194246Smarius if (sc->sc_txsoft[i].txs_dmamap != NULL) 453194246Smarius bus_dmamap_destroy(sc->sc_tdmatag, 454194246Smarius sc->sc_txsoft[i].txs_dmamap); 455194246Smarius bus_dmamap_unload(sc->sc_cdmatag, sc->sc_cddmamap); 456194246Smarius fail_cmem: 457194246Smarius bus_dmamem_free(sc->sc_cdmatag, sc->sc_control_data, 458194246Smarius sc->sc_cddmamap); 459194246Smarius fail_ctag: 460194246Smarius bus_dma_tag_destroy(sc->sc_cdmatag); 461194246Smarius fail_ttag: 462194246Smarius bus_dma_tag_destroy(sc->sc_tdmatag); 463194246Smarius fail_rtag: 464194246Smarius bus_dma_tag_destroy(sc->sc_rdmatag); 465194246Smarius fail_ptag: 466194246Smarius bus_dma_tag_destroy(sc->sc_pdmatag); 467194904Smarius fail_taskq: 468194904Smarius taskqueue_free(sc->sc_tq); 469194246Smarius fail_ifnet: 470194246Smarius if_free(ifp); 471194246Smarius return (error); 472194246Smarius} 473194246Smarius 474194246Smariusstatic void 475194246Smariuscas_detach(struct cas_softc *sc) 476194246Smarius{ 477194246Smarius struct ifnet *ifp = sc->sc_ifp; 478194246Smarius int i; 479194246Smarius 480194904Smarius ether_ifdetach(ifp); 481194246Smarius CAS_LOCK(sc); 482194246Smarius cas_stop(ifp); 483194246Smarius CAS_UNLOCK(sc); 484194246Smarius callout_drain(&sc->sc_tick_ch); 485194246Smarius callout_drain(&sc->sc_rx_ch); 486194904Smarius taskqueue_drain(sc->sc_tq, &sc->sc_intr_task); 487194904Smarius taskqueue_drain(sc->sc_tq, &sc->sc_tx_task); 488194246Smarius if_free(ifp); 489194904Smarius taskqueue_free(sc->sc_tq); 490194246Smarius device_delete_child(sc->sc_dev, sc->sc_miibus); 491194246Smarius 492194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 493194246Smarius if (sc->sc_rxdsoft[i].rxds_dmamap != NULL) 494194246Smarius bus_dmamap_sync(sc->sc_rdmatag, 495194246Smarius sc->sc_rxdsoft[i].rxds_dmamap, 496194246Smarius BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 497194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 498194246Smarius if (sc->sc_rxdsoft[i].rxds_paddr != 0) 499194246Smarius bus_dmamap_unload(sc->sc_rdmatag, 500194246Smarius sc->sc_rxdsoft[i].rxds_dmamap); 501194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 502194246Smarius if (sc->sc_rxdsoft[i].rxds_buf != NULL) 503194246Smarius bus_dmamem_free(sc->sc_rdmatag, 504194246Smarius sc->sc_rxdsoft[i].rxds_buf, 505194246Smarius sc->sc_rxdsoft[i].rxds_dmamap); 506194246Smarius for (i = 0; i < CAS_TXQUEUELEN; i++) 507194246Smarius if (sc->sc_txsoft[i].txs_dmamap != NULL) 508194246Smarius bus_dmamap_destroy(sc->sc_tdmatag, 509194246Smarius sc->sc_txsoft[i].txs_dmamap); 510194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 511194246Smarius bus_dmamap_unload(sc->sc_cdmatag, sc->sc_cddmamap); 512194246Smarius bus_dmamem_free(sc->sc_cdmatag, sc->sc_control_data, 513194246Smarius sc->sc_cddmamap); 514194246Smarius bus_dma_tag_destroy(sc->sc_cdmatag); 515194246Smarius bus_dma_tag_destroy(sc->sc_tdmatag); 516194246Smarius bus_dma_tag_destroy(sc->sc_rdmatag); 517194246Smarius bus_dma_tag_destroy(sc->sc_pdmatag); 518194246Smarius} 519194246Smarius 520194246Smariusstatic void 521194246Smariuscas_suspend(struct cas_softc *sc) 522194246Smarius{ 523194246Smarius struct ifnet *ifp = sc->sc_ifp; 524194246Smarius 525194246Smarius CAS_LOCK(sc); 526194246Smarius cas_stop(ifp); 527194246Smarius CAS_UNLOCK(sc); 528194246Smarius} 529194246Smarius 530194246Smariusstatic void 531194246Smariuscas_resume(struct cas_softc *sc) 532194246Smarius{ 533194246Smarius struct ifnet *ifp = sc->sc_ifp; 534194246Smarius 535194246Smarius CAS_LOCK(sc); 536194246Smarius /* 537194246Smarius * On resume all registers have to be initialized again like 538194246Smarius * after power-on. 539194246Smarius */ 540194246Smarius sc->sc_flags &= ~CAS_INITED; 541194246Smarius if (ifp->if_flags & IFF_UP) 542194246Smarius cas_init_locked(sc); 543194246Smarius CAS_UNLOCK(sc); 544194246Smarius} 545194246Smarius 546194246Smariusstatic inline void 547194246Smariuscas_rxcksum(struct mbuf *m, uint16_t cksum) 548194246Smarius{ 549194246Smarius struct ether_header *eh; 550194246Smarius struct ip *ip; 551194246Smarius struct udphdr *uh; 552194246Smarius uint16_t *opts; 553194246Smarius int32_t hlen, len, pktlen; 554194246Smarius uint32_t temp32; 555194246Smarius 556194246Smarius pktlen = m->m_pkthdr.len; 557194246Smarius if (pktlen < sizeof(struct ether_header) + sizeof(struct ip)) 558194246Smarius return; 559194246Smarius eh = mtod(m, struct ether_header *); 560194246Smarius if (eh->ether_type != htons(ETHERTYPE_IP)) 561194246Smarius return; 562194246Smarius ip = (struct ip *)(eh + 1); 563194246Smarius if (ip->ip_v != IPVERSION) 564194246Smarius return; 565194246Smarius 566194246Smarius hlen = ip->ip_hl << 2; 567194246Smarius pktlen -= sizeof(struct ether_header); 568194246Smarius if (hlen < sizeof(struct ip)) 569194246Smarius return; 570194246Smarius if (ntohs(ip->ip_len) < hlen) 571194246Smarius return; 572194246Smarius if (ntohs(ip->ip_len) != pktlen) 573194246Smarius return; 574194246Smarius if (ip->ip_off & htons(IP_MF | IP_OFFMASK)) 575194246Smarius return; /* Cannot handle fragmented packet. */ 576194246Smarius 577194246Smarius switch (ip->ip_p) { 578194246Smarius case IPPROTO_TCP: 579194246Smarius if (pktlen < (hlen + sizeof(struct tcphdr))) 580194246Smarius return; 581194246Smarius break; 582194246Smarius case IPPROTO_UDP: 583194246Smarius if (pktlen < (hlen + sizeof(struct udphdr))) 584194246Smarius return; 585194246Smarius uh = (struct udphdr *)((uint8_t *)ip + hlen); 586194246Smarius if (uh->uh_sum == 0) 587194246Smarius return; /* no checksum */ 588194246Smarius break; 589194246Smarius default: 590194246Smarius return; 591194246Smarius } 592194246Smarius 593194246Smarius cksum = ~cksum; 594194246Smarius /* checksum fixup for IP options */ 595194246Smarius len = hlen - sizeof(struct ip); 596194246Smarius if (len > 0) { 597194246Smarius opts = (uint16_t *)(ip + 1); 598194246Smarius for (; len > 0; len -= sizeof(uint16_t), opts++) { 599194246Smarius temp32 = cksum - *opts; 600194246Smarius temp32 = (temp32 >> 16) + (temp32 & 65535); 601194246Smarius cksum = temp32 & 65535; 602194246Smarius } 603194246Smarius } 604194246Smarius m->m_pkthdr.csum_flags |= CSUM_DATA_VALID; 605194246Smarius m->m_pkthdr.csum_data = cksum; 606194246Smarius} 607194246Smarius 608194246Smariusstatic void 609194246Smariuscas_cddma_callback(void *xsc, bus_dma_segment_t *segs, int nsegs, int error) 610194246Smarius{ 611194246Smarius struct cas_softc *sc = xsc; 612194246Smarius 613194246Smarius if (error != 0) 614194246Smarius return; 615194246Smarius if (nsegs != 1) 616194246Smarius panic("%s: bad control buffer segment count", __func__); 617194246Smarius sc->sc_cddma = segs[0].ds_addr; 618194246Smarius} 619194246Smarius 620194246Smariusstatic void 621194246Smariuscas_rxdma_callback(void *xsc, bus_dma_segment_t *segs, int nsegs, int error) 622194246Smarius{ 623194246Smarius struct cas_softc *sc = xsc; 624194246Smarius 625194246Smarius if (error != 0) 626194246Smarius return; 627194246Smarius if (nsegs != 1) 628194246Smarius panic("%s: bad RX buffer segment count", __func__); 629194246Smarius sc->sc_rxdsoft[sc->sc_rxdptr].rxds_paddr = segs[0].ds_addr; 630194246Smarius} 631194246Smarius 632194246Smariusstatic void 633194246Smariuscas_tick(void *arg) 634194246Smarius{ 635194246Smarius struct cas_softc *sc = arg; 636194904Smarius struct ifnet *ifp = sc->sc_ifp; 637194246Smarius uint32_t v; 638194246Smarius 639194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 640194246Smarius 641194246Smarius /* 642194246Smarius * Unload collision and error counters. 643194246Smarius */ 644194246Smarius ifp->if_collisions += 645194246Smarius CAS_READ_4(sc, CAS_MAC_NORM_COLL_CNT) + 646194246Smarius CAS_READ_4(sc, CAS_MAC_FIRST_COLL_CNT); 647194246Smarius v = CAS_READ_4(sc, CAS_MAC_EXCESS_COLL_CNT) + 648194246Smarius CAS_READ_4(sc, CAS_MAC_LATE_COLL_CNT); 649194246Smarius ifp->if_collisions += v; 650194246Smarius ifp->if_oerrors += v; 651194246Smarius ifp->if_ierrors += 652194246Smarius CAS_READ_4(sc, CAS_MAC_RX_LEN_ERR_CNT) + 653194246Smarius CAS_READ_4(sc, CAS_MAC_RX_ALIGN_ERR) + 654194246Smarius CAS_READ_4(sc, CAS_MAC_RX_CRC_ERR_CNT) + 655194246Smarius CAS_READ_4(sc, CAS_MAC_RX_CODE_VIOL); 656194246Smarius 657194246Smarius /* 658194246Smarius * Then clear the hardware counters. 659194246Smarius */ 660194246Smarius CAS_WRITE_4(sc, CAS_MAC_NORM_COLL_CNT, 0); 661194246Smarius CAS_WRITE_4(sc, CAS_MAC_FIRST_COLL_CNT, 0); 662194246Smarius CAS_WRITE_4(sc, CAS_MAC_EXCESS_COLL_CNT, 0); 663194246Smarius CAS_WRITE_4(sc, CAS_MAC_LATE_COLL_CNT, 0); 664194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_LEN_ERR_CNT, 0); 665194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_ALIGN_ERR, 0); 666194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CRC_ERR_CNT, 0); 667194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CODE_VIOL, 0); 668194246Smarius 669194246Smarius mii_tick(sc->sc_mii); 670194246Smarius 671194904Smarius if (sc->sc_txfree != CAS_MAXTXFREE) 672194904Smarius cas_tint(sc); 673194246Smarius 674194904Smarius cas_watchdog(sc); 675194904Smarius 676194246Smarius callout_reset(&sc->sc_tick_ch, hz, cas_tick, sc); 677194246Smarius} 678194246Smarius 679194246Smariusstatic int 680194246Smariuscas_bitwait(struct cas_softc *sc, bus_addr_t r, uint32_t clr, uint32_t set) 681194246Smarius{ 682194246Smarius int i; 683194246Smarius uint32_t reg; 684194246Smarius 685194246Smarius for (i = CAS_TRIES; i--; DELAY(100)) { 686194246Smarius reg = CAS_READ_4(sc, r); 687194246Smarius if ((reg & clr) == 0 && (reg & set) == set) 688194246Smarius return (1); 689194246Smarius } 690194246Smarius return (0); 691194246Smarius} 692194246Smarius 693194246Smariusstatic void 694194246Smariuscas_reset(struct cas_softc *sc) 695194246Smarius{ 696194246Smarius 697194246Smarius#ifdef CAS_DEBUG 698194246Smarius CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 699194246Smarius#endif 700194246Smarius /* Disable all interrupts in order to avoid spurious ones. */ 701194246Smarius CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 702194246Smarius 703194246Smarius cas_reset_rx(sc); 704194246Smarius cas_reset_tx(sc); 705194246Smarius 706194246Smarius /* 707194246Smarius * Do a full reset modulo the result of the last auto-negotiation 708194246Smarius * when using the SERDES. 709194246Smarius */ 710194246Smarius CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_RX | CAS_RESET_TX | 711194246Smarius ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 712194246Smarius CAS_BARRIER(sc, CAS_RESET, 4, 713194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 714194246Smarius DELAY(3000); 715194246Smarius if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_RX | CAS_RESET_TX, 0)) 716194246Smarius device_printf(sc->sc_dev, "cannot reset device\n"); 717194246Smarius} 718194246Smarius 719194246Smariusstatic void 720194246Smariuscas_stop(struct ifnet *ifp) 721194246Smarius{ 722194246Smarius struct cas_softc *sc = ifp->if_softc; 723194246Smarius struct cas_txsoft *txs; 724194246Smarius 725194246Smarius#ifdef CAS_DEBUG 726194246Smarius CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 727194246Smarius#endif 728194246Smarius 729194246Smarius callout_stop(&sc->sc_tick_ch); 730194246Smarius callout_stop(&sc->sc_rx_ch); 731194246Smarius 732194246Smarius /* Disable all interrupts in order to avoid spurious ones. */ 733194246Smarius CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 734194246Smarius 735194246Smarius cas_reset_tx(sc); 736194246Smarius cas_reset_rx(sc); 737194246Smarius 738194246Smarius /* 739194246Smarius * Release any queued transmit buffers. 740194246Smarius */ 741194246Smarius while ((txs = STAILQ_FIRST(&sc->sc_txdirtyq)) != NULL) { 742194246Smarius STAILQ_REMOVE_HEAD(&sc->sc_txdirtyq, txs_q); 743194246Smarius if (txs->txs_ndescs != 0) { 744194246Smarius bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 745194246Smarius BUS_DMASYNC_POSTWRITE); 746194246Smarius bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 747194246Smarius if (txs->txs_mbuf != NULL) { 748194246Smarius m_freem(txs->txs_mbuf); 749194246Smarius txs->txs_mbuf = NULL; 750194246Smarius } 751194246Smarius } 752194246Smarius STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 753194246Smarius } 754194246Smarius 755194246Smarius /* 756194246Smarius * Mark the interface down and cancel the watchdog timer. 757194246Smarius */ 758194246Smarius ifp->if_drv_flags &= ~(IFF_DRV_RUNNING | IFF_DRV_OACTIVE); 759194246Smarius sc->sc_flags &= ~CAS_LINK; 760194246Smarius sc->sc_wdog_timer = 0; 761194246Smarius} 762194246Smarius 763194246Smariusstatic int 764194246Smariuscas_reset_rx(struct cas_softc *sc) 765194246Smarius{ 766194246Smarius 767194246Smarius /* 768194246Smarius * Resetting while DMA is in progress can cause a bus hang, so we 769194246Smarius * disable DMA first. 770194246Smarius */ 771223951Smarius (void)cas_disable_rx(sc); 772194246Smarius CAS_WRITE_4(sc, CAS_RX_CONF, 0); 773194246Smarius CAS_BARRIER(sc, CAS_RX_CONF, 4, 774194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 775194246Smarius if (!cas_bitwait(sc, CAS_RX_CONF, CAS_RX_CONF_RXDMA_EN, 0)) 776194246Smarius device_printf(sc->sc_dev, "cannot disable RX DMA\n"); 777194246Smarius 778194246Smarius /* Finally, reset the ERX. */ 779194246Smarius CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_RX | 780194246Smarius ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 781194246Smarius CAS_BARRIER(sc, CAS_RESET, 4, 782194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 783223951Smarius if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_RX, 0)) { 784194246Smarius device_printf(sc->sc_dev, "cannot reset receiver\n"); 785194246Smarius return (1); 786194246Smarius } 787194246Smarius return (0); 788194246Smarius} 789194246Smarius 790194246Smariusstatic int 791194246Smariuscas_reset_tx(struct cas_softc *sc) 792194246Smarius{ 793194246Smarius 794194246Smarius /* 795194246Smarius * Resetting while DMA is in progress can cause a bus hang, so we 796194246Smarius * disable DMA first. 797194246Smarius */ 798223951Smarius (void)cas_disable_tx(sc); 799194246Smarius CAS_WRITE_4(sc, CAS_TX_CONF, 0); 800194246Smarius CAS_BARRIER(sc, CAS_TX_CONF, 4, 801194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 802194246Smarius if (!cas_bitwait(sc, CAS_TX_CONF, CAS_TX_CONF_TXDMA_EN, 0)) 803194246Smarius device_printf(sc->sc_dev, "cannot disable TX DMA\n"); 804194246Smarius 805194246Smarius /* Finally, reset the ETX. */ 806194246Smarius CAS_WRITE_4(sc, CAS_RESET, CAS_RESET_TX | 807194246Smarius ((sc->sc_flags & CAS_SERDES) != 0 ? CAS_RESET_PCS_DIS : 0)); 808194246Smarius CAS_BARRIER(sc, CAS_RESET, 4, 809194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 810223951Smarius if (!cas_bitwait(sc, CAS_RESET, CAS_RESET_TX, 0)) { 811194246Smarius device_printf(sc->sc_dev, "cannot reset transmitter\n"); 812194246Smarius return (1); 813194246Smarius } 814194246Smarius return (0); 815194246Smarius} 816194246Smarius 817194246Smariusstatic int 818194246Smariuscas_disable_rx(struct cas_softc *sc) 819194246Smarius{ 820194246Smarius 821194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CONF, 822194246Smarius CAS_READ_4(sc, CAS_MAC_RX_CONF) & ~CAS_MAC_RX_CONF_EN); 823194246Smarius CAS_BARRIER(sc, CAS_MAC_RX_CONF, 4, 824194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 825223951Smarius if (cas_bitwait(sc, CAS_MAC_RX_CONF, CAS_MAC_RX_CONF_EN, 0)) 826223951Smarius return (1); 827247579Smarius if (bootverbose) 828247579Smarius device_printf(sc->sc_dev, "cannot disable RX MAC\n"); 829223951Smarius return (0); 830194246Smarius} 831194246Smarius 832194246Smariusstatic int 833194246Smariuscas_disable_tx(struct cas_softc *sc) 834194246Smarius{ 835194246Smarius 836194246Smarius CAS_WRITE_4(sc, CAS_MAC_TX_CONF, 837194246Smarius CAS_READ_4(sc, CAS_MAC_TX_CONF) & ~CAS_MAC_TX_CONF_EN); 838194246Smarius CAS_BARRIER(sc, CAS_MAC_TX_CONF, 4, 839194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 840223951Smarius if (cas_bitwait(sc, CAS_MAC_TX_CONF, CAS_MAC_TX_CONF_EN, 0)) 841223951Smarius return (1); 842247579Smarius if (bootverbose) 843247579Smarius device_printf(sc->sc_dev, "cannot disable TX MAC\n"); 844223951Smarius return (0); 845194246Smarius} 846194246Smarius 847194246Smariusstatic inline void 848194246Smariuscas_rxcompinit(struct cas_rx_comp *rxcomp) 849194246Smarius{ 850194246Smarius 851194246Smarius rxcomp->crc_word1 = 0; 852194246Smarius rxcomp->crc_word2 = 0; 853194246Smarius rxcomp->crc_word3 = 854194246Smarius htole64(CAS_SET(ETHER_HDR_LEN + sizeof(struct ip), CAS_RC3_CSO)); 855194246Smarius rxcomp->crc_word4 = htole64(CAS_RC4_ZERO); 856194246Smarius} 857194246Smarius 858194246Smariusstatic void 859194246Smariuscas_meminit(struct cas_softc *sc) 860194246Smarius{ 861194246Smarius int i; 862194246Smarius 863194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 864194246Smarius 865194246Smarius /* 866194246Smarius * Initialize the transmit descriptor ring. 867194246Smarius */ 868194246Smarius for (i = 0; i < CAS_NTXDESC; i++) { 869194246Smarius sc->sc_txdescs[i].cd_flags = 0; 870194246Smarius sc->sc_txdescs[i].cd_buf_ptr = 0; 871194246Smarius } 872194246Smarius sc->sc_txfree = CAS_MAXTXFREE; 873194246Smarius sc->sc_txnext = 0; 874194246Smarius sc->sc_txwin = 0; 875194246Smarius 876194246Smarius /* 877194246Smarius * Initialize the receive completion ring. 878194246Smarius */ 879194246Smarius for (i = 0; i < CAS_NRXCOMP; i++) 880194246Smarius cas_rxcompinit(&sc->sc_rxcomps[i]); 881194246Smarius sc->sc_rxcptr = 0; 882194246Smarius 883194246Smarius /* 884194246Smarius * Initialize the first receive descriptor ring. We leave 885194246Smarius * the second one zeroed as we don't actually use it. 886194246Smarius */ 887194246Smarius for (i = 0; i < CAS_NRXDESC; i++) 888194246Smarius CAS_INIT_RXDESC(sc, i, i); 889194246Smarius sc->sc_rxdptr = 0; 890194246Smarius 891194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 892194246Smarius} 893194246Smarius 894194246Smariusstatic u_int 895194246Smariuscas_descsize(u_int sz) 896194246Smarius{ 897194246Smarius 898194246Smarius switch (sz) { 899194246Smarius case 32: 900194246Smarius return (CAS_DESC_32); 901194246Smarius case 64: 902194246Smarius return (CAS_DESC_64); 903194246Smarius case 128: 904194246Smarius return (CAS_DESC_128); 905194246Smarius case 256: 906194246Smarius return (CAS_DESC_256); 907194246Smarius case 512: 908194246Smarius return (CAS_DESC_512); 909194246Smarius case 1024: 910194246Smarius return (CAS_DESC_1K); 911194246Smarius case 2048: 912194246Smarius return (CAS_DESC_2K); 913194246Smarius case 4096: 914194246Smarius return (CAS_DESC_4K); 915194246Smarius case 8192: 916194246Smarius return (CAS_DESC_8K); 917194246Smarius default: 918194246Smarius printf("%s: invalid descriptor ring size %d\n", __func__, sz); 919194246Smarius return (CAS_DESC_32); 920194246Smarius } 921194246Smarius} 922194246Smarius 923194246Smariusstatic u_int 924194246Smariuscas_rxcompsize(u_int sz) 925194246Smarius{ 926194246Smarius 927194246Smarius switch (sz) { 928194246Smarius case 128: 929194246Smarius return (CAS_RX_CONF_COMP_128); 930194246Smarius case 256: 931194246Smarius return (CAS_RX_CONF_COMP_256); 932194246Smarius case 512: 933194246Smarius return (CAS_RX_CONF_COMP_512); 934194246Smarius case 1024: 935194246Smarius return (CAS_RX_CONF_COMP_1K); 936194246Smarius case 2048: 937194246Smarius return (CAS_RX_CONF_COMP_2K); 938194246Smarius case 4096: 939194246Smarius return (CAS_RX_CONF_COMP_4K); 940194246Smarius case 8192: 941194246Smarius return (CAS_RX_CONF_COMP_8K); 942194246Smarius case 16384: 943194246Smarius return (CAS_RX_CONF_COMP_16K); 944194246Smarius case 32768: 945194246Smarius return (CAS_RX_CONF_COMP_32K); 946194246Smarius default: 947194246Smarius printf("%s: invalid dcompletion ring size %d\n", __func__, sz); 948194246Smarius return (CAS_RX_CONF_COMP_128); 949194246Smarius } 950194246Smarius} 951194246Smarius 952194246Smariusstatic void 953194246Smariuscas_init(void *xsc) 954194246Smarius{ 955194246Smarius struct cas_softc *sc = xsc; 956194246Smarius 957194246Smarius CAS_LOCK(sc); 958194246Smarius cas_init_locked(sc); 959194246Smarius CAS_UNLOCK(sc); 960194246Smarius} 961194246Smarius 962194246Smarius/* 963194246Smarius * Initialization of interface; set up initialization block 964194246Smarius * and transmit/receive descriptor rings. 965194246Smarius */ 966194246Smariusstatic void 967194246Smariuscas_init_locked(struct cas_softc *sc) 968194246Smarius{ 969194246Smarius struct ifnet *ifp = sc->sc_ifp; 970194246Smarius uint32_t v; 971194246Smarius 972194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 973194246Smarius 974194904Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) != 0) 975194904Smarius return; 976194904Smarius 977194246Smarius#ifdef CAS_DEBUG 978194246Smarius CTR2(KTR_CAS, "%s: %s: calling stop", device_get_name(sc->sc_dev), 979194246Smarius __func__); 980194246Smarius#endif 981194246Smarius /* 982194246Smarius * Initialization sequence. The numbered steps below correspond 983194246Smarius * to the sequence outlined in section 6.3.5.1 in the Ethernet 984194246Smarius * Channel Engine manual (part of the PCIO manual). 985194246Smarius * See also the STP2002-STQ document from Sun Microsystems. 986194246Smarius */ 987194246Smarius 988194246Smarius /* step 1 & 2. Reset the Ethernet Channel. */ 989194246Smarius cas_stop(ifp); 990194246Smarius cas_reset(sc); 991194246Smarius#ifdef CAS_DEBUG 992194246Smarius CTR2(KTR_CAS, "%s: %s: restarting", device_get_name(sc->sc_dev), 993194246Smarius __func__); 994194246Smarius#endif 995194246Smarius 996207585Smarius if ((sc->sc_flags & CAS_SERDES) == 0) 997207585Smarius /* Re-initialize the MIF. */ 998207585Smarius cas_mifinit(sc); 999194246Smarius 1000194246Smarius /* step 3. Setup data structures in host memory. */ 1001194246Smarius cas_meminit(sc); 1002194246Smarius 1003194246Smarius /* step 4. TX MAC registers & counters */ 1004194246Smarius cas_init_regs(sc); 1005194246Smarius 1006194246Smarius /* step 5. RX MAC registers & counters */ 1007194246Smarius 1008194246Smarius /* step 6 & 7. Program Ring Base Addresses. */ 1009194246Smarius CAS_WRITE_4(sc, CAS_TX_DESC3_BASE_HI, 1010194246Smarius (((uint64_t)CAS_CDTXDADDR(sc, 0)) >> 32)); 1011194246Smarius CAS_WRITE_4(sc, CAS_TX_DESC3_BASE_LO, 1012194246Smarius CAS_CDTXDADDR(sc, 0) & 0xffffffff); 1013194246Smarius 1014194246Smarius CAS_WRITE_4(sc, CAS_RX_COMP_BASE_HI, 1015194246Smarius (((uint64_t)CAS_CDRXCADDR(sc, 0)) >> 32)); 1016194246Smarius CAS_WRITE_4(sc, CAS_RX_COMP_BASE_LO, 1017194246Smarius CAS_CDRXCADDR(sc, 0) & 0xffffffff); 1018194246Smarius 1019194246Smarius CAS_WRITE_4(sc, CAS_RX_DESC_BASE_HI, 1020194246Smarius (((uint64_t)CAS_CDRXDADDR(sc, 0)) >> 32)); 1021194246Smarius CAS_WRITE_4(sc, CAS_RX_DESC_BASE_LO, 1022194246Smarius CAS_CDRXDADDR(sc, 0) & 0xffffffff); 1023194246Smarius 1024194246Smarius if ((sc->sc_flags & CAS_REG_PLUS) != 0) { 1025194246Smarius CAS_WRITE_4(sc, CAS_RX_DESC2_BASE_HI, 1026194246Smarius (((uint64_t)CAS_CDRXD2ADDR(sc, 0)) >> 32)); 1027194246Smarius CAS_WRITE_4(sc, CAS_RX_DESC2_BASE_LO, 1028194246Smarius CAS_CDRXD2ADDR(sc, 0) & 0xffffffff); 1029194246Smarius } 1030194246Smarius 1031194246Smarius#ifdef CAS_DEBUG 1032194246Smarius CTR5(KTR_CAS, 1033194246Smarius "loading TXDR %lx, RXCR %lx, RXDR %lx, RXD2R %lx, cddma %lx", 1034194246Smarius CAS_CDTXDADDR(sc, 0), CAS_CDRXCADDR(sc, 0), CAS_CDRXDADDR(sc, 0), 1035194246Smarius CAS_CDRXD2ADDR(sc, 0), sc->sc_cddma); 1036194246Smarius#endif 1037194246Smarius 1038194246Smarius /* step 8. Global Configuration & Interrupt Masks */ 1039194246Smarius 1040194246Smarius /* Disable weighted round robin. */ 1041194246Smarius CAS_WRITE_4(sc, CAS_CAW, CAS_CAW_RR_DIS); 1042194246Smarius 1043194246Smarius /* 1044194246Smarius * Enable infinite bursts for revisions without PCI issues if 1045194246Smarius * applicable. Doing so greatly improves the TX performance on 1046247579Smarius * !__sparc64__ (on sparc64, setting CAS_INF_BURST improves TX 1047247579Smarius * performance only marginally but hurts RX throughput quite a bit). 1048194246Smarius */ 1049194246Smarius CAS_WRITE_4(sc, CAS_INF_BURST, 1050194246Smarius#if !defined(__sparc64__) 1051194246Smarius (sc->sc_flags & CAS_TABORT) == 0 ? CAS_INF_BURST_EN : 1052194246Smarius#endif 1053194246Smarius 0); 1054194246Smarius 1055194246Smarius /* Set up interrupts. */ 1056194246Smarius CAS_WRITE_4(sc, CAS_INTMASK, 1057194904Smarius ~(CAS_INTR_TX_INT_ME | CAS_INTR_TX_TAG_ERR | 1058194246Smarius CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_TAG_ERR | 1059194246Smarius CAS_INTR_RX_COMP_FULL | CAS_INTR_RX_BUF_AEMPTY | 1060194246Smarius CAS_INTR_RX_COMP_AFULL | CAS_INTR_RX_LEN_MMATCH | 1061194246Smarius CAS_INTR_PCI_ERROR_INT 1062194246Smarius#ifdef CAS_DEBUG 1063194246Smarius | CAS_INTR_PCS_INT | CAS_INTR_MIF 1064194246Smarius#endif 1065194246Smarius )); 1066194904Smarius /* Don't clear top level interrupts when CAS_STATUS_ALIAS is read. */ 1067194904Smarius CAS_WRITE_4(sc, CAS_CLEAR_ALIAS, 0); 1068194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_MASK, ~CAS_MAC_RX_OVERFLOW); 1069194246Smarius CAS_WRITE_4(sc, CAS_MAC_TX_MASK, 1070194246Smarius ~(CAS_MAC_TX_UNDERRUN | CAS_MAC_TX_MAX_PKT_ERR)); 1071194246Smarius#ifdef CAS_DEBUG 1072194246Smarius CAS_WRITE_4(sc, CAS_MAC_CTRL_MASK, 1073194246Smarius ~(CAS_MAC_CTRL_PAUSE_RCVD | CAS_MAC_CTRL_PAUSE | 1074194246Smarius CAS_MAC_CTRL_NON_PAUSE)); 1075194246Smarius#else 1076194246Smarius CAS_WRITE_4(sc, CAS_MAC_CTRL_MASK, 1077194246Smarius CAS_MAC_CTRL_PAUSE_RCVD | CAS_MAC_CTRL_PAUSE | 1078194246Smarius CAS_MAC_CTRL_NON_PAUSE); 1079194246Smarius#endif 1080194246Smarius 1081194246Smarius /* Enable PCI error interrupts. */ 1082194246Smarius CAS_WRITE_4(sc, CAS_ERROR_MASK, 1083194246Smarius ~(CAS_ERROR_DTRTO | CAS_ERROR_OTHER | CAS_ERROR_DMAW_ZERO | 1084194246Smarius CAS_ERROR_DMAR_ZERO | CAS_ERROR_RTRTO)); 1085194246Smarius 1086194246Smarius /* Enable PCI error interrupts in BIM configuration. */ 1087194246Smarius CAS_WRITE_4(sc, CAS_BIM_CONF, 1088194246Smarius CAS_BIM_CONF_DPAR_EN | CAS_BIM_CONF_RMA_EN | CAS_BIM_CONF_RTA_EN); 1089194246Smarius 1090194246Smarius /* 1091194246Smarius * step 9. ETX Configuration: encode receive descriptor ring size, 1092194246Smarius * enable DMA and disable pre-interrupt writeback completion. 1093194246Smarius */ 1094194246Smarius v = cas_descsize(CAS_NTXDESC) << CAS_TX_CONF_DESC3_SHFT; 1095194246Smarius CAS_WRITE_4(sc, CAS_TX_CONF, v | CAS_TX_CONF_TXDMA_EN | 1096194246Smarius CAS_TX_CONF_RDPP_DIS | CAS_TX_CONF_PICWB_DIS); 1097194246Smarius 1098194246Smarius /* step 10. ERX Configuration */ 1099194246Smarius 1100194246Smarius /* 1101194246Smarius * Encode receive completion and descriptor ring sizes, set the 1102194246Smarius * swivel offset. 1103194246Smarius */ 1104194246Smarius v = cas_rxcompsize(CAS_NRXCOMP) << CAS_RX_CONF_COMP_SHFT; 1105194246Smarius v |= cas_descsize(CAS_NRXDESC) << CAS_RX_CONF_DESC_SHFT; 1106194246Smarius if ((sc->sc_flags & CAS_REG_PLUS) != 0) 1107194246Smarius v |= cas_descsize(CAS_NRXDESC2) << CAS_RX_CONF_DESC2_SHFT; 1108194246Smarius CAS_WRITE_4(sc, CAS_RX_CONF, 1109194246Smarius v | (ETHER_ALIGN << CAS_RX_CONF_SOFF_SHFT)); 1110194246Smarius 1111194246Smarius /* Set the PAUSE thresholds. We use the maximum OFF threshold. */ 1112194246Smarius CAS_WRITE_4(sc, CAS_RX_PTHRS, 1113215721Smarius (111 << CAS_RX_PTHRS_XOFF_SHFT) | (15 << CAS_RX_PTHRS_XON_SHFT)); 1114194246Smarius 1115194246Smarius /* RX blanking */ 1116194246Smarius CAS_WRITE_4(sc, CAS_RX_BLANK, 1117194246Smarius (15 << CAS_RX_BLANK_TIME_SHFT) | (5 << CAS_RX_BLANK_PKTS_SHFT)); 1118194246Smarius 1119194246Smarius /* Set RX_COMP_AFULL threshold to half of the RX completions. */ 1120194246Smarius CAS_WRITE_4(sc, CAS_RX_AEMPTY_THRS, 1121194246Smarius (CAS_NRXCOMP / 2) << CAS_RX_AEMPTY_COMP_SHFT); 1122194246Smarius 1123194246Smarius /* Initialize the RX page size register as appropriate for 8k. */ 1124194246Smarius CAS_WRITE_4(sc, CAS_RX_PSZ, 1125194246Smarius (CAS_RX_PSZ_8K << CAS_RX_PSZ_SHFT) | 1126194246Smarius (4 << CAS_RX_PSZ_MB_CNT_SHFT) | 1127194246Smarius (CAS_RX_PSZ_MB_STRD_2K << CAS_RX_PSZ_MB_STRD_SHFT) | 1128194246Smarius (CAS_RX_PSZ_MB_OFF_64 << CAS_RX_PSZ_MB_OFF_SHFT)); 1129194246Smarius 1130194246Smarius /* Disable RX random early detection. */ 1131194246Smarius CAS_WRITE_4(sc, CAS_RX_RED, 0); 1132194246Smarius 1133194246Smarius /* Zero the RX reassembly DMA table. */ 1134194246Smarius for (v = 0; v <= CAS_RX_REAS_DMA_ADDR_LC; v++) { 1135194246Smarius CAS_WRITE_4(sc, CAS_RX_REAS_DMA_ADDR, v); 1136194246Smarius CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_LO, 0); 1137194246Smarius CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_MD, 0); 1138194246Smarius CAS_WRITE_4(sc, CAS_RX_REAS_DMA_DATA_HI, 0); 1139194246Smarius } 1140194246Smarius 1141194246Smarius /* Ensure the RX control FIFO and RX IPP FIFO addresses are zero. */ 1142194246Smarius CAS_WRITE_4(sc, CAS_RX_CTRL_FIFO, 0); 1143194246Smarius CAS_WRITE_4(sc, CAS_RX_IPP_ADDR, 0); 1144194246Smarius 1145194246Smarius /* Finally, enable RX DMA. */ 1146194246Smarius CAS_WRITE_4(sc, CAS_RX_CONF, 1147194246Smarius CAS_READ_4(sc, CAS_RX_CONF) | CAS_RX_CONF_RXDMA_EN); 1148194246Smarius 1149194246Smarius /* step 11. Configure Media. */ 1150194246Smarius 1151194246Smarius /* step 12. RX_MAC Configuration Register */ 1152223951Smarius v = CAS_READ_4(sc, CAS_MAC_RX_CONF); 1153223951Smarius v &= ~(CAS_MAC_RX_CONF_STRPPAD | CAS_MAC_RX_CONF_EN); 1154223951Smarius v |= CAS_MAC_RX_CONF_STRPFCS; 1155223951Smarius sc->sc_mac_rxcfg = v; 1156223951Smarius /* 1157223951Smarius * Clear the RX filter and reprogram it. This will also set the 1158223951Smarius * current RX MAC configuration and enable it. 1159223951Smarius */ 1160223951Smarius cas_setladrf(sc); 1161194246Smarius 1162194246Smarius /* step 13. TX_MAC Configuration Register */ 1163194246Smarius v = CAS_READ_4(sc, CAS_MAC_TX_CONF); 1164194246Smarius v |= CAS_MAC_TX_CONF_EN; 1165223951Smarius (void)cas_disable_tx(sc); 1166194246Smarius CAS_WRITE_4(sc, CAS_MAC_TX_CONF, v); 1167194246Smarius 1168194246Smarius /* step 14. Issue Transmit Pending command. */ 1169194246Smarius 1170220943Smarius /* step 15. Give the receiver a swift kick. */ 1171194246Smarius CAS_WRITE_4(sc, CAS_RX_KICK, CAS_NRXDESC - 4); 1172194246Smarius CAS_WRITE_4(sc, CAS_RX_COMP_TAIL, 0); 1173194246Smarius if ((sc->sc_flags & CAS_REG_PLUS) != 0) 1174194246Smarius CAS_WRITE_4(sc, CAS_RX_KICK2, CAS_NRXDESC2 - 4); 1175194246Smarius 1176194246Smarius ifp->if_drv_flags |= IFF_DRV_RUNNING; 1177194246Smarius ifp->if_drv_flags &= ~IFF_DRV_OACTIVE; 1178194246Smarius 1179194246Smarius mii_mediachg(sc->sc_mii); 1180194246Smarius 1181194246Smarius /* Start the one second timer. */ 1182194246Smarius sc->sc_wdog_timer = 0; 1183194246Smarius callout_reset(&sc->sc_tick_ch, hz, cas_tick, sc); 1184194246Smarius} 1185194246Smarius 1186194246Smariusstatic int 1187194246Smariuscas_load_txmbuf(struct cas_softc *sc, struct mbuf **m_head) 1188194246Smarius{ 1189194246Smarius bus_dma_segment_t txsegs[CAS_NTXSEGS]; 1190194246Smarius struct cas_txsoft *txs; 1191194246Smarius struct ip *ip; 1192194246Smarius struct mbuf *m; 1193194246Smarius uint64_t cflags; 1194194246Smarius int error, nexttx, nsegs, offset, seg; 1195194246Smarius 1196194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1197194246Smarius 1198194246Smarius /* Get a work queue entry. */ 1199194246Smarius if ((txs = STAILQ_FIRST(&sc->sc_txfreeq)) == NULL) { 1200194246Smarius /* Ran out of descriptors. */ 1201194246Smarius return (ENOBUFS); 1202194246Smarius } 1203194246Smarius 1204194246Smarius cflags = 0; 1205194246Smarius if (((*m_head)->m_pkthdr.csum_flags & CAS_CSUM_FEATURES) != 0) { 1206194246Smarius if (M_WRITABLE(*m_head) == 0) { 1207243857Sglebius m = m_dup(*m_head, M_NOWAIT); 1208194246Smarius m_freem(*m_head); 1209194246Smarius *m_head = m; 1210194246Smarius if (m == NULL) 1211194246Smarius return (ENOBUFS); 1212194246Smarius } 1213194246Smarius offset = sizeof(struct ether_header); 1214194246Smarius m = m_pullup(*m_head, offset + sizeof(struct ip)); 1215194246Smarius if (m == NULL) { 1216194246Smarius *m_head = NULL; 1217194246Smarius return (ENOBUFS); 1218194246Smarius } 1219194246Smarius ip = (struct ip *)(mtod(m, caddr_t) + offset); 1220194246Smarius offset += (ip->ip_hl << 2); 1221194246Smarius cflags = (offset << CAS_TD_CKSUM_START_SHFT) | 1222194246Smarius ((offset + m->m_pkthdr.csum_data) << 1223194246Smarius CAS_TD_CKSUM_STUFF_SHFT) | CAS_TD_CKSUM_EN; 1224194246Smarius *m_head = m; 1225194246Smarius } 1226194246Smarius 1227194246Smarius error = bus_dmamap_load_mbuf_sg(sc->sc_tdmatag, txs->txs_dmamap, 1228194246Smarius *m_head, txsegs, &nsegs, BUS_DMA_NOWAIT); 1229194246Smarius if (error == EFBIG) { 1230243857Sglebius m = m_collapse(*m_head, M_NOWAIT, CAS_NTXSEGS); 1231194246Smarius if (m == NULL) { 1232194246Smarius m_freem(*m_head); 1233194246Smarius *m_head = NULL; 1234194246Smarius return (ENOBUFS); 1235194246Smarius } 1236194246Smarius *m_head = m; 1237194246Smarius error = bus_dmamap_load_mbuf_sg(sc->sc_tdmatag, 1238194246Smarius txs->txs_dmamap, *m_head, txsegs, &nsegs, 1239194246Smarius BUS_DMA_NOWAIT); 1240194246Smarius if (error != 0) { 1241194246Smarius m_freem(*m_head); 1242194246Smarius *m_head = NULL; 1243194246Smarius return (error); 1244194246Smarius } 1245194246Smarius } else if (error != 0) 1246194246Smarius return (error); 1247194246Smarius /* If nsegs is wrong then the stack is corrupt. */ 1248194246Smarius KASSERT(nsegs <= CAS_NTXSEGS, 1249194246Smarius ("%s: too many DMA segments (%d)", __func__, nsegs)); 1250194246Smarius if (nsegs == 0) { 1251194246Smarius m_freem(*m_head); 1252194246Smarius *m_head = NULL; 1253194246Smarius return (EIO); 1254194246Smarius } 1255194246Smarius 1256194246Smarius /* 1257194246Smarius * Ensure we have enough descriptors free to describe 1258194246Smarius * the packet. Note, we always reserve one descriptor 1259194246Smarius * at the end of the ring as a termination point, in 1260194246Smarius * order to prevent wrap-around. 1261194246Smarius */ 1262194246Smarius if (nsegs > sc->sc_txfree - 1) { 1263194246Smarius txs->txs_ndescs = 0; 1264194246Smarius bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 1265194246Smarius return (ENOBUFS); 1266194246Smarius } 1267194246Smarius 1268194246Smarius txs->txs_ndescs = nsegs; 1269194246Smarius txs->txs_firstdesc = sc->sc_txnext; 1270194246Smarius nexttx = txs->txs_firstdesc; 1271194246Smarius for (seg = 0; seg < nsegs; seg++, nexttx = CAS_NEXTTX(nexttx)) { 1272194246Smarius#ifdef CAS_DEBUG 1273194246Smarius CTR6(KTR_CAS, 1274194246Smarius "%s: mapping seg %d (txd %d), len %lx, addr %#lx (%#lx)", 1275194246Smarius __func__, seg, nexttx, txsegs[seg].ds_len, 1276194246Smarius txsegs[seg].ds_addr, htole64(txsegs[seg].ds_addr)); 1277194246Smarius#endif 1278194246Smarius sc->sc_txdescs[nexttx].cd_buf_ptr = 1279194246Smarius htole64(txsegs[seg].ds_addr); 1280194246Smarius KASSERT(txsegs[seg].ds_len < 1281194246Smarius CAS_TD_BUF_LEN_MASK >> CAS_TD_BUF_LEN_SHFT, 1282194246Smarius ("%s: segment size too large!", __func__)); 1283194246Smarius sc->sc_txdescs[nexttx].cd_flags = 1284194246Smarius htole64(txsegs[seg].ds_len << CAS_TD_BUF_LEN_SHFT); 1285194246Smarius txs->txs_lastdesc = nexttx; 1286194246Smarius } 1287194246Smarius 1288194246Smarius /* Set EOF on the last descriptor. */ 1289194246Smarius#ifdef CAS_DEBUG 1290194246Smarius CTR3(KTR_CAS, "%s: end of frame at segment %d, TX %d", 1291194246Smarius __func__, seg, nexttx); 1292194246Smarius#endif 1293194246Smarius sc->sc_txdescs[txs->txs_lastdesc].cd_flags |= 1294194246Smarius htole64(CAS_TD_END_OF_FRAME); 1295194246Smarius 1296194246Smarius /* Lastly set SOF on the first descriptor. */ 1297194246Smarius#ifdef CAS_DEBUG 1298194246Smarius CTR3(KTR_CAS, "%s: start of frame at segment %d, TX %d", 1299194246Smarius __func__, seg, nexttx); 1300194246Smarius#endif 1301194904Smarius if (sc->sc_txwin += nsegs > CAS_MAXTXFREE * 2 / 3) { 1302194246Smarius sc->sc_txwin = 0; 1303194246Smarius sc->sc_txdescs[txs->txs_firstdesc].cd_flags |= 1304194246Smarius htole64(cflags | CAS_TD_START_OF_FRAME | CAS_TD_INT_ME); 1305194246Smarius } else 1306194246Smarius sc->sc_txdescs[txs->txs_firstdesc].cd_flags |= 1307194246Smarius htole64(cflags | CAS_TD_START_OF_FRAME); 1308194246Smarius 1309194246Smarius /* Sync the DMA map. */ 1310194246Smarius bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 1311194246Smarius BUS_DMASYNC_PREWRITE); 1312194246Smarius 1313194246Smarius#ifdef CAS_DEBUG 1314194246Smarius CTR4(KTR_CAS, "%s: setting firstdesc=%d, lastdesc=%d, ndescs=%d", 1315194246Smarius __func__, txs->txs_firstdesc, txs->txs_lastdesc, 1316194246Smarius txs->txs_ndescs); 1317194246Smarius#endif 1318194246Smarius STAILQ_REMOVE_HEAD(&sc->sc_txfreeq, txs_q); 1319194246Smarius STAILQ_INSERT_TAIL(&sc->sc_txdirtyq, txs, txs_q); 1320194246Smarius txs->txs_mbuf = *m_head; 1321194246Smarius 1322194246Smarius sc->sc_txnext = CAS_NEXTTX(txs->txs_lastdesc); 1323194246Smarius sc->sc_txfree -= txs->txs_ndescs; 1324194246Smarius 1325194246Smarius return (0); 1326194246Smarius} 1327194246Smarius 1328194246Smariusstatic void 1329194246Smariuscas_init_regs(struct cas_softc *sc) 1330194246Smarius{ 1331194246Smarius int i; 1332194246Smarius const u_char *laddr = IF_LLADDR(sc->sc_ifp); 1333194246Smarius 1334194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1335194246Smarius 1336194246Smarius /* These registers are not cleared on reset. */ 1337194246Smarius if ((sc->sc_flags & CAS_INITED) == 0) { 1338194246Smarius /* magic values */ 1339194246Smarius CAS_WRITE_4(sc, CAS_MAC_IPG0, 0); 1340194246Smarius CAS_WRITE_4(sc, CAS_MAC_IPG1, 8); 1341194246Smarius CAS_WRITE_4(sc, CAS_MAC_IPG2, 4); 1342194246Smarius 1343194246Smarius /* min frame length */ 1344194246Smarius CAS_WRITE_4(sc, CAS_MAC_MIN_FRAME, ETHER_MIN_LEN); 1345194246Smarius /* max frame length and max burst size */ 1346194246Smarius CAS_WRITE_4(sc, CAS_MAC_MAX_BF, 1347194246Smarius ((ETHER_MAX_LEN_JUMBO + ETHER_VLAN_ENCAP_LEN) << 1348194246Smarius CAS_MAC_MAX_BF_FRM_SHFT) | 1349194246Smarius (0x2000 << CAS_MAC_MAX_BF_BST_SHFT)); 1350194246Smarius 1351194246Smarius /* more magic values */ 1352194246Smarius CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x7); 1353194246Smarius CAS_WRITE_4(sc, CAS_MAC_JAM_SIZE, 0x4); 1354194246Smarius CAS_WRITE_4(sc, CAS_MAC_ATTEMPT_LIMIT, 0x10); 1355215721Smarius CAS_WRITE_4(sc, CAS_MAC_CTRL_TYPE, 0x8808); 1356194246Smarius 1357194246Smarius /* random number seed */ 1358194246Smarius CAS_WRITE_4(sc, CAS_MAC_RANDOM_SEED, 1359194246Smarius ((laddr[5] << 8) | laddr[4]) & 0x3ff); 1360194246Smarius 1361194246Smarius /* secondary MAC addresses: 0:0:0:0:0:0 */ 1362194246Smarius for (i = CAS_MAC_ADDR3; i <= CAS_MAC_ADDR41; 1363194246Smarius i += CAS_MAC_ADDR4 - CAS_MAC_ADDR3) 1364194246Smarius CAS_WRITE_4(sc, i, 0); 1365194246Smarius 1366194246Smarius /* MAC control address: 01:80:c2:00:00:01 */ 1367194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR42, 0x0001); 1368194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR43, 0xc200); 1369194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR44, 0x0180); 1370194246Smarius 1371194246Smarius /* MAC filter address: 0:0:0:0:0:0 */ 1372194246Smarius CAS_WRITE_4(sc, CAS_MAC_AFILTER0, 0); 1373194246Smarius CAS_WRITE_4(sc, CAS_MAC_AFILTER1, 0); 1374194246Smarius CAS_WRITE_4(sc, CAS_MAC_AFILTER2, 0); 1375194246Smarius CAS_WRITE_4(sc, CAS_MAC_AFILTER_MASK1_2, 0); 1376194246Smarius CAS_WRITE_4(sc, CAS_MAC_AFILTER_MASK0, 0); 1377194246Smarius 1378194246Smarius /* Zero the hash table. */ 1379194246Smarius for (i = CAS_MAC_HASH0; i <= CAS_MAC_HASH15; 1380194246Smarius i += CAS_MAC_HASH1 - CAS_MAC_HASH0) 1381194246Smarius CAS_WRITE_4(sc, i, 0); 1382194246Smarius 1383194246Smarius sc->sc_flags |= CAS_INITED; 1384194246Smarius } 1385194246Smarius 1386194246Smarius /* Counters need to be zeroed. */ 1387194246Smarius CAS_WRITE_4(sc, CAS_MAC_NORM_COLL_CNT, 0); 1388194246Smarius CAS_WRITE_4(sc, CAS_MAC_FIRST_COLL_CNT, 0); 1389194246Smarius CAS_WRITE_4(sc, CAS_MAC_EXCESS_COLL_CNT, 0); 1390194246Smarius CAS_WRITE_4(sc, CAS_MAC_LATE_COLL_CNT, 0); 1391194246Smarius CAS_WRITE_4(sc, CAS_MAC_DEFER_TMR_CNT, 0); 1392194246Smarius CAS_WRITE_4(sc, CAS_MAC_PEAK_ATTEMPTS, 0); 1393194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_FRAME_COUNT, 0); 1394194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_LEN_ERR_CNT, 0); 1395194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_ALIGN_ERR, 0); 1396194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CRC_ERR_CNT, 0); 1397194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CODE_VIOL, 0); 1398194246Smarius 1399194246Smarius /* Set XOFF PAUSE time. */ 1400194246Smarius CAS_WRITE_4(sc, CAS_MAC_SPC, 0x1BF0 << CAS_MAC_SPC_TIME_SHFT); 1401194246Smarius 1402194246Smarius /* Set the station address. */ 1403194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR0, (laddr[4] << 8) | laddr[5]); 1404194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR1, (laddr[2] << 8) | laddr[3]); 1405194246Smarius CAS_WRITE_4(sc, CAS_MAC_ADDR2, (laddr[0] << 8) | laddr[1]); 1406194246Smarius 1407194246Smarius /* Enable MII outputs. */ 1408194246Smarius CAS_WRITE_4(sc, CAS_MAC_XIF_CONF, CAS_MAC_XIF_CONF_TX_OE); 1409194246Smarius} 1410194246Smarius 1411194246Smariusstatic void 1412194904Smariuscas_tx_task(void *arg, int pending __unused) 1413194246Smarius{ 1414194904Smarius struct ifnet *ifp; 1415194246Smarius 1416194904Smarius ifp = (struct ifnet *)arg; 1417194904Smarius cas_start(ifp); 1418194246Smarius} 1419194246Smarius 1420194246Smariusstatic inline void 1421194246Smariuscas_txkick(struct cas_softc *sc) 1422194246Smarius{ 1423194246Smarius 1424194246Smarius /* 1425194246Smarius * Update the TX kick register. This register has to point to the 1426194246Smarius * descriptor after the last valid one and for optimum performance 1427194246Smarius * should be incremented in multiples of 4 (the DMA engine fetches/ 1428194246Smarius * updates descriptors in batches of 4). 1429194246Smarius */ 1430194246Smarius#ifdef CAS_DEBUG 1431194246Smarius CTR3(KTR_CAS, "%s: %s: kicking TX %d", 1432194246Smarius device_get_name(sc->sc_dev), __func__, sc->sc_txnext); 1433194246Smarius#endif 1434194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1435194246Smarius CAS_WRITE_4(sc, CAS_TX_KICK3, sc->sc_txnext); 1436194246Smarius} 1437194246Smarius 1438194246Smariusstatic void 1439194904Smariuscas_start(struct ifnet *ifp) 1440194246Smarius{ 1441194246Smarius struct cas_softc *sc = ifp->if_softc; 1442194246Smarius struct mbuf *m; 1443194246Smarius int kicked, ntx; 1444194246Smarius 1445194904Smarius CAS_LOCK(sc); 1446194246Smarius 1447194246Smarius if ((ifp->if_drv_flags & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) != 1448194904Smarius IFF_DRV_RUNNING || (sc->sc_flags & CAS_LINK) == 0) { 1449194904Smarius CAS_UNLOCK(sc); 1450194246Smarius return; 1451194904Smarius } 1452194246Smarius 1453194904Smarius if (sc->sc_txfree < CAS_MAXTXFREE / 4) 1454194904Smarius cas_tint(sc); 1455194904Smarius 1456194246Smarius#ifdef CAS_DEBUG 1457194246Smarius CTR4(KTR_CAS, "%s: %s: txfree %d, txnext %d", 1458194246Smarius device_get_name(sc->sc_dev), __func__, sc->sc_txfree, 1459194246Smarius sc->sc_txnext); 1460194246Smarius#endif 1461194246Smarius ntx = 0; 1462194246Smarius kicked = 0; 1463194246Smarius for (; !IFQ_DRV_IS_EMPTY(&ifp->if_snd) && sc->sc_txfree > 1;) { 1464194246Smarius IFQ_DRV_DEQUEUE(&ifp->if_snd, m); 1465194246Smarius if (m == NULL) 1466194246Smarius break; 1467194246Smarius if (cas_load_txmbuf(sc, &m) != 0) { 1468194246Smarius if (m == NULL) 1469194246Smarius break; 1470194246Smarius ifp->if_drv_flags |= IFF_DRV_OACTIVE; 1471194246Smarius IFQ_DRV_PREPEND(&ifp->if_snd, m); 1472194246Smarius break; 1473194246Smarius } 1474194246Smarius if ((sc->sc_txnext % 4) == 0) { 1475194246Smarius cas_txkick(sc); 1476194246Smarius kicked = 1; 1477194246Smarius } else 1478194246Smarius kicked = 0; 1479194246Smarius ntx++; 1480194246Smarius BPF_MTAP(ifp, m); 1481194246Smarius } 1482194246Smarius 1483194246Smarius if (ntx > 0) { 1484194246Smarius if (kicked == 0) 1485194246Smarius cas_txkick(sc); 1486194246Smarius#ifdef CAS_DEBUG 1487194246Smarius CTR2(KTR_CAS, "%s: packets enqueued, OWN on %d", 1488194246Smarius device_get_name(sc->sc_dev), sc->sc_txnext); 1489194246Smarius#endif 1490194246Smarius 1491194246Smarius /* Set a watchdog timer in case the chip flakes out. */ 1492194246Smarius sc->sc_wdog_timer = 5; 1493194246Smarius#ifdef CAS_DEBUG 1494194246Smarius CTR3(KTR_CAS, "%s: %s: watchdog %d", 1495194246Smarius device_get_name(sc->sc_dev), __func__, 1496194246Smarius sc->sc_wdog_timer); 1497194246Smarius#endif 1498194246Smarius } 1499194904Smarius 1500194904Smarius CAS_UNLOCK(sc); 1501194246Smarius} 1502194246Smarius 1503194246Smariusstatic void 1504194246Smariuscas_tint(struct cas_softc *sc) 1505194246Smarius{ 1506194246Smarius struct ifnet *ifp = sc->sc_ifp; 1507194246Smarius struct cas_txsoft *txs; 1508194246Smarius int progress; 1509194246Smarius uint32_t txlast; 1510194246Smarius#ifdef CAS_DEBUG 1511194246Smarius int i; 1512194246Smarius 1513194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1514194246Smarius 1515194246Smarius CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 1516194246Smarius#endif 1517194246Smarius 1518194246Smarius /* 1519194246Smarius * Go through our TX list and free mbufs for those 1520194246Smarius * frames that have been transmitted. 1521194246Smarius */ 1522194246Smarius progress = 0; 1523194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD); 1524194246Smarius while ((txs = STAILQ_FIRST(&sc->sc_txdirtyq)) != NULL) { 1525194246Smarius#ifdef CAS_DEBUG 1526194246Smarius if ((ifp->if_flags & IFF_DEBUG) != 0) { 1527194246Smarius printf(" txsoft %p transmit chain:\n", txs); 1528194246Smarius for (i = txs->txs_firstdesc;; i = CAS_NEXTTX(i)) { 1529194246Smarius printf("descriptor %d: ", i); 1530194246Smarius printf("cd_flags: 0x%016llx\t", 1531194246Smarius (long long)le64toh( 1532194246Smarius sc->sc_txdescs[i].cd_flags)); 1533194246Smarius printf("cd_buf_ptr: 0x%016llx\n", 1534194246Smarius (long long)le64toh( 1535194246Smarius sc->sc_txdescs[i].cd_buf_ptr)); 1536194246Smarius if (i == txs->txs_lastdesc) 1537194246Smarius break; 1538194246Smarius } 1539194246Smarius } 1540194246Smarius#endif 1541194246Smarius 1542194246Smarius /* 1543194246Smarius * In theory, we could harvest some descriptors before 1544194246Smarius * the ring is empty, but that's a bit complicated. 1545194246Smarius * 1546194246Smarius * CAS_TX_COMPn points to the last descriptor 1547194246Smarius * processed + 1. 1548194246Smarius */ 1549194246Smarius txlast = CAS_READ_4(sc, CAS_TX_COMP3); 1550194246Smarius#ifdef CAS_DEBUG 1551194246Smarius CTR4(KTR_CAS, "%s: txs->txs_firstdesc = %d, " 1552194246Smarius "txs->txs_lastdesc = %d, txlast = %d", 1553194246Smarius __func__, txs->txs_firstdesc, txs->txs_lastdesc, txlast); 1554194246Smarius#endif 1555194246Smarius if (txs->txs_firstdesc <= txs->txs_lastdesc) { 1556194246Smarius if ((txlast >= txs->txs_firstdesc) && 1557194246Smarius (txlast <= txs->txs_lastdesc)) 1558194246Smarius break; 1559194246Smarius } else { 1560194246Smarius /* Ick -- this command wraps. */ 1561194246Smarius if ((txlast >= txs->txs_firstdesc) || 1562194246Smarius (txlast <= txs->txs_lastdesc)) 1563194246Smarius break; 1564194246Smarius } 1565194246Smarius 1566194246Smarius#ifdef CAS_DEBUG 1567194246Smarius CTR1(KTR_CAS, "%s: releasing a descriptor", __func__); 1568194246Smarius#endif 1569194246Smarius STAILQ_REMOVE_HEAD(&sc->sc_txdirtyq, txs_q); 1570194246Smarius 1571194246Smarius sc->sc_txfree += txs->txs_ndescs; 1572194246Smarius 1573194246Smarius bus_dmamap_sync(sc->sc_tdmatag, txs->txs_dmamap, 1574194246Smarius BUS_DMASYNC_POSTWRITE); 1575194246Smarius bus_dmamap_unload(sc->sc_tdmatag, txs->txs_dmamap); 1576194246Smarius if (txs->txs_mbuf != NULL) { 1577194246Smarius m_freem(txs->txs_mbuf); 1578194246Smarius txs->txs_mbuf = NULL; 1579194246Smarius } 1580194246Smarius 1581194246Smarius STAILQ_INSERT_TAIL(&sc->sc_txfreeq, txs, txs_q); 1582194246Smarius 1583194246Smarius ifp->if_opackets++; 1584194246Smarius progress = 1; 1585194246Smarius } 1586194246Smarius 1587194246Smarius#ifdef CAS_DEBUG 1588215721Smarius CTR5(KTR_CAS, "%s: CAS_TX_SM1 %x CAS_TX_SM2 %x CAS_TX_DESC_BASE %llx " 1589194246Smarius "CAS_TX_COMP3 %x", 1590215721Smarius __func__, CAS_READ_4(sc, CAS_TX_SM1), CAS_READ_4(sc, CAS_TX_SM2), 1591215721Smarius ((long long)CAS_READ_4(sc, CAS_TX_DESC3_BASE_HI) << 32) | 1592215721Smarius CAS_READ_4(sc, CAS_TX_DESC3_BASE_LO), 1593194246Smarius CAS_READ_4(sc, CAS_TX_COMP3)); 1594194246Smarius#endif 1595194246Smarius 1596194246Smarius if (progress) { 1597194904Smarius /* We freed some descriptors, so reset IFF_DRV_OACTIVE. */ 1598194246Smarius ifp->if_drv_flags &= ~IFF_DRV_OACTIVE; 1599194246Smarius if (STAILQ_EMPTY(&sc->sc_txdirtyq)) 1600194246Smarius sc->sc_wdog_timer = 0; 1601194246Smarius } 1602194246Smarius 1603194246Smarius#ifdef CAS_DEBUG 1604194246Smarius CTR3(KTR_CAS, "%s: %s: watchdog %d", 1605194246Smarius device_get_name(sc->sc_dev), __func__, sc->sc_wdog_timer); 1606194246Smarius#endif 1607194246Smarius} 1608194246Smarius 1609194246Smariusstatic void 1610194246Smariuscas_rint_timeout(void *arg) 1611194246Smarius{ 1612194246Smarius struct cas_softc *sc = arg; 1613194246Smarius 1614223986Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1615194246Smarius 1616194246Smarius cas_rint(sc); 1617194246Smarius} 1618194246Smarius 1619194246Smariusstatic void 1620194246Smariuscas_rint(struct cas_softc *sc) 1621194246Smarius{ 1622194246Smarius struct cas_rxdsoft *rxds, *rxds2; 1623194246Smarius struct ifnet *ifp = sc->sc_ifp; 1624194246Smarius struct mbuf *m, *m2; 1625194246Smarius uint64_t word1, word2, word3, word4; 1626194246Smarius uint32_t rxhead; 1627194246Smarius u_int idx, idx2, len, off, skip; 1628194246Smarius 1629223986Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1630194246Smarius 1631194246Smarius callout_stop(&sc->sc_rx_ch); 1632194246Smarius 1633194246Smarius#ifdef CAS_DEBUG 1634194246Smarius CTR2(KTR_CAS, "%s: %s", device_get_name(sc->sc_dev), __func__); 1635194246Smarius#endif 1636194246Smarius 1637194246Smarius#define PRINTWORD(n, delimiter) \ 1638194246Smarius printf("word ## n: 0x%016llx%c", (long long)word ## n, delimiter) 1639194246Smarius 1640194246Smarius#define SKIPASSERT(n) \ 1641194246Smarius KASSERT(sc->sc_rxcomps[sc->sc_rxcptr].crc_word ## n == 0, \ 1642194246Smarius ("%s: word ## n not 0", __func__)) 1643194246Smarius 1644194246Smarius#define WORDTOH(n) \ 1645194246Smarius word ## n = le64toh(sc->sc_rxcomps[sc->sc_rxcptr].crc_word ## n) 1646194246Smarius 1647194246Smarius /* 1648194246Smarius * Read the completion head register once. This limits 1649194246Smarius * how long the following loop can execute. 1650194246Smarius */ 1651194246Smarius rxhead = CAS_READ_4(sc, CAS_RX_COMP_HEAD); 1652194246Smarius#ifdef CAS_DEBUG 1653194246Smarius CTR4(KTR_CAS, "%s: sc->sc_rxcptr %d, sc->sc_rxdptr %d, head %d", 1654215721Smarius __func__, sc->sc_rxcptr, sc->sc_rxdptr, rxhead); 1655194246Smarius#endif 1656194246Smarius skip = 0; 1657194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); 1658194246Smarius for (; sc->sc_rxcptr != rxhead; 1659194246Smarius sc->sc_rxcptr = CAS_NEXTRXCOMP(sc->sc_rxcptr)) { 1660194246Smarius if (skip != 0) { 1661194246Smarius SKIPASSERT(1); 1662194246Smarius SKIPASSERT(2); 1663194246Smarius SKIPASSERT(3); 1664194246Smarius 1665194246Smarius --skip; 1666194246Smarius goto skip; 1667194246Smarius } 1668194246Smarius 1669194246Smarius WORDTOH(1); 1670194246Smarius WORDTOH(2); 1671194246Smarius WORDTOH(3); 1672194246Smarius WORDTOH(4); 1673194246Smarius 1674194246Smarius#ifdef CAS_DEBUG 1675194246Smarius if ((ifp->if_flags & IFF_DEBUG) != 0) { 1676194246Smarius printf(" completion %d: ", sc->sc_rxcptr); 1677194246Smarius PRINTWORD(1, '\t'); 1678194246Smarius PRINTWORD(2, '\t'); 1679194246Smarius PRINTWORD(3, '\t'); 1680194246Smarius PRINTWORD(4, '\n'); 1681194246Smarius } 1682194246Smarius#endif 1683194246Smarius 1684194246Smarius if (__predict_false( 1685194246Smarius (word1 & CAS_RC1_TYPE_MASK) == CAS_RC1_TYPE_HW || 1686194246Smarius (word4 & CAS_RC4_ZERO) != 0)) { 1687194246Smarius /* 1688194246Smarius * The descriptor is still marked as owned, although 1689194246Smarius * it is supposed to have completed. This has been 1690194246Smarius * observed on some machines. Just exiting here 1691194246Smarius * might leave the packet sitting around until another 1692194246Smarius * one arrives to trigger a new interrupt, which is 1693194246Smarius * generally undesirable, so set up a timeout. 1694194246Smarius */ 1695194246Smarius callout_reset(&sc->sc_rx_ch, CAS_RXOWN_TICKS, 1696194246Smarius cas_rint_timeout, sc); 1697194246Smarius break; 1698194246Smarius } 1699194246Smarius 1700194246Smarius if (__predict_false( 1701194246Smarius (word4 & (CAS_RC4_BAD | CAS_RC4_LEN_MMATCH)) != 0)) { 1702194246Smarius ifp->if_ierrors++; 1703194246Smarius device_printf(sc->sc_dev, 1704194246Smarius "receive error: CRC error\n"); 1705194246Smarius continue; 1706194246Smarius } 1707194246Smarius 1708194246Smarius KASSERT(CAS_GET(word1, CAS_RC1_DATA_SIZE) == 0 || 1709194246Smarius CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0, 1710194246Smarius ("%s: data and header present", __func__)); 1711194246Smarius KASSERT((word1 & CAS_RC1_SPLIT_PKT) == 0 || 1712194246Smarius CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0, 1713194246Smarius ("%s: split and header present", __func__)); 1714194246Smarius KASSERT(CAS_GET(word1, CAS_RC1_DATA_SIZE) == 0 || 1715194246Smarius (word1 & CAS_RC1_RELEASE_HDR) == 0, 1716194246Smarius ("%s: data present but header release", __func__)); 1717194246Smarius KASSERT(CAS_GET(word2, CAS_RC2_HDR_SIZE) == 0 || 1718194246Smarius (word1 & CAS_RC1_RELEASE_DATA) == 0, 1719194246Smarius ("%s: header present but data release", __func__)); 1720194246Smarius 1721194246Smarius if ((len = CAS_GET(word2, CAS_RC2_HDR_SIZE)) != 0) { 1722194246Smarius idx = CAS_GET(word2, CAS_RC2_HDR_INDEX); 1723194246Smarius off = CAS_GET(word2, CAS_RC2_HDR_OFF); 1724194246Smarius#ifdef CAS_DEBUG 1725194246Smarius CTR4(KTR_CAS, "%s: hdr at idx %d, off %d, len %d", 1726194246Smarius __func__, idx, off, len); 1727194246Smarius#endif 1728194246Smarius rxds = &sc->sc_rxdsoft[idx]; 1729243857Sglebius MGETHDR(m, M_NOWAIT, MT_DATA); 1730194246Smarius if (m != NULL) { 1731194246Smarius refcount_acquire(&rxds->rxds_refcount); 1732194246Smarius bus_dmamap_sync(sc->sc_rdmatag, 1733194246Smarius rxds->rxds_dmamap, BUS_DMASYNC_POSTREAD); 1734194973Smarius#if __FreeBSD_version < 800016 1735194246Smarius MEXTADD(m, (caddr_t)rxds->rxds_buf + 1736194246Smarius off * 256 + ETHER_ALIGN, len, cas_free, 1737194973Smarius rxds, M_RDONLY, EXT_NET_DRV); 1738194246Smarius#else 1739194973Smarius MEXTADD(m, (caddr_t)rxds->rxds_buf + 1740194973Smarius off * 256 + ETHER_ALIGN, len, cas_free, 1741194246Smarius sc, (void *)(uintptr_t)idx, 1742194973Smarius M_RDONLY, EXT_NET_DRV); 1743194246Smarius#endif 1744194246Smarius if ((m->m_flags & M_EXT) == 0) { 1745194246Smarius m_freem(m); 1746194246Smarius m = NULL; 1747194246Smarius } 1748194246Smarius } 1749194246Smarius if (m != NULL) { 1750194246Smarius m->m_pkthdr.rcvif = ifp; 1751194246Smarius m->m_pkthdr.len = m->m_len = len; 1752194246Smarius ifp->if_ipackets++; 1753194246Smarius if ((ifp->if_capenable & IFCAP_RXCSUM) != 0) 1754194246Smarius cas_rxcksum(m, CAS_GET(word4, 1755194246Smarius CAS_RC4_TCP_CSUM)); 1756194246Smarius /* Pass it on. */ 1757223986Smarius CAS_UNLOCK(sc); 1758194246Smarius (*ifp->if_input)(ifp, m); 1759223986Smarius CAS_LOCK(sc); 1760194246Smarius } else 1761223951Smarius ifp->if_iqdrops++; 1762194246Smarius 1763194246Smarius if ((word1 & CAS_RC1_RELEASE_HDR) != 0 && 1764194246Smarius refcount_release(&rxds->rxds_refcount) != 0) 1765194246Smarius cas_add_rxdesc(sc, idx); 1766194246Smarius } else if ((len = CAS_GET(word1, CAS_RC1_DATA_SIZE)) != 0) { 1767194246Smarius idx = CAS_GET(word1, CAS_RC1_DATA_INDEX); 1768194246Smarius off = CAS_GET(word1, CAS_RC1_DATA_OFF); 1769194246Smarius#ifdef CAS_DEBUG 1770194246Smarius CTR4(KTR_CAS, "%s: data at idx %d, off %d, len %d", 1771194246Smarius __func__, idx, off, len); 1772194246Smarius#endif 1773194246Smarius rxds = &sc->sc_rxdsoft[idx]; 1774243857Sglebius MGETHDR(m, M_NOWAIT, MT_DATA); 1775194246Smarius if (m != NULL) { 1776194246Smarius refcount_acquire(&rxds->rxds_refcount); 1777194246Smarius off += ETHER_ALIGN; 1778194246Smarius m->m_len = min(CAS_PAGE_SIZE - off, len); 1779194246Smarius bus_dmamap_sync(sc->sc_rdmatag, 1780194246Smarius rxds->rxds_dmamap, BUS_DMASYNC_POSTREAD); 1781194973Smarius#if __FreeBSD_version < 800016 1782194246Smarius MEXTADD(m, (caddr_t)rxds->rxds_buf + off, 1783194973Smarius m->m_len, cas_free, rxds, M_RDONLY, 1784194973Smarius EXT_NET_DRV); 1785194246Smarius#else 1786194973Smarius MEXTADD(m, (caddr_t)rxds->rxds_buf + off, 1787194973Smarius m->m_len, cas_free, sc, 1788194973Smarius (void *)(uintptr_t)idx, M_RDONLY, 1789194973Smarius EXT_NET_DRV); 1790194246Smarius#endif 1791194246Smarius if ((m->m_flags & M_EXT) == 0) { 1792194246Smarius m_freem(m); 1793194246Smarius m = NULL; 1794194246Smarius } 1795194246Smarius } 1796194246Smarius idx2 = 0; 1797208776Smarius m2 = NULL; 1798194246Smarius rxds2 = NULL; 1799194246Smarius if ((word1 & CAS_RC1_SPLIT_PKT) != 0) { 1800194246Smarius KASSERT((word1 & CAS_RC1_RELEASE_NEXT) != 0, 1801194246Smarius ("%s: split but no release next", 1802194246Smarius __func__)); 1803194246Smarius 1804194246Smarius idx2 = CAS_GET(word2, CAS_RC2_NEXT_INDEX); 1805194246Smarius#ifdef CAS_DEBUG 1806194246Smarius CTR2(KTR_CAS, "%s: split at idx %d", 1807194246Smarius __func__, idx2); 1808194246Smarius#endif 1809194246Smarius rxds2 = &sc->sc_rxdsoft[idx2]; 1810208776Smarius if (m != NULL) { 1811243857Sglebius MGET(m2, M_NOWAIT, MT_DATA); 1812208776Smarius if (m2 != NULL) { 1813208776Smarius refcount_acquire( 1814208776Smarius &rxds2->rxds_refcount); 1815208776Smarius m2->m_len = len - m->m_len; 1816208776Smarius bus_dmamap_sync( 1817208776Smarius sc->sc_rdmatag, 1818208776Smarius rxds2->rxds_dmamap, 1819208776Smarius BUS_DMASYNC_POSTREAD); 1820194973Smarius#if __FreeBSD_version < 800016 1821208776Smarius MEXTADD(m2, 1822208776Smarius (caddr_t)rxds2->rxds_buf, 1823208776Smarius m2->m_len, cas_free, 1824208776Smarius rxds2, M_RDONLY, 1825208776Smarius EXT_NET_DRV); 1826194973Smarius#else 1827208776Smarius MEXTADD(m2, 1828208776Smarius (caddr_t)rxds2->rxds_buf, 1829208776Smarius m2->m_len, cas_free, sc, 1830208776Smarius (void *)(uintptr_t)idx2, 1831208776Smarius M_RDONLY, EXT_NET_DRV); 1832194246Smarius#endif 1833208776Smarius if ((m2->m_flags & M_EXT) == 1834208776Smarius 0) { 1835208776Smarius m_freem(m2); 1836208776Smarius m2 = NULL; 1837208776Smarius } 1838194246Smarius } 1839194246Smarius } 1840194246Smarius if (m2 != NULL) 1841194246Smarius m->m_next = m2; 1842208776Smarius else if (m != NULL) { 1843194246Smarius m_freem(m); 1844194246Smarius m = NULL; 1845194246Smarius } 1846194246Smarius } 1847194246Smarius if (m != NULL) { 1848194246Smarius m->m_pkthdr.rcvif = ifp; 1849194246Smarius m->m_pkthdr.len = len; 1850194246Smarius ifp->if_ipackets++; 1851194246Smarius if ((ifp->if_capenable & IFCAP_RXCSUM) != 0) 1852194246Smarius cas_rxcksum(m, CAS_GET(word4, 1853194246Smarius CAS_RC4_TCP_CSUM)); 1854194246Smarius /* Pass it on. */ 1855223986Smarius CAS_UNLOCK(sc); 1856194246Smarius (*ifp->if_input)(ifp, m); 1857223986Smarius CAS_LOCK(sc); 1858194246Smarius } else 1859223951Smarius ifp->if_iqdrops++; 1860194246Smarius 1861194246Smarius if ((word1 & CAS_RC1_RELEASE_DATA) != 0 && 1862194246Smarius refcount_release(&rxds->rxds_refcount) != 0) 1863194246Smarius cas_add_rxdesc(sc, idx); 1864194246Smarius if ((word1 & CAS_RC1_SPLIT_PKT) != 0 && 1865194246Smarius refcount_release(&rxds2->rxds_refcount) != 0) 1866194246Smarius cas_add_rxdesc(sc, idx2); 1867194246Smarius } 1868194246Smarius 1869194246Smarius skip = CAS_GET(word1, CAS_RC1_SKIP); 1870194246Smarius 1871194246Smarius skip: 1872194246Smarius cas_rxcompinit(&sc->sc_rxcomps[sc->sc_rxcptr]); 1873194904Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0) 1874194904Smarius break; 1875194246Smarius } 1876194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1877194246Smarius CAS_WRITE_4(sc, CAS_RX_COMP_TAIL, sc->sc_rxcptr); 1878194246Smarius 1879194246Smarius#undef PRINTWORD 1880194246Smarius#undef SKIPASSERT 1881194246Smarius#undef WORDTOH 1882194246Smarius 1883194246Smarius#ifdef CAS_DEBUG 1884194246Smarius CTR4(KTR_CAS, "%s: done sc->sc_rxcptr %d, sc->sc_rxdptr %d, head %d", 1885215721Smarius __func__, sc->sc_rxcptr, sc->sc_rxdptr, 1886194246Smarius CAS_READ_4(sc, CAS_RX_COMP_HEAD)); 1887194246Smarius#endif 1888194246Smarius} 1889194246Smarius 1890254842Sandrestatic int 1891254799Sandrecas_free(struct mbuf *m, void *arg1, void *arg2) 1892194246Smarius{ 1893194246Smarius struct cas_rxdsoft *rxds; 1894194246Smarius struct cas_softc *sc; 1895223986Smarius u_int idx, locked; 1896194246Smarius 1897194246Smarius#if __FreeBSD_version < 800016 1898194246Smarius rxds = arg2; 1899194246Smarius sc = rxds->rxds_sc; 1900194246Smarius idx = rxds->rxds_idx; 1901194246Smarius#else 1902194246Smarius sc = arg1; 1903194246Smarius idx = (uintptr_t)arg2; 1904194246Smarius rxds = &sc->sc_rxdsoft[idx]; 1905194246Smarius#endif 1906194246Smarius if (refcount_release(&rxds->rxds_refcount) == 0) 1907254842Sandre return (EXT_FREE_OK); 1908194246Smarius 1909194246Smarius /* 1910194246Smarius * NB: this function can be called via m_freem(9) within 1911194246Smarius * this driver! 1912194246Smarius */ 1913223986Smarius if ((locked = CAS_LOCK_OWNED(sc)) == 0) 1914223986Smarius CAS_LOCK(sc); 1915194246Smarius cas_add_rxdesc(sc, idx); 1916223986Smarius if (locked == 0) 1917223986Smarius CAS_UNLOCK(sc); 1918254842Sandre return (EXT_FREE_OK); 1919194246Smarius} 1920194246Smarius 1921194246Smariusstatic inline void 1922194246Smariuscas_add_rxdesc(struct cas_softc *sc, u_int idx) 1923194246Smarius{ 1924194246Smarius 1925223986Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1926194246Smarius 1927194246Smarius bus_dmamap_sync(sc->sc_rdmatag, sc->sc_rxdsoft[idx].rxds_dmamap, 1928194246Smarius BUS_DMASYNC_PREREAD); 1929194246Smarius CAS_UPDATE_RXDESC(sc, sc->sc_rxdptr, idx); 1930194246Smarius sc->sc_rxdptr = CAS_NEXTRXDESC(sc->sc_rxdptr); 1931194246Smarius 1932194246Smarius /* 1933194246Smarius * Update the RX kick register. This register has to point to the 1934194246Smarius * descriptor after the last valid one (before the current batch) 1935194246Smarius * and for optimum performance should be incremented in multiples 1936194246Smarius * of 4 (the DMA engine fetches/updates descriptors in batches of 4). 1937194246Smarius */ 1938194246Smarius if ((sc->sc_rxdptr % 4) == 0) { 1939194246Smarius CAS_CDSYNC(sc, BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); 1940194246Smarius CAS_WRITE_4(sc, CAS_RX_KICK, 1941194246Smarius (sc->sc_rxdptr + CAS_NRXDESC - 4) & CAS_NRXDESC_MASK); 1942194246Smarius } 1943194246Smarius} 1944194246Smarius 1945194246Smariusstatic void 1946194246Smariuscas_eint(struct cas_softc *sc, u_int status) 1947194246Smarius{ 1948194904Smarius struct ifnet *ifp = sc->sc_ifp; 1949194246Smarius 1950223986Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 1951194246Smarius 1952194904Smarius ifp->if_ierrors++; 1953194904Smarius 1954194246Smarius device_printf(sc->sc_dev, "%s: status 0x%x", __func__, status); 1955194246Smarius if ((status & CAS_INTR_PCI_ERROR_INT) != 0) { 1956194246Smarius status = CAS_READ_4(sc, CAS_ERROR_STATUS); 1957194246Smarius printf(", PCI bus error 0x%x", status); 1958194246Smarius if ((status & CAS_ERROR_OTHER) != 0) { 1959194246Smarius status = pci_read_config(sc->sc_dev, PCIR_STATUS, 2); 1960194246Smarius printf(", PCI status 0x%x", status); 1961194246Smarius pci_write_config(sc->sc_dev, PCIR_STATUS, status, 2); 1962194246Smarius } 1963194246Smarius } 1964194246Smarius printf("\n"); 1965194246Smarius 1966194904Smarius ifp->if_drv_flags &= ~IFF_DRV_RUNNING; 1967223986Smarius cas_init_locked(sc); 1968194904Smarius if (!IFQ_DRV_IS_EMPTY(&ifp->if_snd)) 1969194904Smarius taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 1970194246Smarius} 1971194246Smarius 1972194904Smariusstatic int 1973194246Smariuscas_intr(void *v) 1974194246Smarius{ 1975194246Smarius struct cas_softc *sc = v; 1976194904Smarius 1977194904Smarius if (__predict_false((CAS_READ_4(sc, CAS_STATUS_ALIAS) & 1978194904Smarius CAS_INTR_SUMMARY) == 0)) 1979194904Smarius return (FILTER_STRAY); 1980194904Smarius 1981194904Smarius /* Disable interrupts. */ 1982194904Smarius CAS_WRITE_4(sc, CAS_INTMASK, 0xffffffff); 1983194904Smarius taskqueue_enqueue(sc->sc_tq, &sc->sc_intr_task); 1984194904Smarius 1985194904Smarius return (FILTER_HANDLED); 1986194904Smarius} 1987194904Smarius 1988194904Smariusstatic void 1989194904Smariuscas_intr_task(void *arg, int pending __unused) 1990194904Smarius{ 1991194904Smarius struct cas_softc *sc = arg; 1992194904Smarius struct ifnet *ifp = sc->sc_ifp; 1993194246Smarius uint32_t status, status2; 1994194246Smarius 1995194904Smarius CAS_LOCK_ASSERT(sc, MA_NOTOWNED); 1996194904Smarius 1997194904Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0) 1998194904Smarius return; 1999194904Smarius 2000194246Smarius status = CAS_READ_4(sc, CAS_STATUS); 2001194246Smarius if (__predict_false((status & CAS_INTR_SUMMARY) == 0)) 2002194904Smarius goto done; 2003194246Smarius 2004223986Smarius CAS_LOCK(sc); 2005194246Smarius#ifdef CAS_DEBUG 2006194246Smarius CTR4(KTR_CAS, "%s: %s: cplt %x, status %x", 2007194246Smarius device_get_name(sc->sc_dev), __func__, 2008215721Smarius (status >> CAS_STATUS_TX_COMP3_SHFT), (u_int)status); 2009194246Smarius 2010194246Smarius /* 2011194246Smarius * PCS interrupts must be cleared, otherwise no traffic is passed! 2012194246Smarius */ 2013194246Smarius if ((status & CAS_INTR_PCS_INT) != 0) { 2014194246Smarius status2 = 2015194246Smarius CAS_READ_4(sc, CAS_PCS_INTR_STATUS) | 2016194246Smarius CAS_READ_4(sc, CAS_PCS_INTR_STATUS); 2017194246Smarius if ((status2 & CAS_PCS_INTR_LINK) != 0) 2018194246Smarius device_printf(sc->sc_dev, 2019194246Smarius "%s: PCS link status changed\n", __func__); 2020194246Smarius } 2021194246Smarius if ((status & CAS_MAC_CTRL_STATUS) != 0) { 2022194246Smarius status2 = CAS_READ_4(sc, CAS_MAC_CTRL_STATUS); 2023194246Smarius if ((status2 & CAS_MAC_CTRL_PAUSE) != 0) 2024194246Smarius device_printf(sc->sc_dev, 2025194246Smarius "%s: PAUSE received (PAUSE time %d slots)\n", 2026194246Smarius __func__, 2027194246Smarius (status2 & CAS_MAC_CTRL_STATUS_PT_MASK) >> 2028194246Smarius CAS_MAC_CTRL_STATUS_PT_SHFT); 2029194246Smarius if ((status2 & CAS_MAC_CTRL_PAUSE) != 0) 2030194246Smarius device_printf(sc->sc_dev, 2031194246Smarius "%s: transited to PAUSE state\n", __func__); 2032194246Smarius if ((status2 & CAS_MAC_CTRL_NON_PAUSE) != 0) 2033194246Smarius device_printf(sc->sc_dev, 2034194246Smarius "%s: transited to non-PAUSE state\n", __func__); 2035194246Smarius } 2036194246Smarius if ((status & CAS_INTR_MIF) != 0) 2037194246Smarius device_printf(sc->sc_dev, "%s: MIF interrupt\n", __func__); 2038194246Smarius#endif 2039194246Smarius 2040194246Smarius if (__predict_false((status & 2041194246Smarius (CAS_INTR_TX_TAG_ERR | CAS_INTR_RX_TAG_ERR | 2042194246Smarius CAS_INTR_RX_LEN_MMATCH | CAS_INTR_PCI_ERROR_INT)) != 0)) { 2043194246Smarius cas_eint(sc, status); 2044223986Smarius CAS_UNLOCK(sc); 2045194246Smarius return; 2046194246Smarius } 2047194246Smarius 2048194246Smarius if (__predict_false(status & CAS_INTR_TX_MAC_INT)) { 2049194246Smarius status2 = CAS_READ_4(sc, CAS_MAC_TX_STATUS); 2050194246Smarius if ((status2 & 2051194246Smarius (CAS_MAC_TX_UNDERRUN | CAS_MAC_TX_MAX_PKT_ERR)) != 0) 2052223986Smarius ifp->if_oerrors++; 2053194246Smarius else if ((status2 & ~CAS_MAC_TX_FRAME_XMTD) != 0) 2054194246Smarius device_printf(sc->sc_dev, 2055194246Smarius "MAC TX fault, status %x\n", status2); 2056194246Smarius } 2057194246Smarius 2058194246Smarius if (__predict_false(status & CAS_INTR_RX_MAC_INT)) { 2059194246Smarius status2 = CAS_READ_4(sc, CAS_MAC_RX_STATUS); 2060194246Smarius if ((status2 & CAS_MAC_RX_OVERFLOW) != 0) 2061223986Smarius ifp->if_ierrors++; 2062194246Smarius else if ((status2 & ~CAS_MAC_RX_FRAME_RCVD) != 0) 2063194246Smarius device_printf(sc->sc_dev, 2064194246Smarius "MAC RX fault, status %x\n", status2); 2065194246Smarius } 2066194246Smarius 2067194246Smarius if ((status & 2068194246Smarius (CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_COMP_FULL | 2069194246Smarius CAS_INTR_RX_BUF_AEMPTY | CAS_INTR_RX_COMP_AFULL)) != 0) { 2070194246Smarius cas_rint(sc); 2071194904Smarius#ifdef CAS_DEBUG 2072194246Smarius if (__predict_false((status & 2073194246Smarius (CAS_INTR_RX_BUF_NA | CAS_INTR_RX_COMP_FULL | 2074194246Smarius CAS_INTR_RX_BUF_AEMPTY | CAS_INTR_RX_COMP_AFULL)) != 0)) 2075194246Smarius device_printf(sc->sc_dev, 2076194246Smarius "RX fault, status %x\n", status); 2077194904Smarius#endif 2078194246Smarius } 2079194246Smarius 2080194246Smarius if ((status & 2081223986Smarius (CAS_INTR_TX_INT_ME | CAS_INTR_TX_ALL | CAS_INTR_TX_DONE)) != 0) 2082194246Smarius cas_tint(sc); 2083223986Smarius 2084223986Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0) { 2085194904Smarius CAS_UNLOCK(sc); 2086194904Smarius return; 2087223986Smarius } else if (!IFQ_DRV_IS_EMPTY(&ifp->if_snd)) 2088194904Smarius taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 2089223986Smarius CAS_UNLOCK(sc); 2090194904Smarius 2091194904Smarius status = CAS_READ_4(sc, CAS_STATUS_ALIAS); 2092194904Smarius if (__predict_false((status & CAS_INTR_SUMMARY) != 0)) { 2093194904Smarius taskqueue_enqueue(sc->sc_tq, &sc->sc_intr_task); 2094194904Smarius return; 2095194904Smarius } 2096194904Smarius 2097194904Smarius done: 2098194904Smarius /* Re-enable interrupts. */ 2099194904Smarius CAS_WRITE_4(sc, CAS_INTMASK, 2100194904Smarius ~(CAS_INTR_TX_INT_ME | CAS_INTR_TX_TAG_ERR | 2101194904Smarius CAS_INTR_RX_DONE | CAS_INTR_RX_BUF_NA | CAS_INTR_RX_TAG_ERR | 2102194904Smarius CAS_INTR_RX_COMP_FULL | CAS_INTR_RX_BUF_AEMPTY | 2103194904Smarius CAS_INTR_RX_COMP_AFULL | CAS_INTR_RX_LEN_MMATCH | 2104194904Smarius CAS_INTR_PCI_ERROR_INT 2105194904Smarius#ifdef CAS_DEBUG 2106194904Smarius | CAS_INTR_PCS_INT | CAS_INTR_MIF 2107194904Smarius#endif 2108194904Smarius )); 2109194246Smarius} 2110194246Smarius 2111194904Smariusstatic void 2112194246Smariuscas_watchdog(struct cas_softc *sc) 2113194246Smarius{ 2114194246Smarius struct ifnet *ifp = sc->sc_ifp; 2115194246Smarius 2116194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 2117194246Smarius 2118194246Smarius#ifdef CAS_DEBUG 2119194246Smarius CTR4(KTR_CAS, 2120215721Smarius "%s: CAS_RX_CONF %x CAS_MAC_RX_STATUS %x CAS_MAC_RX_CONF %x", 2121215721Smarius __func__, CAS_READ_4(sc, CAS_RX_CONF), 2122194246Smarius CAS_READ_4(sc, CAS_MAC_RX_STATUS), 2123215721Smarius CAS_READ_4(sc, CAS_MAC_RX_CONF)); 2124194246Smarius CTR4(KTR_CAS, 2125215721Smarius "%s: CAS_TX_CONF %x CAS_MAC_TX_STATUS %x CAS_MAC_TX_CONF %x", 2126215721Smarius __func__, CAS_READ_4(sc, CAS_TX_CONF), 2127194246Smarius CAS_READ_4(sc, CAS_MAC_TX_STATUS), 2128215721Smarius CAS_READ_4(sc, CAS_MAC_TX_CONF)); 2129194246Smarius#endif 2130194246Smarius 2131194246Smarius if (sc->sc_wdog_timer == 0 || --sc->sc_wdog_timer != 0) 2132194904Smarius return; 2133194246Smarius 2134194246Smarius if ((sc->sc_flags & CAS_LINK) != 0) 2135194246Smarius device_printf(sc->sc_dev, "device timeout\n"); 2136194246Smarius else if (bootverbose) 2137194246Smarius device_printf(sc->sc_dev, "device timeout (no link)\n"); 2138194246Smarius ++ifp->if_oerrors; 2139194246Smarius 2140194246Smarius /* Try to get more packets going. */ 2141194904Smarius ifp->if_drv_flags &= ~IFF_DRV_RUNNING; 2142194246Smarius cas_init_locked(sc); 2143194904Smarius if (!IFQ_DRV_IS_EMPTY(&ifp->if_snd)) 2144194904Smarius taskqueue_enqueue(sc->sc_tq, &sc->sc_tx_task); 2145194246Smarius} 2146194246Smarius 2147194246Smariusstatic void 2148194246Smariuscas_mifinit(struct cas_softc *sc) 2149194246Smarius{ 2150194246Smarius 2151194246Smarius /* Configure the MIF in frame mode. */ 2152194246Smarius CAS_WRITE_4(sc, CAS_MIF_CONF, 2153194246Smarius CAS_READ_4(sc, CAS_MIF_CONF) & ~CAS_MIF_CONF_BB_MODE); 2154207585Smarius CAS_BARRIER(sc, CAS_MIF_CONF, 4, 2155207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2156194246Smarius} 2157194246Smarius 2158194246Smarius/* 2159194246Smarius * MII interface 2160194246Smarius * 2161194246Smarius * The MII interface supports at least three different operating modes: 2162194246Smarius * 2163194246Smarius * Bitbang mode is implemented using data, clock and output enable registers. 2164194246Smarius * 2165194246Smarius * Frame mode is implemented by loading a complete frame into the frame 2166194246Smarius * register and polling the valid bit for completion. 2167194246Smarius * 2168194246Smarius * Polling mode uses the frame register but completion is indicated by 2169194246Smarius * an interrupt. 2170194246Smarius * 2171194246Smarius */ 2172194246Smariusstatic int 2173194246Smariuscas_mii_readreg(device_t dev, int phy, int reg) 2174194246Smarius{ 2175194246Smarius struct cas_softc *sc; 2176194246Smarius int n; 2177194246Smarius uint32_t v; 2178194246Smarius 2179194246Smarius#ifdef CAS_DEBUG_PHY 2180194246Smarius printf("%s: phy %d reg %d\n", __func__, phy, reg); 2181194246Smarius#endif 2182194246Smarius 2183194246Smarius sc = device_get_softc(dev); 2184194246Smarius if ((sc->sc_flags & CAS_SERDES) != 0) { 2185194246Smarius switch (reg) { 2186194246Smarius case MII_BMCR: 2187194246Smarius reg = CAS_PCS_CTRL; 2188194246Smarius break; 2189194246Smarius case MII_BMSR: 2190194246Smarius reg = CAS_PCS_STATUS; 2191194246Smarius break; 2192194246Smarius case MII_PHYIDR1: 2193194246Smarius case MII_PHYIDR2: 2194194246Smarius return (0); 2195194246Smarius case MII_ANAR: 2196194246Smarius reg = CAS_PCS_ANAR; 2197194246Smarius break; 2198194246Smarius case MII_ANLPAR: 2199194246Smarius reg = CAS_PCS_ANLPAR; 2200194246Smarius break; 2201194246Smarius case MII_EXTSR: 2202194246Smarius return (EXTSR_1000XFDX | EXTSR_1000XHDX); 2203194246Smarius default: 2204194246Smarius device_printf(sc->sc_dev, 2205194246Smarius "%s: unhandled register %d\n", __func__, reg); 2206194246Smarius return (0); 2207194246Smarius } 2208194246Smarius return (CAS_READ_4(sc, reg)); 2209194246Smarius } 2210194246Smarius 2211194246Smarius /* Construct the frame command. */ 2212194246Smarius v = CAS_MIF_FRAME_READ | 2213194246Smarius (phy << CAS_MIF_FRAME_PHY_SHFT) | 2214194246Smarius (reg << CAS_MIF_FRAME_REG_SHFT); 2215194246Smarius 2216194246Smarius CAS_WRITE_4(sc, CAS_MIF_FRAME, v); 2217194246Smarius CAS_BARRIER(sc, CAS_MIF_FRAME, 4, 2218194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2219194246Smarius for (n = 0; n < 100; n++) { 2220194246Smarius DELAY(1); 2221194246Smarius v = CAS_READ_4(sc, CAS_MIF_FRAME); 2222194246Smarius if (v & CAS_MIF_FRAME_TA_LSB) 2223194246Smarius return (v & CAS_MIF_FRAME_DATA); 2224194246Smarius } 2225194246Smarius 2226194246Smarius device_printf(sc->sc_dev, "%s: timed out\n", __func__); 2227194246Smarius return (0); 2228194246Smarius} 2229194246Smarius 2230194246Smariusstatic int 2231194246Smariuscas_mii_writereg(device_t dev, int phy, int reg, int val) 2232194246Smarius{ 2233194246Smarius struct cas_softc *sc; 2234194246Smarius int n; 2235194246Smarius uint32_t v; 2236194246Smarius 2237194246Smarius#ifdef CAS_DEBUG_PHY 2238194246Smarius printf("%s: phy %d reg %d val %x\n", phy, reg, val, __func__); 2239194246Smarius#endif 2240194246Smarius 2241194246Smarius sc = device_get_softc(dev); 2242194246Smarius if ((sc->sc_flags & CAS_SERDES) != 0) { 2243194246Smarius switch (reg) { 2244194246Smarius case MII_BMSR: 2245194246Smarius reg = CAS_PCS_STATUS; 2246194246Smarius break; 2247194246Smarius case MII_BMCR: 2248194246Smarius reg = CAS_PCS_CTRL; 2249194246Smarius if ((val & CAS_PCS_CTRL_RESET) == 0) 2250194246Smarius break; 2251194246Smarius CAS_WRITE_4(sc, CAS_PCS_CTRL, val); 2252194246Smarius CAS_BARRIER(sc, CAS_PCS_CTRL, 4, 2253194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2254194246Smarius if (!cas_bitwait(sc, CAS_PCS_CTRL, 2255194246Smarius CAS_PCS_CTRL_RESET, 0)) 2256194246Smarius device_printf(sc->sc_dev, 2257194246Smarius "cannot reset PCS\n"); 2258194246Smarius /* FALLTHROUGH */ 2259194246Smarius case MII_ANAR: 2260194246Smarius CAS_WRITE_4(sc, CAS_PCS_CONF, 0); 2261194246Smarius CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2262194246Smarius BUS_SPACE_BARRIER_WRITE); 2263194246Smarius CAS_WRITE_4(sc, CAS_PCS_ANAR, val); 2264207585Smarius CAS_BARRIER(sc, CAS_PCS_ANAR, 4, 2265207585Smarius BUS_SPACE_BARRIER_WRITE); 2266194246Smarius CAS_WRITE_4(sc, CAS_PCS_SERDES_CTRL, 2267194246Smarius CAS_PCS_SERDES_CTRL_ESD); 2268207585Smarius CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2269207585Smarius BUS_SPACE_BARRIER_WRITE); 2270194246Smarius CAS_WRITE_4(sc, CAS_PCS_CONF, 2271194246Smarius CAS_PCS_CONF_EN); 2272207585Smarius CAS_BARRIER(sc, CAS_PCS_CONF, 4, 2273207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2274194246Smarius return (0); 2275194246Smarius case MII_ANLPAR: 2276194246Smarius reg = CAS_PCS_ANLPAR; 2277194246Smarius break; 2278194246Smarius default: 2279194246Smarius device_printf(sc->sc_dev, 2280194246Smarius "%s: unhandled register %d\n", __func__, reg); 2281194246Smarius return (0); 2282194246Smarius } 2283194246Smarius CAS_WRITE_4(sc, reg, val); 2284207585Smarius CAS_BARRIER(sc, reg, 4, 2285207585Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2286194246Smarius return (0); 2287194246Smarius } 2288194246Smarius 2289194246Smarius /* Construct the frame command. */ 2290194246Smarius v = CAS_MIF_FRAME_WRITE | 2291194246Smarius (phy << CAS_MIF_FRAME_PHY_SHFT) | 2292194246Smarius (reg << CAS_MIF_FRAME_REG_SHFT) | 2293194246Smarius (val & CAS_MIF_FRAME_DATA); 2294194246Smarius 2295194246Smarius CAS_WRITE_4(sc, CAS_MIF_FRAME, v); 2296194246Smarius CAS_BARRIER(sc, CAS_MIF_FRAME, 4, 2297194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2298194246Smarius for (n = 0; n < 100; n++) { 2299194246Smarius DELAY(1); 2300194246Smarius v = CAS_READ_4(sc, CAS_MIF_FRAME); 2301194246Smarius if (v & CAS_MIF_FRAME_TA_LSB) 2302194246Smarius return (1); 2303194246Smarius } 2304194246Smarius 2305194246Smarius device_printf(sc->sc_dev, "%s: timed out\n", __func__); 2306194246Smarius return (0); 2307194246Smarius} 2308194246Smarius 2309194246Smariusstatic void 2310194246Smariuscas_mii_statchg(device_t dev) 2311194246Smarius{ 2312194246Smarius struct cas_softc *sc; 2313194246Smarius struct ifnet *ifp; 2314194246Smarius int gigabit; 2315194246Smarius uint32_t rxcfg, txcfg, v; 2316194246Smarius 2317194246Smarius sc = device_get_softc(dev); 2318194246Smarius ifp = sc->sc_ifp; 2319194246Smarius 2320194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 2321194246Smarius 2322194246Smarius#ifdef CAS_DEBUG 2323194246Smarius if ((ifp->if_flags & IFF_DEBUG) != 0) 2324213893Smarius device_printf(sc->sc_dev, "%s: status changen", __func__); 2325194246Smarius#endif 2326194246Smarius 2327194246Smarius if ((sc->sc_mii->mii_media_status & IFM_ACTIVE) != 0 && 2328194246Smarius IFM_SUBTYPE(sc->sc_mii->mii_media_active) != IFM_NONE) 2329194246Smarius sc->sc_flags |= CAS_LINK; 2330194246Smarius else 2331194246Smarius sc->sc_flags &= ~CAS_LINK; 2332194246Smarius 2333194246Smarius switch (IFM_SUBTYPE(sc->sc_mii->mii_media_active)) { 2334194246Smarius case IFM_1000_SX: 2335194246Smarius case IFM_1000_LX: 2336194246Smarius case IFM_1000_CX: 2337194246Smarius case IFM_1000_T: 2338194246Smarius gigabit = 1; 2339194246Smarius break; 2340194246Smarius default: 2341194246Smarius gigabit = 0; 2342194246Smarius } 2343194246Smarius 2344194246Smarius /* 2345194246Smarius * The configuration done here corresponds to the steps F) and 2346194246Smarius * G) and as far as enabling of RX and TX MAC goes also step H) 2347194246Smarius * of the initialization sequence outlined in section 11.2.1 of 2348194246Smarius * the Cassini+ ASIC Specification. 2349194246Smarius */ 2350194246Smarius 2351223951Smarius rxcfg = sc->sc_mac_rxcfg; 2352223951Smarius rxcfg &= ~CAS_MAC_RX_CONF_CARR; 2353194246Smarius txcfg = CAS_MAC_TX_CONF_EN_IPG0 | CAS_MAC_TX_CONF_NGU | 2354194246Smarius CAS_MAC_TX_CONF_NGUL; 2355194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) != 0) 2356194246Smarius txcfg |= CAS_MAC_TX_CONF_ICARR | CAS_MAC_TX_CONF_ICOLLIS; 2357194246Smarius else if (gigabit != 0) { 2358194246Smarius rxcfg |= CAS_MAC_RX_CONF_CARR; 2359194246Smarius txcfg |= CAS_MAC_TX_CONF_CARR; 2360194246Smarius } 2361223951Smarius (void)cas_disable_tx(sc); 2362194246Smarius CAS_WRITE_4(sc, CAS_MAC_TX_CONF, txcfg); 2363223951Smarius (void)cas_disable_rx(sc); 2364194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CONF, rxcfg); 2365194246Smarius 2366194246Smarius v = CAS_READ_4(sc, CAS_MAC_CTRL_CONF) & 2367194246Smarius ~(CAS_MAC_CTRL_CONF_TXP | CAS_MAC_CTRL_CONF_RXP); 2368194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & 2369194246Smarius IFM_ETH_RXPAUSE) != 0) 2370194246Smarius v |= CAS_MAC_CTRL_CONF_RXP; 2371194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & 2372194246Smarius IFM_ETH_TXPAUSE) != 0) 2373194246Smarius v |= CAS_MAC_CTRL_CONF_TXP; 2374194246Smarius CAS_WRITE_4(sc, CAS_MAC_CTRL_CONF, v); 2375194246Smarius 2376194246Smarius /* 2377194246Smarius * All supported chips have a bug causing incorrect checksum 2378194246Smarius * to be calculated when letting them strip the FCS in half- 2379194246Smarius * duplex mode. In theory we could disable FCS stripping and 2380194246Smarius * manually adjust the checksum accordingly. It seems to make 2381194246Smarius * more sense to optimze for the common case and just disable 2382194246Smarius * hardware checksumming in half-duplex mode though. 2383194246Smarius */ 2384194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) { 2385194246Smarius ifp->if_capenable &= ~IFCAP_HWCSUM; 2386194246Smarius ifp->if_hwassist = 0; 2387194246Smarius } else if ((sc->sc_flags & CAS_NO_CSUM) == 0) { 2388194246Smarius ifp->if_capenable = ifp->if_capabilities; 2389194246Smarius ifp->if_hwassist = CAS_CSUM_FEATURES; 2390194246Smarius } 2391194246Smarius 2392194246Smarius if (sc->sc_variant == CAS_SATURN) { 2393194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) 2394194246Smarius /* silicon bug workaround */ 2395194246Smarius CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x41); 2396194246Smarius else 2397194246Smarius CAS_WRITE_4(sc, CAS_MAC_PREAMBLE_LEN, 0x7); 2398194246Smarius } 2399194246Smarius 2400194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0 && 2401194246Smarius gigabit != 0) 2402194246Smarius CAS_WRITE_4(sc, CAS_MAC_SLOT_TIME, 2403194246Smarius CAS_MAC_SLOT_TIME_CARR); 2404194246Smarius else 2405194246Smarius CAS_WRITE_4(sc, CAS_MAC_SLOT_TIME, 2406194246Smarius CAS_MAC_SLOT_TIME_NORM); 2407194246Smarius 2408194246Smarius /* XIF Configuration */ 2409194246Smarius v = CAS_MAC_XIF_CONF_TX_OE | CAS_MAC_XIF_CONF_LNKLED; 2410194246Smarius if ((sc->sc_flags & CAS_SERDES) == 0) { 2411194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) == 0) 2412194246Smarius v |= CAS_MAC_XIF_CONF_NOECHO; 2413194246Smarius v |= CAS_MAC_XIF_CONF_BUF_OE; 2414194246Smarius } 2415194246Smarius if (gigabit != 0) 2416194246Smarius v |= CAS_MAC_XIF_CONF_GMII; 2417194246Smarius if ((IFM_OPTIONS(sc->sc_mii->mii_media_active) & IFM_FDX) != 0) 2418194246Smarius v |= CAS_MAC_XIF_CONF_FDXLED; 2419194246Smarius CAS_WRITE_4(sc, CAS_MAC_XIF_CONF, v); 2420194246Smarius 2421223951Smarius sc->sc_mac_rxcfg = rxcfg; 2422223986Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) != 0 && 2423194246Smarius (sc->sc_flags & CAS_LINK) != 0) { 2424194246Smarius CAS_WRITE_4(sc, CAS_MAC_TX_CONF, 2425194246Smarius txcfg | CAS_MAC_TX_CONF_EN); 2426194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CONF, 2427194246Smarius rxcfg | CAS_MAC_RX_CONF_EN); 2428194246Smarius } 2429194246Smarius} 2430194246Smarius 2431194246Smariusstatic int 2432194246Smariuscas_mediachange(struct ifnet *ifp) 2433194246Smarius{ 2434194246Smarius struct cas_softc *sc = ifp->if_softc; 2435194246Smarius int error; 2436194246Smarius 2437194246Smarius /* XXX add support for serial media. */ 2438194246Smarius 2439194246Smarius CAS_LOCK(sc); 2440194246Smarius error = mii_mediachg(sc->sc_mii); 2441194246Smarius CAS_UNLOCK(sc); 2442194246Smarius return (error); 2443194246Smarius} 2444194246Smarius 2445194246Smariusstatic void 2446194246Smariuscas_mediastatus(struct ifnet *ifp, struct ifmediareq *ifmr) 2447194246Smarius{ 2448194246Smarius struct cas_softc *sc = ifp->if_softc; 2449194246Smarius 2450194246Smarius CAS_LOCK(sc); 2451194246Smarius if ((ifp->if_flags & IFF_UP) == 0) { 2452194246Smarius CAS_UNLOCK(sc); 2453194246Smarius return; 2454194246Smarius } 2455194246Smarius 2456194246Smarius mii_pollstat(sc->sc_mii); 2457194246Smarius ifmr->ifm_active = sc->sc_mii->mii_media_active; 2458194246Smarius ifmr->ifm_status = sc->sc_mii->mii_media_status; 2459194246Smarius CAS_UNLOCK(sc); 2460194246Smarius} 2461194246Smarius 2462194246Smariusstatic int 2463194246Smariuscas_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data) 2464194246Smarius{ 2465194246Smarius struct cas_softc *sc = ifp->if_softc; 2466194246Smarius struct ifreq *ifr = (struct ifreq *)data; 2467194246Smarius int error; 2468194246Smarius 2469194246Smarius error = 0; 2470194246Smarius switch (cmd) { 2471194246Smarius case SIOCSIFFLAGS: 2472194246Smarius CAS_LOCK(sc); 2473194246Smarius if ((ifp->if_flags & IFF_UP) != 0) { 2474194246Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) != 0 && 2475194246Smarius ((ifp->if_flags ^ sc->sc_ifflags) & 2476194246Smarius (IFF_ALLMULTI | IFF_PROMISC)) != 0) 2477194246Smarius cas_setladrf(sc); 2478194246Smarius else 2479194246Smarius cas_init_locked(sc); 2480194246Smarius } else if ((ifp->if_drv_flags & IFF_DRV_RUNNING) != 0) 2481194246Smarius cas_stop(ifp); 2482194246Smarius sc->sc_ifflags = ifp->if_flags; 2483194246Smarius CAS_UNLOCK(sc); 2484194246Smarius break; 2485194246Smarius case SIOCSIFCAP: 2486194246Smarius CAS_LOCK(sc); 2487194246Smarius if ((sc->sc_flags & CAS_NO_CSUM) != 0) { 2488194246Smarius error = EINVAL; 2489194246Smarius CAS_UNLOCK(sc); 2490194246Smarius break; 2491194246Smarius } 2492194246Smarius ifp->if_capenable = ifr->ifr_reqcap; 2493194246Smarius if ((ifp->if_capenable & IFCAP_TXCSUM) != 0) 2494194246Smarius ifp->if_hwassist = CAS_CSUM_FEATURES; 2495194246Smarius else 2496194246Smarius ifp->if_hwassist = 0; 2497194246Smarius CAS_UNLOCK(sc); 2498194246Smarius break; 2499194246Smarius case SIOCADDMULTI: 2500194246Smarius case SIOCDELMULTI: 2501194246Smarius CAS_LOCK(sc); 2502194904Smarius if ((ifp->if_drv_flags & IFF_DRV_RUNNING) != 0) 2503194904Smarius cas_setladrf(sc); 2504194246Smarius CAS_UNLOCK(sc); 2505194246Smarius break; 2506194246Smarius case SIOCSIFMTU: 2507194246Smarius if ((ifr->ifr_mtu < ETHERMIN) || 2508194246Smarius (ifr->ifr_mtu > ETHERMTU_JUMBO)) 2509194246Smarius error = EINVAL; 2510194246Smarius else 2511194246Smarius ifp->if_mtu = ifr->ifr_mtu; 2512194246Smarius break; 2513194246Smarius case SIOCGIFMEDIA: 2514194246Smarius case SIOCSIFMEDIA: 2515194246Smarius error = ifmedia_ioctl(ifp, ifr, &sc->sc_mii->mii_media, cmd); 2516194246Smarius break; 2517194246Smarius default: 2518194246Smarius error = ether_ioctl(ifp, cmd, data); 2519194246Smarius break; 2520194246Smarius } 2521194246Smarius 2522194246Smarius return (error); 2523194246Smarius} 2524194246Smarius 2525194246Smariusstatic void 2526194246Smariuscas_setladrf(struct cas_softc *sc) 2527194246Smarius{ 2528194246Smarius struct ifnet *ifp = sc->sc_ifp; 2529194246Smarius struct ifmultiaddr *inm; 2530194246Smarius int i; 2531194246Smarius uint32_t hash[16]; 2532194246Smarius uint32_t crc, v; 2533194246Smarius 2534194246Smarius CAS_LOCK_ASSERT(sc, MA_OWNED); 2535194246Smarius 2536194246Smarius /* 2537223951Smarius * Turn off the RX MAC and the hash filter as required by the Sun 2538223951Smarius * Cassini programming restrictions. 2539194246Smarius */ 2540223951Smarius v = sc->sc_mac_rxcfg & ~(CAS_MAC_RX_CONF_HFILTER | 2541223951Smarius CAS_MAC_RX_CONF_EN); 2542194246Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CONF, v); 2543194246Smarius CAS_BARRIER(sc, CAS_MAC_RX_CONF, 4, 2544194246Smarius BUS_SPACE_BARRIER_READ | BUS_SPACE_BARRIER_WRITE); 2545223951Smarius if (!cas_bitwait(sc, CAS_MAC_RX_CONF, CAS_MAC_RX_CONF_HFILTER | 2546223951Smarius CAS_MAC_RX_CONF_EN, 0)) 2547223951Smarius device_printf(sc->sc_dev, 2548223951Smarius "cannot disable RX MAC or hash filter\n"); 2549194246Smarius 2550223951Smarius v &= ~(CAS_MAC_RX_CONF_PROMISC | CAS_MAC_RX_CONF_PGRP); 2551194246Smarius if ((ifp->if_flags & IFF_PROMISC) != 0) { 2552194246Smarius v |= CAS_MAC_RX_CONF_PROMISC; 2553194246Smarius goto chipit; 2554194246Smarius } 2555194246Smarius if ((ifp->if_flags & IFF_ALLMULTI) != 0) { 2556194246Smarius v |= CAS_MAC_RX_CONF_PGRP; 2557194246Smarius goto chipit; 2558194246Smarius } 2559194246Smarius 2560194246Smarius /* 2561194246Smarius * Set up multicast address filter by passing all multicast 2562194246Smarius * addresses through a crc generator, and then using the high 2563194246Smarius * order 8 bits as an index into the 256 bit logical address 2564194246Smarius * filter. The high order 4 bits selects the word, while the 2565194246Smarius * other 4 bits select the bit within the word (where bit 0 2566194246Smarius * is the MSB). 2567194246Smarius */ 2568194246Smarius 2569194246Smarius /* Clear the hash table. */ 2570194246Smarius memset(hash, 0, sizeof(hash)); 2571194246Smarius 2572195049Srwatson if_maddr_rlock(ifp); 2573194246Smarius TAILQ_FOREACH(inm, &ifp->if_multiaddrs, ifma_link) { 2574194246Smarius if (inm->ifma_addr->sa_family != AF_LINK) 2575194246Smarius continue; 2576194246Smarius crc = ether_crc32_le(LLADDR((struct sockaddr_dl *) 2577194246Smarius inm->ifma_addr), ETHER_ADDR_LEN); 2578194246Smarius 2579194246Smarius /* We just want the 8 most significant bits. */ 2580194246Smarius crc >>= 24; 2581194246Smarius 2582194246Smarius /* Set the corresponding bit in the filter. */ 2583194246Smarius hash[crc >> 4] |= 1 << (15 - (crc & 15)); 2584194246Smarius } 2585195049Srwatson if_maddr_runlock(ifp); 2586194246Smarius 2587194246Smarius v |= CAS_MAC_RX_CONF_HFILTER; 2588194246Smarius 2589194246Smarius /* Now load the hash table into the chip (if we are using it). */ 2590194246Smarius for (i = 0; i < 16; i++) 2591194246Smarius CAS_WRITE_4(sc, 2592194246Smarius CAS_MAC_HASH0 + i * (CAS_MAC_HASH1 - CAS_MAC_HASH0), 2593194246Smarius hash[i]); 2594194246Smarius 2595194246Smarius chipit: 2596223951Smarius sc->sc_mac_rxcfg = v; 2597223951Smarius CAS_WRITE_4(sc, CAS_MAC_RX_CONF, v | CAS_MAC_RX_CONF_EN); 2598194246Smarius} 2599194246Smarius 2600194246Smariusstatic int cas_pci_attach(device_t dev); 2601194246Smariusstatic int cas_pci_detach(device_t dev); 2602194246Smariusstatic int cas_pci_probe(device_t dev); 2603194246Smariusstatic int cas_pci_resume(device_t dev); 2604194246Smariusstatic int cas_pci_suspend(device_t dev); 2605194246Smarius 2606194246Smariusstatic device_method_t cas_pci_methods[] = { 2607194246Smarius /* Device interface */ 2608194246Smarius DEVMETHOD(device_probe, cas_pci_probe), 2609194246Smarius DEVMETHOD(device_attach, cas_pci_attach), 2610194246Smarius DEVMETHOD(device_detach, cas_pci_detach), 2611194246Smarius DEVMETHOD(device_suspend, cas_pci_suspend), 2612194246Smarius DEVMETHOD(device_resume, cas_pci_resume), 2613194246Smarius /* Use the suspend handler here, it is all that is required. */ 2614194246Smarius DEVMETHOD(device_shutdown, cas_pci_suspend), 2615194246Smarius 2616194246Smarius /* MII interface */ 2617194246Smarius DEVMETHOD(miibus_readreg, cas_mii_readreg), 2618194246Smarius DEVMETHOD(miibus_writereg, cas_mii_writereg), 2619194246Smarius DEVMETHOD(miibus_statchg, cas_mii_statchg), 2620194246Smarius 2621227843Smarius DEVMETHOD_END 2622194246Smarius}; 2623194246Smarius 2624194246Smariusstatic driver_t cas_pci_driver = { 2625194246Smarius "cas", 2626194246Smarius cas_pci_methods, 2627194246Smarius sizeof(struct cas_softc) 2628194246Smarius}; 2629194246Smarius 2630194246SmariusDRIVER_MODULE(cas, pci, cas_pci_driver, cas_devclass, 0, 0); 2631194246SmariusDRIVER_MODULE(miibus, cas, miibus_driver, miibus_devclass, 0, 0); 2632194246SmariusMODULE_DEPEND(cas, pci, 1, 1, 1); 2633194246Smarius 2634194246Smariusstatic const struct cas_pci_dev { 2635194246Smarius uint32_t cpd_devid; 2636194246Smarius uint8_t cpd_revid; 2637194246Smarius int cpd_variant; 2638194246Smarius const char *cpd_desc; 2639242625Sdim} cas_pci_devlist[] = { 2640194246Smarius { 0x0035100b, 0x0, CAS_SATURN, "NS DP83065 Saturn Gigabit Ethernet" }, 2641194246Smarius { 0xabba108e, 0x10, CAS_CASPLUS, "Sun Cassini+ Gigabit Ethernet" }, 2642194246Smarius { 0xabba108e, 0x0, CAS_CAS, "Sun Cassini Gigabit Ethernet" }, 2643194246Smarius { 0, 0, 0, NULL } 2644194246Smarius}; 2645194246Smarius 2646194246Smariusstatic int 2647194246Smariuscas_pci_probe(device_t dev) 2648194246Smarius{ 2649194246Smarius int i; 2650194246Smarius 2651194246Smarius for (i = 0; cas_pci_devlist[i].cpd_desc != NULL; i++) { 2652194246Smarius if (pci_get_devid(dev) == cas_pci_devlist[i].cpd_devid && 2653194246Smarius pci_get_revid(dev) >= cas_pci_devlist[i].cpd_revid) { 2654194246Smarius device_set_desc(dev, cas_pci_devlist[i].cpd_desc); 2655194246Smarius return (BUS_PROBE_DEFAULT); 2656194246Smarius } 2657194246Smarius } 2658194246Smarius 2659194246Smarius return (ENXIO); 2660194246Smarius} 2661194246Smarius 2662194246Smariusstatic struct resource_spec cas_pci_res_spec[] = { 2663194246Smarius { SYS_RES_IRQ, 0, RF_SHAREABLE | RF_ACTIVE }, /* CAS_RES_INTR */ 2664194246Smarius { SYS_RES_MEMORY, PCIR_BAR(0), RF_ACTIVE }, /* CAS_RES_MEM */ 2665194246Smarius { -1, 0 } 2666194246Smarius}; 2667194246Smarius 2668207585Smarius#define CAS_LOCAL_MAC_ADDRESS "local-mac-address" 2669207585Smarius#define CAS_PHY_INTERFACE "phy-interface" 2670207585Smarius#define CAS_PHY_TYPE "phy-type" 2671207585Smarius#define CAS_PHY_TYPE_PCS "pcs" 2672207585Smarius 2673194246Smariusstatic int 2674194246Smariuscas_pci_attach(device_t dev) 2675194246Smarius{ 2676207585Smarius char buf[sizeof(CAS_LOCAL_MAC_ADDRESS)]; 2677194246Smarius struct cas_softc *sc; 2678194246Smarius int i; 2679194246Smarius#if !(defined(__powerpc__) || defined(__sparc64__)) 2680194246Smarius u_char enaddr[4][ETHER_ADDR_LEN]; 2681207585Smarius u_int j, k, lma, pcs[4], phy; 2682194246Smarius#endif 2683194246Smarius 2684194246Smarius sc = device_get_softc(dev); 2685194246Smarius sc->sc_variant = CAS_UNKNOWN; 2686194246Smarius for (i = 0; cas_pci_devlist[i].cpd_desc != NULL; i++) { 2687194246Smarius if (pci_get_devid(dev) == cas_pci_devlist[i].cpd_devid && 2688194246Smarius pci_get_revid(dev) >= cas_pci_devlist[i].cpd_revid) { 2689194246Smarius sc->sc_variant = cas_pci_devlist[i].cpd_variant; 2690194246Smarius break; 2691194246Smarius } 2692194246Smarius } 2693194246Smarius if (sc->sc_variant == CAS_UNKNOWN) { 2694194246Smarius device_printf(dev, "unknown adaptor\n"); 2695194246Smarius return (ENXIO); 2696194246Smarius } 2697194246Smarius 2698247579Smarius /* PCI configuration */ 2699247579Smarius pci_write_config(dev, PCIR_COMMAND, 2700247579Smarius pci_read_config(dev, PCIR_COMMAND, 2) | PCIM_CMD_BUSMASTEREN | 2701247579Smarius PCIM_CMD_MWRICEN | PCIM_CMD_PERRESPEN | PCIM_CMD_SERRESPEN, 2); 2702194246Smarius 2703194246Smarius sc->sc_dev = dev; 2704194246Smarius if (sc->sc_variant == CAS_CAS && pci_get_devid(dev) < 0x02) 2705194246Smarius /* Hardware checksumming may hang TX. */ 2706194246Smarius sc->sc_flags |= CAS_NO_CSUM; 2707194246Smarius if (sc->sc_variant == CAS_CASPLUS || sc->sc_variant == CAS_SATURN) 2708194246Smarius sc->sc_flags |= CAS_REG_PLUS; 2709194246Smarius if (sc->sc_variant == CAS_CAS || 2710194246Smarius (sc->sc_variant == CAS_CASPLUS && pci_get_revid(dev) < 0x11)) 2711194246Smarius sc->sc_flags |= CAS_TABORT; 2712194246Smarius if (bootverbose) 2713194246Smarius device_printf(dev, "flags=0x%x\n", sc->sc_flags); 2714194246Smarius 2715194246Smarius if (bus_alloc_resources(dev, cas_pci_res_spec, sc->sc_res)) { 2716194246Smarius device_printf(dev, "failed to allocate resources\n"); 2717194246Smarius bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2718194246Smarius return (ENXIO); 2719194246Smarius } 2720194246Smarius 2721194246Smarius CAS_LOCK_INIT(sc, device_get_nameunit(dev)); 2722194246Smarius 2723194246Smarius#if defined(__powerpc__) || defined(__sparc64__) 2724194246Smarius OF_getetheraddr(dev, sc->sc_enaddr); 2725207585Smarius if (OF_getprop(ofw_bus_get_node(dev), CAS_PHY_INTERFACE, buf, 2726207585Smarius sizeof(buf)) > 0 || OF_getprop(ofw_bus_get_node(dev), 2727207585Smarius CAS_PHY_TYPE, buf, sizeof(buf)) > 0) { 2728207585Smarius buf[sizeof(buf) - 1] = '\0'; 2729207585Smarius if (strcmp(buf, CAS_PHY_TYPE_PCS) == 0) 2730207585Smarius sc->sc_flags |= CAS_SERDES; 2731207585Smarius } 2732194246Smarius#else 2733194246Smarius /* 2734207585Smarius * Dig out VPD (vital product data) and read the MAC address as well 2735207585Smarius * as the PHY type. The VPD resides in the PCI Expansion ROM (PCI 2736207585Smarius * FCode) and can't be accessed via the PCI capability pointer. 2737207585Smarius * SUNW,pci-ce and SUNW,pci-qge use the Enhanced VPD format described 2738207585Smarius * in the free US Patent 7149820. 2739194246Smarius */ 2740194246Smarius 2741194246Smarius#define PCI_ROMHDR_SIZE 0x1c 2742194246Smarius#define PCI_ROMHDR_SIG 0x00 2743194246Smarius#define PCI_ROMHDR_SIG_MAGIC 0xaa55 /* little endian */ 2744194246Smarius#define PCI_ROMHDR_PTR_DATA 0x18 2745194246Smarius#define PCI_ROM_SIZE 0x18 2746194246Smarius#define PCI_ROM_SIG 0x00 2747194246Smarius#define PCI_ROM_SIG_MAGIC 0x52494350 /* "PCIR", endian */ 2748194246Smarius /* reversed */ 2749194246Smarius#define PCI_ROM_VENDOR 0x04 2750194246Smarius#define PCI_ROM_DEVICE 0x06 2751194246Smarius#define PCI_ROM_PTR_VPD 0x08 2752194246Smarius#define PCI_VPDRES_BYTE0 0x00 2753194246Smarius#define PCI_VPDRES_ISLARGE(x) ((x) & 0x80) 2754194246Smarius#define PCI_VPDRES_LARGE_NAME(x) ((x) & 0x7f) 2755194246Smarius#define PCI_VPDRES_LARGE_LEN_LSB 0x01 2756194246Smarius#define PCI_VPDRES_LARGE_LEN_MSB 0x02 2757194246Smarius#define PCI_VPDRES_LARGE_SIZE 0x03 2758194246Smarius#define PCI_VPDRES_TYPE_ID_STRING 0x02 /* large */ 2759194246Smarius#define PCI_VPDRES_TYPE_VPD 0x10 /* large */ 2760194246Smarius#define PCI_VPD_KEY0 0x00 2761194246Smarius#define PCI_VPD_KEY1 0x01 2762194246Smarius#define PCI_VPD_LEN 0x02 2763194246Smarius#define PCI_VPD_SIZE 0x03 2764194246Smarius 2765194246Smarius#define CAS_ROM_READ_1(sc, offs) \ 2766194246Smarius CAS_READ_1((sc), CAS_PCI_ROM_OFFSET + (offs)) 2767194246Smarius#define CAS_ROM_READ_2(sc, offs) \ 2768194246Smarius CAS_READ_2((sc), CAS_PCI_ROM_OFFSET + (offs)) 2769194246Smarius#define CAS_ROM_READ_4(sc, offs) \ 2770194246Smarius CAS_READ_4((sc), CAS_PCI_ROM_OFFSET + (offs)) 2771194246Smarius 2772207585Smarius lma = phy = 0; 2773207585Smarius memset(enaddr, 0, sizeof(enaddr)); 2774207585Smarius memset(pcs, 0, sizeof(pcs)); 2775207585Smarius 2776194246Smarius /* Enable PCI Expansion ROM access. */ 2777194246Smarius CAS_WRITE_4(sc, CAS_BIM_LDEV_OEN, 2778194246Smarius CAS_BIM_LDEV_OEN_PAD | CAS_BIM_LDEV_OEN_PROM); 2779194246Smarius 2780194246Smarius /* Read PCI Expansion ROM header. */ 2781194246Smarius if (CAS_ROM_READ_2(sc, PCI_ROMHDR_SIG) != PCI_ROMHDR_SIG_MAGIC || 2782194246Smarius (i = CAS_ROM_READ_2(sc, PCI_ROMHDR_PTR_DATA)) < 2783194246Smarius PCI_ROMHDR_SIZE) { 2784194246Smarius device_printf(dev, "unexpected PCI Expansion ROM header\n"); 2785194246Smarius goto fail_prom; 2786194246Smarius } 2787194246Smarius 2788194246Smarius /* Read PCI Expansion ROM data. */ 2789194246Smarius if (CAS_ROM_READ_4(sc, i + PCI_ROM_SIG) != PCI_ROM_SIG_MAGIC || 2790194246Smarius CAS_ROM_READ_2(sc, i + PCI_ROM_VENDOR) != pci_get_vendor(dev) || 2791194246Smarius CAS_ROM_READ_2(sc, i + PCI_ROM_DEVICE) != pci_get_device(dev) || 2792194246Smarius (j = CAS_ROM_READ_2(sc, i + PCI_ROM_PTR_VPD)) < 2793194246Smarius i + PCI_ROM_SIZE) { 2794194246Smarius device_printf(dev, "unexpected PCI Expansion ROM data\n"); 2795194246Smarius goto fail_prom; 2796194246Smarius } 2797194246Smarius 2798194246Smarius /* Read PCI VPD. */ 2799194246Smarius next: 2800194246Smarius if (PCI_VPDRES_ISLARGE(CAS_ROM_READ_1(sc, 2801194246Smarius j + PCI_VPDRES_BYTE0)) == 0) { 2802194246Smarius device_printf(dev, "no large PCI VPD\n"); 2803194246Smarius goto fail_prom; 2804194246Smarius } 2805194246Smarius 2806194246Smarius i = (CAS_ROM_READ_1(sc, j + PCI_VPDRES_LARGE_LEN_MSB) << 8) | 2807194246Smarius CAS_ROM_READ_1(sc, j + PCI_VPDRES_LARGE_LEN_LSB); 2808194246Smarius switch (PCI_VPDRES_LARGE_NAME(CAS_ROM_READ_1(sc, 2809194246Smarius j + PCI_VPDRES_BYTE0))) { 2810194246Smarius case PCI_VPDRES_TYPE_ID_STRING: 2811194246Smarius /* Skip identifier string. */ 2812194246Smarius j += PCI_VPDRES_LARGE_SIZE + i; 2813194246Smarius goto next; 2814194246Smarius case PCI_VPDRES_TYPE_VPD: 2815194246Smarius for (j += PCI_VPDRES_LARGE_SIZE; i > 0; 2816194246Smarius i -= PCI_VPD_SIZE + CAS_ROM_READ_1(sc, j + PCI_VPD_LEN), 2817194246Smarius j += PCI_VPD_SIZE + CAS_ROM_READ_1(sc, j + PCI_VPD_LEN)) { 2818194246Smarius if (CAS_ROM_READ_1(sc, j + PCI_VPD_KEY0) != 'Z') 2819194246Smarius /* no Enhanced VPD */ 2820194246Smarius continue; 2821194246Smarius if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE) != 'I') 2822194246Smarius /* no instance property */ 2823194246Smarius continue; 2824207585Smarius if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE + 3) == 'B') { 2825207585Smarius /* byte array */ 2826207585Smarius if (CAS_ROM_READ_1(sc, 2827207585Smarius j + PCI_VPD_SIZE + 4) != ETHER_ADDR_LEN) 2828207585Smarius continue; 2829207585Smarius bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2830207585Smarius CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 5, 2831207585Smarius buf, sizeof(buf)); 2832207585Smarius buf[sizeof(buf) - 1] = '\0'; 2833207585Smarius if (strcmp(buf, CAS_LOCAL_MAC_ADDRESS) != 0) 2834207585Smarius continue; 2835207585Smarius bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2836207585Smarius CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 2837207585Smarius 5 + sizeof(CAS_LOCAL_MAC_ADDRESS), 2838207585Smarius enaddr[lma], sizeof(enaddr[lma])); 2839207585Smarius lma++; 2840207585Smarius if (lma == 4 && phy == 4) 2841207585Smarius break; 2842207585Smarius } else if (CAS_ROM_READ_1(sc, j + PCI_VPD_SIZE + 3) == 2843207585Smarius 'S') { 2844207585Smarius /* string */ 2845207585Smarius if (CAS_ROM_READ_1(sc, 2846207585Smarius j + PCI_VPD_SIZE + 4) != 2847207585Smarius sizeof(CAS_PHY_TYPE_PCS)) 2848207585Smarius continue; 2849207585Smarius bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2850207585Smarius CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 5, 2851207585Smarius buf, sizeof(buf)); 2852207585Smarius buf[sizeof(buf) - 1] = '\0'; 2853207585Smarius if (strcmp(buf, CAS_PHY_INTERFACE) == 0) 2854207585Smarius k = sizeof(CAS_PHY_INTERFACE); 2855207585Smarius else if (strcmp(buf, CAS_PHY_TYPE) == 0) 2856207585Smarius k = sizeof(CAS_PHY_TYPE); 2857207585Smarius else 2858207585Smarius continue; 2859207585Smarius bus_read_region_1(sc->sc_res[CAS_RES_MEM], 2860207585Smarius CAS_PCI_ROM_OFFSET + j + PCI_VPD_SIZE + 2861207585Smarius 5 + k, buf, sizeof(buf)); 2862207585Smarius buf[sizeof(buf) - 1] = '\0'; 2863207585Smarius if (strcmp(buf, CAS_PHY_TYPE_PCS) == 0) 2864207585Smarius pcs[phy] = 1; 2865207585Smarius phy++; 2866207585Smarius if (lma == 4 && phy == 4) 2867207585Smarius break; 2868207585Smarius } 2869194246Smarius } 2870194246Smarius break; 2871194246Smarius default: 2872194246Smarius device_printf(dev, "unexpected PCI VPD\n"); 2873194246Smarius goto fail_prom; 2874194246Smarius } 2875194246Smarius 2876194246Smarius fail_prom: 2877194246Smarius CAS_WRITE_4(sc, CAS_BIM_LDEV_OEN, 0); 2878194246Smarius 2879207585Smarius if (lma == 0) { 2880194246Smarius device_printf(dev, "could not determine Ethernet address\n"); 2881194246Smarius goto fail; 2882194246Smarius } 2883194246Smarius i = 0; 2884245923Smarius if (lma > 1 && pci_get_slot(dev) < nitems(enaddr)) 2885194246Smarius i = pci_get_slot(dev); 2886194246Smarius memcpy(sc->sc_enaddr, enaddr[i], ETHER_ADDR_LEN); 2887207585Smarius 2888207585Smarius if (phy == 0) { 2889207585Smarius device_printf(dev, "could not determine PHY type\n"); 2890207585Smarius goto fail; 2891207585Smarius } 2892207585Smarius i = 0; 2893245923Smarius if (phy > 1 && pci_get_slot(dev) < nitems(pcs)) 2894207585Smarius i = pci_get_slot(dev); 2895207585Smarius if (pcs[i] != 0) 2896207585Smarius sc->sc_flags |= CAS_SERDES; 2897194246Smarius#endif 2898194246Smarius 2899194246Smarius if (cas_attach(sc) != 0) { 2900194246Smarius device_printf(dev, "could not be attached\n"); 2901194246Smarius goto fail; 2902194246Smarius } 2903194246Smarius 2904194246Smarius if (bus_setup_intr(dev, sc->sc_res[CAS_RES_INTR], INTR_TYPE_NET | 2905194904Smarius INTR_MPSAFE, cas_intr, NULL, sc, &sc->sc_ih) != 0) { 2906194246Smarius device_printf(dev, "failed to set up interrupt\n"); 2907194246Smarius cas_detach(sc); 2908194246Smarius goto fail; 2909194246Smarius } 2910194246Smarius return (0); 2911194246Smarius 2912194246Smarius fail: 2913194246Smarius CAS_LOCK_DESTROY(sc); 2914194246Smarius bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2915194246Smarius return (ENXIO); 2916194246Smarius} 2917194246Smarius 2918194246Smariusstatic int 2919194246Smariuscas_pci_detach(device_t dev) 2920194246Smarius{ 2921194246Smarius struct cas_softc *sc; 2922194246Smarius 2923194246Smarius sc = device_get_softc(dev); 2924194246Smarius bus_teardown_intr(dev, sc->sc_res[CAS_RES_INTR], sc->sc_ih); 2925194246Smarius cas_detach(sc); 2926194246Smarius CAS_LOCK_DESTROY(sc); 2927194246Smarius bus_release_resources(dev, cas_pci_res_spec, sc->sc_res); 2928194246Smarius return (0); 2929194246Smarius} 2930194246Smarius 2931194246Smariusstatic int 2932194246Smariuscas_pci_suspend(device_t dev) 2933194246Smarius{ 2934194246Smarius 2935194246Smarius cas_suspend(device_get_softc(dev)); 2936194246Smarius return (0); 2937194246Smarius} 2938194246Smarius 2939194246Smariusstatic int 2940194246Smariuscas_pci_resume(device_t dev) 2941194246Smarius{ 2942194246Smarius 2943194246Smarius cas_resume(device_get_softc(dev)); 2944194246Smarius return (0); 2945194246Smarius} 2946