7aeac9ef18
Run on LLNW canaries and tested by pho@ gallatin: Using a 14-core, 28-HTT single socket E5-2697 v3 with a 40GbE MLX5 based ConnectX 4-LX NIC, I see an almost 12% improvement in received packet rate, and a larger improvement in bytes delivered all the way to userspace. When the host receiving 64 streams of netperf -H $DUT -t UDP_STREAM -- -m 1, I see, using nstat -I mce0 1 before the patch: InMpps OMpps InGbs OGbs err TCP Est %CPU syscalls csw irq GBfree 4.98 0.00 4.42 0.00 4235592 33 83.80 4720653 2149771 1235 247.32 4.73 0.00 4.20 0.00 4025260 33 82.99 4724900 2139833 1204 247.32 4.72 0.00 4.20 0.00 4035252 33 82.14 4719162 2132023 1264 247.32 4.71 0.00 4.21 0.00 4073206 33 83.68 4744973 2123317 1347 247.32 4.72 0.00 4.21 0.00 4061118 33 80.82 4713615 2188091 1490 247.32 4.72 0.00 4.21 0.00 4051675 33 85.29 4727399 2109011 1205 247.32 4.73 0.00 4.21 0.00 4039056 33 84.65 4724735 2102603 1053 247.32 After the patch InMpps OMpps InGbs OGbs err TCP Est %CPU syscalls csw irq GBfree 5.43 0.00 4.20 0.00 3313143 33 84.96 5434214 1900162 2656 245.51 5.43 0.00 4.20 0.00 3308527 33 85.24 5439695 1809382 2521 245.51 5.42 0.00 4.19 0.00 3316778 33 87.54 5416028 1805835 2256 245.51 5.42 0.00 4.19 0.00 3317673 33 90.44 5426044 1763056 2332 245.51 5.42 0.00 4.19 0.00 3314839 33 88.11 5435732 1792218 2499 245.52 5.44 0.00 4.19 0.00 3293228 33 91.84 5426301 1668597 2121 245.52 Similarly, netperf reports 230Mb/s before the patch, and 270Mb/s after the patch Reviewed by: gallatin Sponsored by: Limelight Networks Differential Revision: https://reviews.freebsd.org/D15366
1874 lines
44 KiB
C
1874 lines
44 KiB
C
/*-
|
|
* Copyright (c) 2016 Hiroki Mori. All rights reserved.
|
|
* Copyright (C) 2007
|
|
* Oleksandr Tymoshenko <gonzo@freebsd.org>. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWFV IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE FV DISCLAIMED.
|
|
* IN NO EVENT SHALL THE AUTHOR OR HIS RELATIVES BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF MIND, USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWFV, EVEN IF ADVISED OF
|
|
* THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* $Id: $
|
|
*
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
/*
|
|
* FV Ethernet interface driver
|
|
* copy from mips/idt/if_kr.c and netbsd code
|
|
*/
|
|
#include <sys/param.h>
|
|
#include <sys/endian.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/sockio.h>
|
|
#include <sys/mbuf.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/lock.h>
|
|
#include <sys/module.h>
|
|
#include <sys/mutex.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/taskqueue.h>
|
|
|
|
#include <net/if.h>
|
|
#include <net/if_arp.h>
|
|
#include <net/ethernet.h>
|
|
#include <net/if_dl.h>
|
|
#include <net/if_media.h>
|
|
#include <net/if_types.h>
|
|
#include <net/if_var.h>
|
|
|
|
#include <net/bpf.h>
|
|
|
|
#include <dev/ofw/ofw_bus.h>
|
|
#include <dev/ofw/ofw_bus_subr.h>
|
|
|
|
#include <machine/bus.h>
|
|
#include <machine/resource.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/rman.h>
|
|
|
|
#include <dev/mii/mii.h>
|
|
#include <dev/mii/miivar.h>
|
|
|
|
/* Todo: move to options.arm */
|
|
/*#define FV_MDIO*/
|
|
|
|
#ifdef FV_MDIO
|
|
#include <dev/mdio/mdio.h>
|
|
#include <dev/etherswitch/miiproxy.h>
|
|
#include "mdio_if.h"
|
|
#endif
|
|
|
|
MODULE_DEPEND(are, ether, 1, 1, 1);
|
|
MODULE_DEPEND(are, miibus, 1, 1, 1);
|
|
#ifdef FV_MDIO
|
|
MODULE_DEPEND(are, mdio, 1, 1, 1);
|
|
#endif
|
|
|
|
#include "miibus_if.h"
|
|
|
|
#include <arm/ralink/if_fvreg.h>
|
|
|
|
#ifdef FV_DEBUG
|
|
void dump_txdesc(struct fv_softc *, int);
|
|
void dump_status_reg(struct fv_softc *);
|
|
#endif
|
|
|
|
static int fv_attach(device_t);
|
|
static int fv_detach(device_t);
|
|
static int fv_ifmedia_upd(struct ifnet *);
|
|
static void fv_ifmedia_sts(struct ifnet *, struct ifmediareq *);
|
|
static int fv_ioctl(struct ifnet *, u_long, caddr_t);
|
|
static void fv_init(void *);
|
|
static void fv_init_locked(struct fv_softc *);
|
|
static void fv_link_task(void *, int);
|
|
static int fv_miibus_readreg(device_t, int, int);
|
|
static void fv_miibus_statchg(device_t);
|
|
static int fv_miibus_writereg(device_t, int, int, int);
|
|
static int fv_probe(device_t);
|
|
static void fv_reset(struct fv_softc *);
|
|
static int fv_resume(device_t);
|
|
static int fv_rx_ring_init(struct fv_softc *);
|
|
static int fv_tx_ring_init(struct fv_softc *);
|
|
static int fv_shutdown(device_t);
|
|
static void fv_start(struct ifnet *);
|
|
static void fv_start_locked(struct ifnet *);
|
|
static void fv_stop(struct fv_softc *);
|
|
static int fv_suspend(device_t);
|
|
|
|
static void fv_rx(struct fv_softc *);
|
|
static void fv_tx(struct fv_softc *);
|
|
static void fv_intr(void *);
|
|
static void fv_tick(void *);
|
|
|
|
static void fv_dmamap_cb(void *, bus_dma_segment_t *, int, int);
|
|
static int fv_dma_alloc(struct fv_softc *);
|
|
static void fv_dma_free(struct fv_softc *);
|
|
static int fv_newbuf(struct fv_softc *, int);
|
|
static __inline void fv_fixup_rx(struct mbuf *);
|
|
|
|
static void fv_hinted_child(device_t bus, const char *dname, int dunit);
|
|
|
|
static void fv_setfilt(struct fv_softc *sc);
|
|
|
|
static device_method_t fv_methods[] = {
|
|
/* Device interface */
|
|
DEVMETHOD(device_probe, fv_probe),
|
|
DEVMETHOD(device_attach, fv_attach),
|
|
DEVMETHOD(device_detach, fv_detach),
|
|
DEVMETHOD(device_suspend, fv_suspend),
|
|
DEVMETHOD(device_resume, fv_resume),
|
|
DEVMETHOD(device_shutdown, fv_shutdown),
|
|
|
|
/* MII interface */
|
|
DEVMETHOD(miibus_readreg, fv_miibus_readreg),
|
|
DEVMETHOD(miibus_writereg, fv_miibus_writereg),
|
|
#if !defined(FV_MDIO)
|
|
DEVMETHOD(miibus_statchg, fv_miibus_statchg),
|
|
#endif
|
|
|
|
/* bus interface */
|
|
DEVMETHOD(bus_add_child, device_add_child_ordered),
|
|
DEVMETHOD(bus_hinted_child, fv_hinted_child),
|
|
|
|
DEVMETHOD_END
|
|
};
|
|
|
|
static driver_t fv_driver = {
|
|
"fv",
|
|
fv_methods,
|
|
sizeof(struct fv_softc)
|
|
};
|
|
|
|
static devclass_t fv_devclass;
|
|
|
|
DRIVER_MODULE(fv, simplebus, fv_driver, fv_devclass, 0, 0);
|
|
#ifdef MII
|
|
DRIVER_MODULE(miibus, fv, miibus_driver, miibus_devclass, 0, 0);
|
|
#endif
|
|
|
|
static struct mtx miibus_mtx;
|
|
MTX_SYSINIT(miibus_mtx, &miibus_mtx, "are mii lock", MTX_DEF);
|
|
|
|
#ifdef FV_MDIO
|
|
static int fvmdio_probe(device_t);
|
|
static int fvmdio_attach(device_t);
|
|
static int fvmdio_detach(device_t);
|
|
|
|
/*
|
|
* Declare an additional, separate driver for accessing the MDIO bus.
|
|
*/
|
|
static device_method_t fvmdio_methods[] = {
|
|
/* Device interface */
|
|
DEVMETHOD(device_probe, fvmdio_probe),
|
|
DEVMETHOD(device_attach, fvmdio_attach),
|
|
DEVMETHOD(device_detach, fvmdio_detach),
|
|
|
|
/* bus interface */
|
|
DEVMETHOD(bus_add_child, device_add_child_ordered),
|
|
|
|
/* MDIO access */
|
|
DEVMETHOD(mdio_readreg, fv_miibus_readreg),
|
|
DEVMETHOD(mdio_writereg, fv_miibus_writereg),
|
|
};
|
|
|
|
DEFINE_CLASS_0(fvmdio, fvmdio_driver, fvmdio_methods,
|
|
sizeof(struct fv_softc));
|
|
static devclass_t fvmdio_devclass;
|
|
|
|
DRIVER_MODULE(miiproxy, fv, miiproxy_driver, miiproxy_devclass, 0, 0);
|
|
DRIVER_MODULE(fvmdio, simplebus, fvmdio_driver, fvmdio_devclass, 0, 0);
|
|
DRIVER_MODULE(mdio, fvmdio, mdio_driver, mdio_devclass, 0, 0);
|
|
#endif
|
|
|
|
/* setup frame code refer dc code */
|
|
|
|
static void
|
|
fv_setfilt(struct fv_softc *sc)
|
|
{
|
|
uint16_t eaddr[(ETHER_ADDR_LEN+1)/2];
|
|
struct fv_desc *sframe;
|
|
int i;
|
|
struct ifnet *ifp;
|
|
struct ifmultiaddr *ifma;
|
|
uint16_t *sp;
|
|
uint8_t *ma;
|
|
|
|
ifp = sc->fv_ifp;
|
|
|
|
i = sc->fv_cdata.fv_tx_prod;
|
|
FV_INC(sc->fv_cdata.fv_tx_prod, FV_TX_RING_CNT);
|
|
sc->fv_cdata.fv_tx_cnt++;
|
|
sframe = &sc->fv_rdata.fv_tx_ring[i];
|
|
sp = (uint16_t *)sc->fv_cdata.fv_sf_buff;
|
|
memset(sp, 0xff, FV_SFRAME_LEN);
|
|
|
|
sframe->fv_addr = sc->fv_rdata.fv_sf_paddr;
|
|
sframe->fv_devcs = ADCTL_Tx_SETUP | FV_DMASIZE(FV_SFRAME_LEN);
|
|
|
|
i = 0;
|
|
if_maddr_rlock(ifp);
|
|
CK_STAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) {
|
|
if (ifma->ifma_addr->sa_family != AF_LINK)
|
|
continue;
|
|
ma = LLADDR((struct sockaddr_dl *)ifma->ifma_addr);
|
|
sp[i] = sp[i+1] = (ma[1] << 8 | ma[0]);
|
|
i += 2;
|
|
sp[i] = sp[i+1] = (ma[3] << 8 | ma[2]);
|
|
i += 2;
|
|
sp[i] = sp[i+1] = (ma[5] << 8 | ma[4]);
|
|
i += 2;
|
|
}
|
|
if_maddr_runlock(ifp);
|
|
|
|
bcopy(IF_LLADDR(sc->fv_ifp), eaddr, ETHER_ADDR_LEN);
|
|
sp[90] = sp[91] = eaddr[0];
|
|
sp[92] = sp[93] = eaddr[1];
|
|
sp[94] = sp[95] = eaddr[2];
|
|
|
|
sframe->fv_stat = ADSTAT_OWN;
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map, BUS_DMASYNC_PREWRITE);
|
|
bus_dmamap_sync(sc->fv_cdata.fv_sf_tag,
|
|
sc->fv_cdata.fv_sf_buff_map, BUS_DMASYNC_PREWRITE);
|
|
CSR_WRITE_4(sc, CSR_TXPOLL, 0xFFFFFFFF);
|
|
DELAY(10000);
|
|
}
|
|
|
|
static int
|
|
fv_probe(device_t dev)
|
|
{
|
|
|
|
if (!ofw_bus_status_okay(dev))
|
|
return (ENXIO);
|
|
|
|
if (!ofw_bus_is_compatible(dev, "fv,ethernet"))
|
|
return (ENXIO);
|
|
|
|
device_set_desc(dev, "FV Ethernet interface");
|
|
return (BUS_PROBE_DEFAULT);
|
|
}
|
|
|
|
static int
|
|
fv_attach(device_t dev)
|
|
{
|
|
struct ifnet *ifp;
|
|
struct fv_softc *sc;
|
|
int error = 0, rid;
|
|
int unit;
|
|
int i;
|
|
|
|
sc = device_get_softc(dev);
|
|
unit = device_get_unit(dev);
|
|
sc->fv_dev = dev;
|
|
sc->fv_ofw = ofw_bus_get_node(dev);
|
|
|
|
i = OF_getprop(sc->fv_ofw, "local-mac-address", (void *)&sc->fv_eaddr, 6);
|
|
if (i != 6) {
|
|
/* hardcode macaddress */
|
|
sc->fv_eaddr[0] = 0x00;
|
|
sc->fv_eaddr[1] = 0x0C;
|
|
sc->fv_eaddr[2] = 0x42;
|
|
sc->fv_eaddr[3] = 0x09;
|
|
sc->fv_eaddr[4] = 0x5E;
|
|
sc->fv_eaddr[5] = 0x6B;
|
|
}
|
|
|
|
mtx_init(&sc->fv_mtx, device_get_nameunit(dev), MTX_NETWORK_LOCK,
|
|
MTX_DEF);
|
|
callout_init_mtx(&sc->fv_stat_callout, &sc->fv_mtx, 0);
|
|
TASK_INIT(&sc->fv_link_task, 0, fv_link_task, sc);
|
|
|
|
/* Map control/status registers. */
|
|
sc->fv_rid = 0;
|
|
sc->fv_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &sc->fv_rid,
|
|
RF_ACTIVE | RF_SHAREABLE);
|
|
|
|
if (sc->fv_res == NULL) {
|
|
device_printf(dev, "couldn't map memory\n");
|
|
error = ENXIO;
|
|
goto fail;
|
|
}
|
|
|
|
sc->fv_btag = rman_get_bustag(sc->fv_res);
|
|
sc->fv_bhandle = rman_get_bushandle(sc->fv_res);
|
|
|
|
/* Allocate interrupts */
|
|
rid = 0;
|
|
sc->fv_irq = bus_alloc_resource_any(dev, SYS_RES_IRQ, &rid,
|
|
RF_SHAREABLE | RF_ACTIVE);
|
|
|
|
if (sc->fv_irq == NULL) {
|
|
device_printf(dev, "couldn't map interrupt\n");
|
|
error = ENXIO;
|
|
goto fail;
|
|
}
|
|
|
|
/* Allocate ifnet structure. */
|
|
ifp = sc->fv_ifp = if_alloc(IFT_ETHER);
|
|
|
|
if (ifp == NULL) {
|
|
device_printf(dev, "couldn't allocate ifnet structure\n");
|
|
error = ENOSPC;
|
|
goto fail;
|
|
}
|
|
ifp->if_softc = sc;
|
|
if_initname(ifp, device_get_name(dev), device_get_unit(dev));
|
|
ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
|
|
ifp->if_ioctl = fv_ioctl;
|
|
ifp->if_start = fv_start;
|
|
ifp->if_init = fv_init;
|
|
|
|
/* ifqmaxlen is sysctl value in net/if.c */
|
|
IFQ_SET_MAXLEN(&ifp->if_snd, ifqmaxlen);
|
|
ifp->if_snd.ifq_maxlen = ifqmaxlen;
|
|
IFQ_SET_READY(&ifp->if_snd);
|
|
|
|
ifp->if_capenable = ifp->if_capabilities;
|
|
|
|
if (fv_dma_alloc(sc) != 0) {
|
|
error = ENXIO;
|
|
goto fail;
|
|
}
|
|
|
|
/* TODO: calculate prescale */
|
|
/*
|
|
CSR_WRITE_4(sc, FV_ETHMCP, (165000000 / (1250000 + 1)) & ~1);
|
|
|
|
CSR_WRITE_4(sc, FV_MIIMCFG, FV_MIIMCFG_R);
|
|
DELAY(1000);
|
|
CSR_WRITE_4(sc, FV_MIIMCFG, 0);
|
|
*/
|
|
CSR_WRITE_4(sc, CSR_BUSMODE, BUSMODE_SWR);
|
|
DELAY(1000);
|
|
|
|
#ifdef FV_MDIO
|
|
sc->fv_miiproxy = mii_attach_proxy(sc->fv_dev);
|
|
#endif
|
|
|
|
#ifdef MII
|
|
/* Do MII setup. */
|
|
error = mii_attach(dev, &sc->fv_miibus, ifp, fv_ifmedia_upd,
|
|
fv_ifmedia_sts, BMSR_DEFCAPMASK, MII_PHY_ANY, MII_OFFSET_ANY, 0);
|
|
if (error != 0) {
|
|
device_printf(dev, "attaching PHYs failed\n");
|
|
goto fail;
|
|
}
|
|
#else
|
|
ifmedia_init(&sc->fv_ifmedia, 0, fv_ifmedia_upd, fv_ifmedia_sts);
|
|
|
|
ifmedia_add(&sc->fv_ifmedia, IFM_ETHER | IFM_AUTO, 0, NULL);
|
|
ifmedia_set(&sc->fv_ifmedia, IFM_ETHER | IFM_AUTO);
|
|
#endif
|
|
|
|
/* Call MI attach routine. */
|
|
ether_ifattach(ifp, sc->fv_eaddr);
|
|
|
|
/* Hook interrupt last to avoid having to lock softc */
|
|
error = bus_setup_intr(dev, sc->fv_irq, INTR_TYPE_NET | INTR_MPSAFE,
|
|
NULL, fv_intr, sc, &sc->fv_intrhand);
|
|
|
|
if (error) {
|
|
device_printf(dev, "couldn't set up irq\n");
|
|
ether_ifdetach(ifp);
|
|
goto fail;
|
|
}
|
|
|
|
fail:
|
|
if (error)
|
|
fv_detach(dev);
|
|
|
|
return (error);
|
|
}
|
|
|
|
static int
|
|
fv_detach(device_t dev)
|
|
{
|
|
struct fv_softc *sc = device_get_softc(dev);
|
|
struct ifnet *ifp = sc->fv_ifp;
|
|
|
|
KASSERT(mtx_initialized(&sc->fv_mtx), ("vr mutex not initialized"));
|
|
|
|
/* These should only be active if attach succeeded */
|
|
if (device_is_attached(dev)) {
|
|
FV_LOCK(sc);
|
|
sc->fv_detach = 1;
|
|
fv_stop(sc);
|
|
FV_UNLOCK(sc);
|
|
taskqueue_drain(taskqueue_swi, &sc->fv_link_task);
|
|
ether_ifdetach(ifp);
|
|
}
|
|
#ifdef MII
|
|
if (sc->fv_miibus)
|
|
device_delete_child(dev, sc->fv_miibus);
|
|
#endif
|
|
bus_generic_detach(dev);
|
|
|
|
if (sc->fv_intrhand)
|
|
bus_teardown_intr(dev, sc->fv_irq, sc->fv_intrhand);
|
|
if (sc->fv_irq)
|
|
bus_release_resource(dev, SYS_RES_IRQ, 0, sc->fv_irq);
|
|
|
|
if (sc->fv_res)
|
|
bus_release_resource(dev, SYS_RES_MEMORY, sc->fv_rid,
|
|
sc->fv_res);
|
|
|
|
if (ifp)
|
|
if_free(ifp);
|
|
|
|
fv_dma_free(sc);
|
|
|
|
mtx_destroy(&sc->fv_mtx);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
static int
|
|
fv_suspend(device_t dev)
|
|
{
|
|
|
|
panic("%s", __func__);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
fv_resume(device_t dev)
|
|
{
|
|
|
|
panic("%s", __func__);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
fv_shutdown(device_t dev)
|
|
{
|
|
struct fv_softc *sc;
|
|
|
|
sc = device_get_softc(dev);
|
|
|
|
FV_LOCK(sc);
|
|
fv_stop(sc);
|
|
FV_UNLOCK(sc);
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
fv_miibus_readbits(struct fv_softc *sc, int count)
|
|
{
|
|
int result;
|
|
|
|
result = 0;
|
|
while(count--) {
|
|
result <<= 1;
|
|
CSR_WRITE_4(sc, CSR_MIIMNG, MII_RD);
|
|
DELAY(10);
|
|
CSR_WRITE_4(sc, CSR_MIIMNG, MII_RD | MII_CLK);
|
|
DELAY(10);
|
|
if (CSR_READ_4(sc, CSR_MIIMNG) & MII_DIN)
|
|
result |= 1;
|
|
}
|
|
|
|
return (result);
|
|
}
|
|
|
|
static int
|
|
fv_miibus_writebits(struct fv_softc *sc, int data, int count)
|
|
{
|
|
int bit;
|
|
|
|
while(count--) {
|
|
bit = ((data) >> count) & 0x1 ? MII_DOUT : 0;
|
|
CSR_WRITE_4(sc, CSR_MIIMNG, bit | MII_WR);
|
|
DELAY(10);
|
|
CSR_WRITE_4(sc, CSR_MIIMNG, bit | MII_WR | MII_CLK);
|
|
DELAY(10);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
fv_miibus_turnaround(struct fv_softc *sc, int cmd)
|
|
{
|
|
if (cmd == MII_WRCMD) {
|
|
fv_miibus_writebits(sc, 0x02, 2);
|
|
} else {
|
|
fv_miibus_readbits(sc, 1);
|
|
}
|
|
}
|
|
|
|
static int
|
|
fv_miibus_readreg(device_t dev, int phy, int reg)
|
|
{
|
|
struct fv_softc * sc = device_get_softc(dev);
|
|
int result;
|
|
|
|
mtx_lock(&miibus_mtx);
|
|
fv_miibus_writebits(sc, MII_PREAMBLE, 32);
|
|
fv_miibus_writebits(sc, MII_RDCMD, 4);
|
|
fv_miibus_writebits(sc, phy, 5);
|
|
fv_miibus_writebits(sc, reg, 5);
|
|
fv_miibus_turnaround(sc, MII_RDCMD);
|
|
result = fv_miibus_readbits(sc, 16);
|
|
fv_miibus_turnaround(sc, MII_RDCMD);
|
|
mtx_unlock(&miibus_mtx);
|
|
|
|
return (result);
|
|
}
|
|
|
|
static int
|
|
fv_miibus_writereg(device_t dev, int phy, int reg, int data)
|
|
{
|
|
struct fv_softc * sc = device_get_softc(dev);
|
|
|
|
mtx_lock(&miibus_mtx);
|
|
fv_miibus_writebits(sc, MII_PREAMBLE, 32);
|
|
fv_miibus_writebits(sc, MII_WRCMD, 4);
|
|
fv_miibus_writebits(sc, phy, 5);
|
|
fv_miibus_writebits(sc, reg, 5);
|
|
fv_miibus_turnaround(sc, MII_WRCMD);
|
|
fv_miibus_writebits(sc, data, 16);
|
|
mtx_unlock(&miibus_mtx);
|
|
|
|
return (0);
|
|
}
|
|
|
|
#if !defined(FV_MDIO)
|
|
static void
|
|
fv_miibus_statchg(device_t dev)
|
|
{
|
|
struct fv_softc *sc;
|
|
|
|
sc = device_get_softc(dev);
|
|
taskqueue_enqueue(taskqueue_swi, &sc->fv_link_task);
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
fv_link_task(void *arg, int pending)
|
|
{
|
|
#ifdef MII
|
|
struct fv_softc *sc;
|
|
struct mii_data *mii;
|
|
struct ifnet *ifp;
|
|
/* int lfdx, mfdx; */
|
|
|
|
sc = (struct fv_softc *)arg;
|
|
|
|
FV_LOCK(sc);
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
ifp = sc->fv_ifp;
|
|
if (mii == NULL || ifp == NULL ||
|
|
(ifp->if_drv_flags & IFF_DRV_RUNNING) == 0) {
|
|
FV_UNLOCK(sc);
|
|
return;
|
|
}
|
|
|
|
if (mii->mii_media_status & IFM_ACTIVE) {
|
|
if (IFM_SUBTYPE(mii->mii_media_active) != IFM_NONE)
|
|
sc->fv_link_status = 1;
|
|
} else
|
|
sc->fv_link_status = 0;
|
|
|
|
FV_UNLOCK(sc);
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fv_reset(struct fv_softc *sc)
|
|
{
|
|
int i;
|
|
|
|
CSR_WRITE_4(sc, CSR_BUSMODE, BUSMODE_SWR);
|
|
|
|
/*
|
|
* The chip doesn't take itself out of reset automatically.
|
|
* We need to do so after 2us.
|
|
*/
|
|
DELAY(1000);
|
|
CSR_WRITE_4(sc, CSR_BUSMODE, 0);
|
|
|
|
for (i = 0; i < 1000; i++) {
|
|
/*
|
|
* Wait a bit for the reset to complete before peeking
|
|
* at the chip again.
|
|
*/
|
|
DELAY(1000);
|
|
if ((CSR_READ_4(sc, CSR_BUSMODE) & BUSMODE_SWR) == 0)
|
|
break;
|
|
}
|
|
|
|
if (CSR_READ_4(sc, CSR_BUSMODE) & BUSMODE_SWR)
|
|
device_printf(sc->fv_dev, "reset time out\n");
|
|
|
|
DELAY(1000);
|
|
}
|
|
|
|
static void
|
|
fv_init(void *xsc)
|
|
{
|
|
struct fv_softc *sc = xsc;
|
|
|
|
FV_LOCK(sc);
|
|
fv_init_locked(sc);
|
|
FV_UNLOCK(sc);
|
|
}
|
|
|
|
static void
|
|
fv_init_locked(struct fv_softc *sc)
|
|
{
|
|
struct ifnet *ifp = sc->fv_ifp;
|
|
#ifdef MII
|
|
struct mii_data *mii;
|
|
#endif
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
#ifdef MII
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
#endif
|
|
|
|
fv_stop(sc);
|
|
fv_reset(sc);
|
|
|
|
/* Init circular RX list. */
|
|
if (fv_rx_ring_init(sc) != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"initialization failed: no memory for rx buffers\n");
|
|
fv_stop(sc);
|
|
return;
|
|
}
|
|
|
|
/* Init tx descriptors. */
|
|
fv_tx_ring_init(sc);
|
|
|
|
/*
|
|
* Initialize the BUSMODE register.
|
|
*/
|
|
CSR_WRITE_4(sc, CSR_BUSMODE,
|
|
/* XXX: not sure if this is a good thing or not... */
|
|
BUSMODE_BAR | BUSMODE_PBL_32LW);
|
|
|
|
/*
|
|
* Initialize the interrupt mask and enable interrupts.
|
|
*/
|
|
/* normal interrupts */
|
|
sc->sc_inten = STATUS_TI | STATUS_TU | STATUS_RI | STATUS_NIS;
|
|
|
|
/* abnormal interrupts */
|
|
sc->sc_inten |= STATUS_TPS | STATUS_TJT | STATUS_UNF |
|
|
STATUS_RU | STATUS_RPS | STATUS_SE | STATUS_AIS;
|
|
|
|
sc->sc_rxint_mask = STATUS_RI|STATUS_RU;
|
|
sc->sc_txint_mask = STATUS_TI|STATUS_UNF|STATUS_TJT;
|
|
|
|
sc->sc_rxint_mask &= sc->sc_inten;
|
|
sc->sc_txint_mask &= sc->sc_inten;
|
|
|
|
CSR_WRITE_4(sc, CSR_INTEN, sc->sc_inten);
|
|
CSR_WRITE_4(sc, CSR_STATUS, 0xffffffff);
|
|
|
|
/*
|
|
* Give the transmit and receive rings to the chip.
|
|
*/
|
|
CSR_WRITE_4(sc, CSR_TXLIST, FV_TX_RING_ADDR(sc, 0));
|
|
CSR_WRITE_4(sc, CSR_RXLIST, FV_RX_RING_ADDR(sc, 0));
|
|
|
|
/*
|
|
* Set the station address.
|
|
*/
|
|
fv_setfilt(sc);
|
|
|
|
|
|
/*
|
|
* Write out the opmode.
|
|
*/
|
|
CSR_WRITE_4(sc, CSR_OPMODE, OPMODE_SR | OPMODE_ST |
|
|
OPMODE_TR_128 | OPMODE_FDX | OPMODE_SPEED);
|
|
/*
|
|
* Start the receive process.
|
|
*/
|
|
CSR_WRITE_4(sc, CSR_RXPOLL, RXPOLL_RPD);
|
|
|
|
sc->fv_link_status = 1;
|
|
#ifdef MII
|
|
mii_mediachg(mii);
|
|
#endif
|
|
|
|
ifp->if_drv_flags |= IFF_DRV_RUNNING;
|
|
ifp->if_drv_flags &= ~IFF_DRV_OACTIVE;
|
|
|
|
callout_reset(&sc->fv_stat_callout, hz, fv_tick, sc);
|
|
}
|
|
|
|
static void
|
|
fv_start(struct ifnet *ifp)
|
|
{
|
|
struct fv_softc *sc;
|
|
|
|
sc = ifp->if_softc;
|
|
|
|
FV_LOCK(sc);
|
|
fv_start_locked(ifp);
|
|
FV_UNLOCK(sc);
|
|
}
|
|
|
|
/*
|
|
* Encapsulate an mbuf chain in a descriptor by coupling the mbuf data
|
|
* pointers to the fragment pointers.
|
|
* Use Implicit Chain implementation.
|
|
*/
|
|
static int
|
|
fv_encap(struct fv_softc *sc, struct mbuf **m_head)
|
|
{
|
|
struct fv_txdesc *txd;
|
|
struct fv_desc *desc;
|
|
struct mbuf *m;
|
|
bus_dma_segment_t txsegs[FV_MAXFRAGS];
|
|
int error, i, nsegs, prod, si;
|
|
int padlen;
|
|
int txstat;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
/*
|
|
* Some VIA Rhine wants packet buffers to be longword
|
|
* aligned, but very often our mbufs aren't. Rather than
|
|
* waste time trying to decide when to copy and when not
|
|
* to copy, just do it all the time.
|
|
*/
|
|
m = m_defrag(*m_head, M_NOWAIT);
|
|
if (m == NULL) {
|
|
device_printf(sc->fv_dev, "fv_encap m_defrag error\n");
|
|
m_freem(*m_head);
|
|
*m_head = NULL;
|
|
return (ENOBUFS);
|
|
}
|
|
*m_head = m;
|
|
|
|
/*
|
|
* The Rhine chip doesn't auto-pad, so we have to make
|
|
* sure to pad short frames out to the minimum frame length
|
|
* ourselves.
|
|
*/
|
|
if ((*m_head)->m_pkthdr.len < FV_MIN_FRAMELEN) {
|
|
m = *m_head;
|
|
padlen = FV_MIN_FRAMELEN - m->m_pkthdr.len;
|
|
if (M_WRITABLE(m) == 0) {
|
|
/* Get a writable copy. */
|
|
m = m_dup(*m_head, M_NOWAIT);
|
|
m_freem(*m_head);
|
|
if (m == NULL) {
|
|
device_printf(sc->fv_dev, "fv_encap m_dup error\n");
|
|
*m_head = NULL;
|
|
return (ENOBUFS);
|
|
}
|
|
*m_head = m;
|
|
}
|
|
if (m->m_next != NULL || M_TRAILINGSPACE(m) < padlen) {
|
|
m = m_defrag(m, M_NOWAIT);
|
|
if (m == NULL) {
|
|
device_printf(sc->fv_dev, "fv_encap m_defrag error\n");
|
|
m_freem(*m_head);
|
|
*m_head = NULL;
|
|
return (ENOBUFS);
|
|
}
|
|
}
|
|
/*
|
|
* Manually pad short frames, and zero the pad space
|
|
* to avoid leaking data.
|
|
*/
|
|
bzero(mtod(m, char *) + m->m_pkthdr.len, padlen);
|
|
m->m_pkthdr.len += padlen;
|
|
m->m_len = m->m_pkthdr.len;
|
|
*m_head = m;
|
|
}
|
|
|
|
prod = sc->fv_cdata.fv_tx_prod;
|
|
txd = &sc->fv_cdata.fv_txdesc[prod];
|
|
error = bus_dmamap_load_mbuf_sg(sc->fv_cdata.fv_tx_tag, txd->tx_dmamap,
|
|
*m_head, txsegs, &nsegs, BUS_DMA_NOWAIT);
|
|
if (error == EFBIG) {
|
|
device_printf(sc->fv_dev, "fv_encap EFBIG error\n");
|
|
m = m_defrag(*m_head, M_NOWAIT);
|
|
if (m == NULL) {
|
|
m_freem(*m_head);
|
|
*m_head = NULL;
|
|
return (ENOBUFS);
|
|
}
|
|
*m_head = m;
|
|
error = bus_dmamap_load_mbuf_sg(sc->fv_cdata.fv_tx_tag,
|
|
txd->tx_dmamap, *m_head, txsegs, &nsegs, BUS_DMA_NOWAIT);
|
|
if (error != 0) {
|
|
m_freem(*m_head);
|
|
*m_head = NULL;
|
|
return (error);
|
|
}
|
|
|
|
} else if (error != 0)
|
|
return (error);
|
|
if (nsegs == 0) {
|
|
m_freem(*m_head);
|
|
*m_head = NULL;
|
|
return (EIO);
|
|
}
|
|
|
|
/* Check number of available descriptors. */
|
|
if (sc->fv_cdata.fv_tx_cnt + nsegs >= (FV_TX_RING_CNT - 1)) {
|
|
bus_dmamap_unload(sc->fv_cdata.fv_tx_tag, txd->tx_dmamap);
|
|
return (ENOBUFS);
|
|
}
|
|
|
|
txd->tx_m = *m_head;
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_tag, txd->tx_dmamap,
|
|
BUS_DMASYNC_PREWRITE);
|
|
|
|
si = prod;
|
|
|
|
/*
|
|
* Make a list of descriptors for this packet.
|
|
*/
|
|
desc = NULL;
|
|
for (i = 0; i < nsegs; i++) {
|
|
desc = &sc->fv_rdata.fv_tx_ring[prod];
|
|
desc->fv_stat = ADSTAT_OWN;
|
|
desc->fv_devcs = txsegs[i].ds_len;
|
|
/* end of descriptor */
|
|
if (prod == FV_TX_RING_CNT - 1)
|
|
desc->fv_devcs |= ADCTL_ER;
|
|
desc->fv_addr = txsegs[i].ds_addr;
|
|
|
|
++sc->fv_cdata.fv_tx_cnt;
|
|
FV_INC(prod, FV_TX_RING_CNT);
|
|
}
|
|
|
|
/*
|
|
* Set mark last fragment with Last/Intr flag
|
|
*/
|
|
if (desc) {
|
|
desc->fv_devcs |= ADCTL_Tx_IC;
|
|
desc->fv_devcs |= ADCTL_Tx_LS;
|
|
}
|
|
|
|
/* Update producer index. */
|
|
sc->fv_cdata.fv_tx_prod = prod;
|
|
|
|
/* Sync descriptors. */
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map,
|
|
BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
|
|
|
|
txstat = (CSR_READ_4(sc, CSR_STATUS) >> 20) & 7;
|
|
if (txstat == 0 || txstat == 6) {
|
|
/* Transmit Process Stat is stop or suspended */
|
|
desc = &sc->fv_rdata.fv_tx_ring[si];
|
|
desc->fv_devcs |= ADCTL_Tx_FS;
|
|
}
|
|
else {
|
|
/* Get previous descriptor */
|
|
si = (si + FV_TX_RING_CNT - 1) % FV_TX_RING_CNT;
|
|
desc = &sc->fv_rdata.fv_tx_ring[si];
|
|
/* join remain data and flugs */
|
|
desc->fv_devcs &= ~ADCTL_Tx_IC;
|
|
desc->fv_devcs &= ~ADCTL_Tx_LS;
|
|
}
|
|
|
|
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
fv_start_locked(struct ifnet *ifp)
|
|
{
|
|
struct fv_softc *sc;
|
|
struct mbuf *m_head;
|
|
int enq;
|
|
int txstat;
|
|
|
|
sc = ifp->if_softc;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
if ((ifp->if_drv_flags & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) !=
|
|
IFF_DRV_RUNNING || sc->fv_link_status == 0 )
|
|
return;
|
|
|
|
for (enq = 0; !IFQ_DRV_IS_EMPTY(&ifp->if_snd) &&
|
|
sc->fv_cdata.fv_tx_cnt < FV_TX_RING_CNT - 2; ) {
|
|
IFQ_DRV_DEQUEUE(&ifp->if_snd, m_head);
|
|
if (m_head == NULL)
|
|
break;
|
|
/*
|
|
* Pack the data into the transmit ring. If we
|
|
* don't have room, set the OACTIVE flag and wait
|
|
* for the NIC to drain the ring.
|
|
*/
|
|
if (fv_encap(sc, &m_head)) {
|
|
if (m_head == NULL)
|
|
break;
|
|
IFQ_DRV_PREPEND(&ifp->if_snd, m_head);
|
|
ifp->if_drv_flags |= IFF_DRV_OACTIVE;
|
|
break;
|
|
}
|
|
|
|
enq++;
|
|
/*
|
|
* If there's a BPF listener, bounce a copy of this frame
|
|
* to him.
|
|
*/
|
|
ETHER_BPF_MTAP(ifp, m_head);
|
|
}
|
|
|
|
if (enq > 0) {
|
|
txstat = (CSR_READ_4(sc, CSR_STATUS) >> 20) & 7;
|
|
if (txstat == 0 || txstat == 6)
|
|
CSR_WRITE_4(sc, CSR_TXPOLL, TXPOLL_TPD);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fv_stop(struct fv_softc *sc)
|
|
{
|
|
struct ifnet *ifp;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
ifp = sc->fv_ifp;
|
|
ifp->if_drv_flags &= ~(IFF_DRV_RUNNING | IFF_DRV_OACTIVE);
|
|
callout_stop(&sc->fv_stat_callout);
|
|
|
|
/* Disable interrupts. */
|
|
CSR_WRITE_4(sc, CSR_INTEN, 0);
|
|
|
|
/* Stop the transmit and receive processes. */
|
|
CSR_WRITE_4(sc, CSR_OPMODE, 0);
|
|
CSR_WRITE_4(sc, CSR_RXLIST, 0);
|
|
CSR_WRITE_4(sc, CSR_TXLIST, 0);
|
|
|
|
}
|
|
|
|
|
|
static int
|
|
fv_ioctl(struct ifnet *ifp, u_long command, caddr_t data)
|
|
{
|
|
struct fv_softc *sc = ifp->if_softc;
|
|
struct ifreq *ifr = (struct ifreq *) data;
|
|
#ifdef MII
|
|
struct mii_data *mii;
|
|
#endif
|
|
int error;
|
|
int csr;
|
|
|
|
switch (command) {
|
|
case SIOCSIFFLAGS:
|
|
FV_LOCK(sc);
|
|
if (ifp->if_flags & IFF_UP) {
|
|
if (ifp->if_drv_flags & IFF_DRV_RUNNING) {
|
|
if ((ifp->if_flags ^ sc->fv_if_flags) &
|
|
IFF_PROMISC) {
|
|
csr = CSR_READ_4(sc, CSR_OPMODE);
|
|
CSR_WRITE_4(sc, CSR_OPMODE, csr |
|
|
OPMODE_PM | OPMODE_PR);
|
|
}
|
|
if ((ifp->if_flags ^ sc->fv_if_flags) &
|
|
IFF_ALLMULTI) {
|
|
csr = CSR_READ_4(sc, CSR_OPMODE);
|
|
CSR_WRITE_4(sc, CSR_OPMODE, csr |
|
|
OPMODE_PM);
|
|
}
|
|
} else {
|
|
if (sc->fv_detach == 0)
|
|
fv_init_locked(sc);
|
|
}
|
|
} else {
|
|
if (ifp->if_drv_flags & IFF_DRV_RUNNING)
|
|
fv_stop(sc);
|
|
}
|
|
sc->fv_if_flags = ifp->if_flags;
|
|
FV_UNLOCK(sc);
|
|
error = 0;
|
|
break;
|
|
case SIOCADDMULTI:
|
|
case SIOCDELMULTI:
|
|
#if 0
|
|
FV_LOCK(sc);
|
|
fv_set_filter(sc);
|
|
FV_UNLOCK(sc);
|
|
#endif
|
|
error = 0;
|
|
break;
|
|
case SIOCGIFMEDIA:
|
|
case SIOCSIFMEDIA:
|
|
#ifdef MII
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
error = ifmedia_ioctl(ifp, ifr, &mii->mii_media, command);
|
|
#else
|
|
error = ifmedia_ioctl(ifp, ifr, &sc->fv_ifmedia, command);
|
|
#endif
|
|
break;
|
|
case SIOCSIFCAP:
|
|
error = 0;
|
|
#if 0
|
|
mask = ifr->ifr_reqcap ^ ifp->if_capenable;
|
|
if ((mask & IFCAP_HWCSUM) != 0) {
|
|
ifp->if_capenable ^= IFCAP_HWCSUM;
|
|
if ((IFCAP_HWCSUM & ifp->if_capenable) &&
|
|
(IFCAP_HWCSUM & ifp->if_capabilities))
|
|
ifp->if_hwassist = FV_CSUM_FEATURES;
|
|
else
|
|
ifp->if_hwassist = 0;
|
|
}
|
|
if ((mask & IFCAP_VLAN_HWTAGGING) != 0) {
|
|
ifp->if_capenable ^= IFCAP_VLAN_HWTAGGING;
|
|
if (IFCAP_VLAN_HWTAGGING & ifp->if_capenable &&
|
|
IFCAP_VLAN_HWTAGGING & ifp->if_capabilities &&
|
|
ifp->if_drv_flags & IFF_DRV_RUNNING) {
|
|
FV_LOCK(sc);
|
|
fv_vlan_setup(sc);
|
|
FV_UNLOCK(sc);
|
|
}
|
|
}
|
|
VLAN_CAPABILITIES(ifp);
|
|
#endif
|
|
break;
|
|
default:
|
|
error = ether_ioctl(ifp, command, data);
|
|
break;
|
|
}
|
|
|
|
return (error);
|
|
}
|
|
|
|
/*
|
|
* Set media options.
|
|
*/
|
|
static int
|
|
fv_ifmedia_upd(struct ifnet *ifp)
|
|
{
|
|
#ifdef MII
|
|
struct fv_softc *sc;
|
|
struct mii_data *mii;
|
|
struct mii_softc *miisc;
|
|
int error;
|
|
|
|
sc = ifp->if_softc;
|
|
FV_LOCK(sc);
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
LIST_FOREACH(miisc, &mii->mii_phys, mii_list)
|
|
PHY_RESET(miisc);
|
|
error = mii_mediachg(mii);
|
|
FV_UNLOCK(sc);
|
|
|
|
return (error);
|
|
#else
|
|
return (0);
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Report current media status.
|
|
*/
|
|
static void
|
|
fv_ifmedia_sts(struct ifnet *ifp, struct ifmediareq *ifmr)
|
|
{
|
|
#ifdef MII
|
|
struct fv_softc *sc = ifp->if_softc;
|
|
struct mii_data *mii;
|
|
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
FV_LOCK(sc);
|
|
mii_pollstat(mii);
|
|
ifmr->ifm_active = mii->mii_media_active;
|
|
ifmr->ifm_status = mii->mii_media_status;
|
|
FV_UNLOCK(sc);
|
|
#else
|
|
ifmr->ifm_status = IFM_AVALID | IFM_ACTIVE;
|
|
#endif
|
|
}
|
|
|
|
struct fv_dmamap_arg {
|
|
bus_addr_t fv_busaddr;
|
|
};
|
|
|
|
static void
|
|
fv_dmamap_cb(void *arg, bus_dma_segment_t *segs, int nseg, int error)
|
|
{
|
|
struct fv_dmamap_arg *ctx;
|
|
|
|
if (error != 0)
|
|
return;
|
|
ctx = arg;
|
|
ctx->fv_busaddr = segs[0].ds_addr;
|
|
}
|
|
|
|
static int
|
|
fv_dma_alloc(struct fv_softc *sc)
|
|
{
|
|
struct fv_dmamap_arg ctx;
|
|
struct fv_txdesc *txd;
|
|
struct fv_rxdesc *rxd;
|
|
int error, i;
|
|
|
|
/* Create parent DMA tag. */
|
|
error = bus_dma_tag_create(
|
|
bus_get_dma_tag(sc->fv_dev), /* parent */
|
|
1, 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR_32BIT, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
BUS_SPACE_MAXSIZE_32BIT, /* maxsize */
|
|
0, /* nsegments */
|
|
BUS_SPACE_MAXSIZE_32BIT, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_parent_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create parent DMA tag\n");
|
|
goto fail;
|
|
}
|
|
/* Create tag for Tx ring. */
|
|
error = bus_dma_tag_create(
|
|
sc->fv_cdata.fv_parent_tag, /* parent */
|
|
FV_RING_ALIGN, 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
FV_TX_RING_SIZE, /* maxsize */
|
|
1, /* nsegments */
|
|
FV_TX_RING_SIZE, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_tx_ring_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create Tx ring DMA tag\n");
|
|
goto fail;
|
|
}
|
|
|
|
/* Create tag for Rx ring. */
|
|
error = bus_dma_tag_create(
|
|
sc->fv_cdata.fv_parent_tag, /* parent */
|
|
FV_RING_ALIGN, 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
FV_RX_RING_SIZE, /* maxsize */
|
|
1, /* nsegments */
|
|
FV_RX_RING_SIZE, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_rx_ring_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create Rx ring DMA tag\n");
|
|
goto fail;
|
|
}
|
|
|
|
/* Create tag for Tx buffers. */
|
|
error = bus_dma_tag_create(
|
|
sc->fv_cdata.fv_parent_tag, /* parent */
|
|
1, 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
MCLBYTES * FV_MAXFRAGS, /* maxsize */
|
|
FV_MAXFRAGS, /* nsegments */
|
|
MCLBYTES, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_tx_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create Tx DMA tag\n");
|
|
goto fail;
|
|
}
|
|
|
|
/* Create tag for Rx buffers. */
|
|
error = bus_dma_tag_create(
|
|
sc->fv_cdata.fv_parent_tag, /* parent */
|
|
FV_RX_ALIGN, 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
MCLBYTES, /* maxsize */
|
|
1, /* nsegments */
|
|
MCLBYTES, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_rx_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create Rx DMA tag\n");
|
|
goto fail;
|
|
}
|
|
|
|
/* Create tag for setup frame buffers. */
|
|
error = bus_dma_tag_create(
|
|
sc->fv_cdata.fv_parent_tag, /* parent */
|
|
sizeof(uint32_t), 0, /* alignment, boundary */
|
|
BUS_SPACE_MAXADDR, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
FV_SFRAME_LEN + FV_MIN_FRAMELEN, /* maxsize */
|
|
1, /* nsegments */
|
|
FV_SFRAME_LEN + FV_MIN_FRAMELEN, /* maxsegsize */
|
|
0, /* flags */
|
|
NULL, NULL, /* lockfunc, lockarg */
|
|
&sc->fv_cdata.fv_sf_tag);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev, "failed to create setup frame DMA tag\n");
|
|
goto fail;
|
|
}
|
|
|
|
/* Allocate DMA'able memory and load the DMA map for Tx ring. */
|
|
error = bus_dmamem_alloc(sc->fv_cdata.fv_tx_ring_tag,
|
|
(void **)&sc->fv_rdata.fv_tx_ring, BUS_DMA_WAITOK |
|
|
BUS_DMA_COHERENT | BUS_DMA_ZERO, &sc->fv_cdata.fv_tx_ring_map);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to allocate DMA'able memory for Tx ring\n");
|
|
goto fail;
|
|
}
|
|
|
|
ctx.fv_busaddr = 0;
|
|
error = bus_dmamap_load(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map, sc->fv_rdata.fv_tx_ring,
|
|
FV_TX_RING_SIZE, fv_dmamap_cb, &ctx, 0);
|
|
if (error != 0 || ctx.fv_busaddr == 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to load DMA'able memory for Tx ring\n");
|
|
goto fail;
|
|
}
|
|
sc->fv_rdata.fv_tx_ring_paddr = ctx.fv_busaddr;
|
|
|
|
/* Allocate DMA'able memory and load the DMA map for Rx ring. */
|
|
error = bus_dmamem_alloc(sc->fv_cdata.fv_rx_ring_tag,
|
|
(void **)&sc->fv_rdata.fv_rx_ring, BUS_DMA_WAITOK |
|
|
BUS_DMA_COHERENT | BUS_DMA_ZERO, &sc->fv_cdata.fv_rx_ring_map);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to allocate DMA'able memory for Rx ring\n");
|
|
goto fail;
|
|
}
|
|
|
|
ctx.fv_busaddr = 0;
|
|
error = bus_dmamap_load(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map, sc->fv_rdata.fv_rx_ring,
|
|
FV_RX_RING_SIZE, fv_dmamap_cb, &ctx, 0);
|
|
if (error != 0 || ctx.fv_busaddr == 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to load DMA'able memory for Rx ring\n");
|
|
goto fail;
|
|
}
|
|
sc->fv_rdata.fv_rx_ring_paddr = ctx.fv_busaddr;
|
|
|
|
/* Allocate DMA'able memory and load the DMA map for setup frame. */
|
|
error = bus_dmamem_alloc(sc->fv_cdata.fv_sf_tag,
|
|
(void **)&sc->fv_cdata.fv_sf_buff, BUS_DMA_WAITOK |
|
|
BUS_DMA_COHERENT | BUS_DMA_ZERO, &sc->fv_cdata.fv_sf_buff_map);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to allocate DMA'able memory for setup frame\n");
|
|
goto fail;
|
|
}
|
|
|
|
ctx.fv_busaddr = 0;
|
|
error = bus_dmamap_load(sc->fv_cdata.fv_sf_tag,
|
|
sc->fv_cdata.fv_sf_buff_map, sc->fv_cdata.fv_sf_buff,
|
|
FV_SFRAME_LEN, fv_dmamap_cb, &ctx, 0);
|
|
if (error != 0 || ctx.fv_busaddr == 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to load DMA'able memory for setup frame\n");
|
|
goto fail;
|
|
}
|
|
sc->fv_rdata.fv_sf_paddr = ctx.fv_busaddr;
|
|
|
|
/* Create DMA maps for Tx buffers. */
|
|
for (i = 0; i < FV_TX_RING_CNT; i++) {
|
|
txd = &sc->fv_cdata.fv_txdesc[i];
|
|
txd->tx_m = NULL;
|
|
txd->tx_dmamap = NULL;
|
|
error = bus_dmamap_create(sc->fv_cdata.fv_tx_tag, 0,
|
|
&txd->tx_dmamap);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to create Tx dmamap\n");
|
|
goto fail;
|
|
}
|
|
}
|
|
/* Create DMA maps for Rx buffers. */
|
|
if ((error = bus_dmamap_create(sc->fv_cdata.fv_rx_tag, 0,
|
|
&sc->fv_cdata.fv_rx_sparemap)) != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to create spare Rx dmamap\n");
|
|
goto fail;
|
|
}
|
|
for (i = 0; i < FV_RX_RING_CNT; i++) {
|
|
rxd = &sc->fv_cdata.fv_rxdesc[i];
|
|
rxd->rx_m = NULL;
|
|
rxd->rx_dmamap = NULL;
|
|
error = bus_dmamap_create(sc->fv_cdata.fv_rx_tag, 0,
|
|
&rxd->rx_dmamap);
|
|
if (error != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"failed to create Rx dmamap\n");
|
|
goto fail;
|
|
}
|
|
}
|
|
|
|
fail:
|
|
return (error);
|
|
}
|
|
|
|
static void
|
|
fv_dma_free(struct fv_softc *sc)
|
|
{
|
|
struct fv_txdesc *txd;
|
|
struct fv_rxdesc *rxd;
|
|
int i;
|
|
|
|
/* Tx ring. */
|
|
if (sc->fv_cdata.fv_tx_ring_tag) {
|
|
if (sc->fv_rdata.fv_tx_ring_paddr)
|
|
bus_dmamap_unload(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map);
|
|
if (sc->fv_rdata.fv_tx_ring)
|
|
bus_dmamem_free(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_rdata.fv_tx_ring,
|
|
sc->fv_cdata.fv_tx_ring_map);
|
|
sc->fv_rdata.fv_tx_ring = NULL;
|
|
sc->fv_rdata.fv_tx_ring_paddr = 0;
|
|
bus_dma_tag_destroy(sc->fv_cdata.fv_tx_ring_tag);
|
|
sc->fv_cdata.fv_tx_ring_tag = NULL;
|
|
}
|
|
/* Rx ring. */
|
|
if (sc->fv_cdata.fv_rx_ring_tag) {
|
|
if (sc->fv_rdata.fv_rx_ring_paddr)
|
|
bus_dmamap_unload(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map);
|
|
if (sc->fv_rdata.fv_rx_ring)
|
|
bus_dmamem_free(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_rdata.fv_rx_ring,
|
|
sc->fv_cdata.fv_rx_ring_map);
|
|
sc->fv_rdata.fv_rx_ring = NULL;
|
|
sc->fv_rdata.fv_rx_ring_paddr = 0;
|
|
bus_dma_tag_destroy(sc->fv_cdata.fv_rx_ring_tag);
|
|
sc->fv_cdata.fv_rx_ring_tag = NULL;
|
|
}
|
|
/* Tx buffers. */
|
|
if (sc->fv_cdata.fv_tx_tag) {
|
|
for (i = 0; i < FV_TX_RING_CNT; i++) {
|
|
txd = &sc->fv_cdata.fv_txdesc[i];
|
|
if (txd->tx_dmamap) {
|
|
bus_dmamap_destroy(sc->fv_cdata.fv_tx_tag,
|
|
txd->tx_dmamap);
|
|
txd->tx_dmamap = NULL;
|
|
}
|
|
}
|
|
bus_dma_tag_destroy(sc->fv_cdata.fv_tx_tag);
|
|
sc->fv_cdata.fv_tx_tag = NULL;
|
|
}
|
|
/* Rx buffers. */
|
|
if (sc->fv_cdata.fv_rx_tag) {
|
|
for (i = 0; i < FV_RX_RING_CNT; i++) {
|
|
rxd = &sc->fv_cdata.fv_rxdesc[i];
|
|
if (rxd->rx_dmamap) {
|
|
bus_dmamap_destroy(sc->fv_cdata.fv_rx_tag,
|
|
rxd->rx_dmamap);
|
|
rxd->rx_dmamap = NULL;
|
|
}
|
|
}
|
|
if (sc->fv_cdata.fv_rx_sparemap) {
|
|
bus_dmamap_destroy(sc->fv_cdata.fv_rx_tag,
|
|
sc->fv_cdata.fv_rx_sparemap);
|
|
sc->fv_cdata.fv_rx_sparemap = 0;
|
|
}
|
|
bus_dma_tag_destroy(sc->fv_cdata.fv_rx_tag);
|
|
sc->fv_cdata.fv_rx_tag = NULL;
|
|
}
|
|
|
|
if (sc->fv_cdata.fv_parent_tag) {
|
|
bus_dma_tag_destroy(sc->fv_cdata.fv_parent_tag);
|
|
sc->fv_cdata.fv_parent_tag = NULL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Initialize the transmit descriptors.
|
|
*/
|
|
static int
|
|
fv_tx_ring_init(struct fv_softc *sc)
|
|
{
|
|
struct fv_ring_data *rd;
|
|
struct fv_txdesc *txd;
|
|
bus_addr_t addr;
|
|
int i;
|
|
|
|
sc->fv_cdata.fv_tx_prod = 0;
|
|
sc->fv_cdata.fv_tx_cons = 0;
|
|
sc->fv_cdata.fv_tx_cnt = 0;
|
|
sc->fv_cdata.fv_tx_pkts = 0;
|
|
|
|
rd = &sc->fv_rdata;
|
|
bzero(rd->fv_tx_ring, FV_TX_RING_SIZE);
|
|
for (i = 0; i < FV_TX_RING_CNT; i++) {
|
|
if (i == FV_TX_RING_CNT - 1)
|
|
addr = FV_TX_RING_ADDR(sc, 0);
|
|
else
|
|
addr = FV_TX_RING_ADDR(sc, i + 1);
|
|
rd->fv_tx_ring[i].fv_stat = 0;
|
|
rd->fv_tx_ring[i].fv_devcs = 0;
|
|
rd->fv_tx_ring[i].fv_addr = 0;
|
|
rd->fv_tx_ring[i].fv_link = addr;
|
|
txd = &sc->fv_cdata.fv_txdesc[i];
|
|
txd->tx_m = NULL;
|
|
}
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map,
|
|
BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Initialize the RX descriptors and allocate mbufs for them. Note that
|
|
* we arrange the descriptors in a closed ring, so that the last descriptor
|
|
* points back to the first.
|
|
*/
|
|
static int
|
|
fv_rx_ring_init(struct fv_softc *sc)
|
|
{
|
|
struct fv_ring_data *rd;
|
|
struct fv_rxdesc *rxd;
|
|
int i;
|
|
|
|
sc->fv_cdata.fv_rx_cons = 0;
|
|
|
|
rd = &sc->fv_rdata;
|
|
bzero(rd->fv_rx_ring, FV_RX_RING_SIZE);
|
|
for (i = 0; i < FV_RX_RING_CNT; i++) {
|
|
rxd = &sc->fv_cdata.fv_rxdesc[i];
|
|
rxd->rx_m = NULL;
|
|
rxd->desc = &rd->fv_rx_ring[i];
|
|
rd->fv_rx_ring[i].fv_stat = ADSTAT_OWN;
|
|
rd->fv_rx_ring[i].fv_devcs = 0;
|
|
if (i == FV_RX_RING_CNT - 1)
|
|
rd->fv_rx_ring[i].fv_devcs |= ADCTL_ER;
|
|
rd->fv_rx_ring[i].fv_addr = 0;
|
|
if (fv_newbuf(sc, i) != 0)
|
|
return (ENOBUFS);
|
|
}
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map,
|
|
BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Initialize an RX descriptor and attach an MBUF cluster.
|
|
*/
|
|
static int
|
|
fv_newbuf(struct fv_softc *sc, int idx)
|
|
{
|
|
struct fv_desc *desc;
|
|
struct fv_rxdesc *rxd;
|
|
struct mbuf *m;
|
|
bus_dma_segment_t segs[1];
|
|
bus_dmamap_t map;
|
|
int nsegs;
|
|
|
|
m = m_getcl(M_NOWAIT, MT_DATA, M_PKTHDR);
|
|
if (m == NULL)
|
|
return (ENOBUFS);
|
|
m->m_len = m->m_pkthdr.len = MCLBYTES;
|
|
|
|
/* tcp header boundary alignment margin */
|
|
m_adj(m, 4);
|
|
|
|
if (bus_dmamap_load_mbuf_sg(sc->fv_cdata.fv_rx_tag,
|
|
sc->fv_cdata.fv_rx_sparemap, m, segs, &nsegs, 0) != 0) {
|
|
m_freem(m);
|
|
return (ENOBUFS);
|
|
}
|
|
KASSERT(nsegs == 1, ("%s: %d segments returned!", __func__, nsegs));
|
|
|
|
rxd = &sc->fv_cdata.fv_rxdesc[idx];
|
|
if (rxd->rx_m != NULL) {
|
|
/* This code make bug. Make scranble on buffer data.
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_tag, rxd->rx_dmamap,
|
|
BUS_DMASYNC_POSTREAD);
|
|
*/
|
|
bus_dmamap_unload(sc->fv_cdata.fv_rx_tag, rxd->rx_dmamap);
|
|
}
|
|
map = rxd->rx_dmamap;
|
|
rxd->rx_dmamap = sc->fv_cdata.fv_rx_sparemap;
|
|
sc->fv_cdata.fv_rx_sparemap = map;
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_tag, rxd->rx_dmamap,
|
|
BUS_DMASYNC_PREREAD);
|
|
rxd->rx_m = m;
|
|
desc = rxd->desc;
|
|
desc->fv_addr = segs[0].ds_addr;
|
|
desc->fv_devcs |= FV_DMASIZE(segs[0].ds_len);
|
|
rxd->saved_ca = desc->fv_addr ;
|
|
rxd->saved_ctl = desc->fv_stat ;
|
|
|
|
return (0);
|
|
}
|
|
|
|
static __inline void
|
|
fv_fixup_rx(struct mbuf *m)
|
|
{
|
|
int i;
|
|
uint16_t *src, *dst;
|
|
|
|
src = mtod(m, uint16_t *);
|
|
dst = src - 1;
|
|
|
|
for (i = 0; i < m->m_len / sizeof(uint16_t); i++) {
|
|
*dst++ = *src++;
|
|
}
|
|
|
|
if (m->m_len % sizeof(uint16_t))
|
|
*(uint8_t *)dst = *(uint8_t *)src;
|
|
|
|
m->m_data -= ETHER_ALIGN;
|
|
}
|
|
|
|
|
|
static void
|
|
fv_tx(struct fv_softc *sc)
|
|
{
|
|
struct fv_txdesc *txd;
|
|
struct fv_desc *cur_tx;
|
|
struct ifnet *ifp;
|
|
uint32_t ctl, devcs;
|
|
int cons, prod, prev_cons;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
cons = sc->fv_cdata.fv_tx_cons;
|
|
prod = sc->fv_cdata.fv_tx_prod;
|
|
if (cons == prod)
|
|
return;
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map,
|
|
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
|
|
|
|
ifp = sc->fv_ifp;
|
|
/*
|
|
* Go through our tx list and free mbufs for those
|
|
* frames that have been transmitted.
|
|
*/
|
|
prev_cons = cons;
|
|
for (; cons != prod; FV_INC(cons, FV_TX_RING_CNT)) {
|
|
cur_tx = &sc->fv_rdata.fv_tx_ring[cons];
|
|
ctl = cur_tx->fv_stat;
|
|
devcs = cur_tx->fv_devcs;
|
|
/* Check if descriptor has "finished" flag */
|
|
if (FV_DMASIZE(devcs) == 0)
|
|
break;
|
|
|
|
sc->fv_cdata.fv_tx_cnt--;
|
|
ifp->if_drv_flags &= ~IFF_DRV_OACTIVE;
|
|
|
|
txd = &sc->fv_cdata.fv_txdesc[cons];
|
|
|
|
if ((ctl & ADSTAT_Tx_ES) == 0)
|
|
if_inc_counter(ifp, IFCOUNTER_OPACKETS, 1);
|
|
else if (ctl & ADSTAT_Tx_UF) { /* only underflow not check collision */
|
|
if_inc_counter(ifp, IFCOUNTER_OERRORS, 1);
|
|
}
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_tag, txd->tx_dmamap,
|
|
BUS_DMASYNC_POSTWRITE);
|
|
bus_dmamap_unload(sc->fv_cdata.fv_tx_tag, txd->tx_dmamap);
|
|
|
|
/* Free only if it's first descriptor in list */
|
|
if (txd->tx_m)
|
|
m_freem(txd->tx_m);
|
|
txd->tx_m = NULL;
|
|
|
|
/* reset descriptor */
|
|
cur_tx->fv_stat = 0;
|
|
cur_tx->fv_devcs = 0;
|
|
cur_tx->fv_addr = 0;
|
|
}
|
|
|
|
sc->fv_cdata.fv_tx_cons = cons;
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_tx_ring_tag,
|
|
sc->fv_cdata.fv_tx_ring_map, BUS_DMASYNC_PREWRITE);
|
|
}
|
|
|
|
|
|
static void
|
|
fv_rx(struct fv_softc *sc)
|
|
{
|
|
struct fv_rxdesc *rxd;
|
|
struct ifnet *ifp = sc->fv_ifp;
|
|
int cons, prog, packet_len, error;
|
|
struct fv_desc *cur_rx;
|
|
struct mbuf *m;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
cons = sc->fv_cdata.fv_rx_cons;
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map,
|
|
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
|
|
|
|
for (prog = 0; prog < FV_RX_RING_CNT; FV_INC(cons, FV_RX_RING_CNT)) {
|
|
cur_rx = &sc->fv_rdata.fv_rx_ring[cons];
|
|
rxd = &sc->fv_cdata.fv_rxdesc[cons];
|
|
m = rxd->rx_m;
|
|
|
|
if ((cur_rx->fv_stat & ADSTAT_OWN) == ADSTAT_OWN)
|
|
break;
|
|
|
|
prog++;
|
|
|
|
if (cur_rx->fv_stat & (ADSTAT_ES | ADSTAT_Rx_TL)) {
|
|
device_printf(sc->fv_dev,
|
|
"Receive Descriptor error %x\n", cur_rx->fv_stat);
|
|
if_inc_counter(ifp, IFCOUNTER_IERRORS, 1);
|
|
packet_len = 0;
|
|
} else {
|
|
packet_len = ADSTAT_Rx_LENGTH(cur_rx->fv_stat);
|
|
}
|
|
|
|
/* Assume it's error */
|
|
error = 1;
|
|
|
|
if (packet_len < 64)
|
|
if_inc_counter(ifp, IFCOUNTER_IERRORS, 1);
|
|
else if ((cur_rx->fv_stat & ADSTAT_Rx_DE) == 0) {
|
|
error = 0;
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_tag, rxd->rx_dmamap,
|
|
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
|
|
m = rxd->rx_m;
|
|
/* Skip 4 bytes of CRC */
|
|
m->m_pkthdr.len = m->m_len = packet_len - ETHER_CRC_LEN;
|
|
|
|
fv_fixup_rx(m);
|
|
m->m_pkthdr.rcvif = ifp;
|
|
if_inc_counter(ifp, IFCOUNTER_IPACKETS, 1);
|
|
|
|
FV_UNLOCK(sc);
|
|
(*ifp->if_input)(ifp, m);
|
|
FV_LOCK(sc);
|
|
}
|
|
|
|
if (error) {
|
|
/* Restore CONTROL and CA values, reset DEVCS */
|
|
cur_rx->fv_stat = rxd->saved_ctl;
|
|
cur_rx->fv_addr = rxd->saved_ca;
|
|
cur_rx->fv_devcs = 0;
|
|
}
|
|
else {
|
|
/* Reinit descriptor */
|
|
cur_rx->fv_stat = ADSTAT_OWN;
|
|
cur_rx->fv_devcs = 0;
|
|
if (cons == FV_RX_RING_CNT - 1)
|
|
cur_rx->fv_devcs |= ADCTL_ER;
|
|
cur_rx->fv_addr = 0;
|
|
if (fv_newbuf(sc, cons) != 0) {
|
|
device_printf(sc->fv_dev,
|
|
"Failed to allocate buffer\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map,
|
|
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
|
|
|
|
}
|
|
|
|
if (prog > 0) {
|
|
sc->fv_cdata.fv_rx_cons = cons;
|
|
|
|
bus_dmamap_sync(sc->fv_cdata.fv_rx_ring_tag,
|
|
sc->fv_cdata.fv_rx_ring_map,
|
|
BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fv_intr(void *arg)
|
|
{
|
|
struct fv_softc *sc = arg;
|
|
uint32_t status;
|
|
struct ifnet *ifp = sc->fv_ifp;
|
|
|
|
FV_LOCK(sc);
|
|
|
|
status = CSR_READ_4(sc, CSR_STATUS);
|
|
/* mask out interrupts */
|
|
while((status & sc->sc_inten) != 0) {
|
|
if (status) {
|
|
CSR_WRITE_4(sc, CSR_STATUS, status);
|
|
}
|
|
if (status & STATUS_UNF) {
|
|
device_printf(sc->fv_dev, "Transmit Underflow\n");
|
|
}
|
|
if (status & sc->sc_rxint_mask) {
|
|
fv_rx(sc);
|
|
}
|
|
if (status & sc->sc_txint_mask) {
|
|
fv_tx(sc);
|
|
}
|
|
if (status & STATUS_AIS) {
|
|
device_printf(sc->fv_dev, "Abnormal Interrupt %x\n",
|
|
status);
|
|
}
|
|
CSR_WRITE_4(sc, CSR_FULLDUP, FULLDUP_CS |
|
|
(1 << FULLDUP_TT_SHIFT) | (3 << FULLDUP_NTP_SHIFT) |
|
|
(2 << FULLDUP_RT_SHIFT) | (2 << FULLDUP_NRP_SHIFT));
|
|
|
|
|
|
status = CSR_READ_4(sc, CSR_STATUS);
|
|
}
|
|
|
|
/* Try to get more packets going. */
|
|
if (!IFQ_DRV_IS_EMPTY(&ifp->if_snd))
|
|
fv_start_locked(ifp);
|
|
|
|
FV_UNLOCK(sc);
|
|
}
|
|
|
|
static void
|
|
fv_tick(void *xsc)
|
|
{
|
|
struct fv_softc *sc = xsc;
|
|
#ifdef MII
|
|
struct mii_data *mii;
|
|
|
|
FV_LOCK_ASSERT(sc);
|
|
|
|
mii = device_get_softc(sc->fv_miibus);
|
|
mii_tick(mii);
|
|
#endif
|
|
callout_reset(&sc->fv_stat_callout, hz, fv_tick, sc);
|
|
}
|
|
|
|
static void
|
|
fv_hinted_child(device_t bus, const char *dname, int dunit)
|
|
{
|
|
BUS_ADD_CHILD(bus, 0, dname, dunit);
|
|
device_printf(bus, "hinted child %s%d\n", dname, dunit);
|
|
}
|
|
|
|
#ifdef FV_MDIO
|
|
static int
|
|
fvmdio_probe(device_t dev)
|
|
{
|
|
if (!ofw_bus_status_okay(dev))
|
|
return (ENXIO);
|
|
|
|
if (!ofw_bus_is_compatible(dev, "fv,mdio"))
|
|
return (ENXIO);
|
|
|
|
device_set_desc(dev, "FV built-in ethernet interface, MDIO controller");
|
|
return(0);
|
|
}
|
|
|
|
static int
|
|
fvmdio_attach(device_t dev)
|
|
{
|
|
struct fv_softc *sc;
|
|
int error;
|
|
|
|
sc = device_get_softc(dev);
|
|
sc->fv_dev = dev;
|
|
sc->fv_rid = 0;
|
|
sc->fv_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
|
|
&sc->fv_rid, RF_ACTIVE | RF_SHAREABLE);
|
|
if (sc->fv_res == NULL) {
|
|
device_printf(dev, "couldn't map memory\n");
|
|
error = ENXIO;
|
|
goto fail;
|
|
}
|
|
|
|
sc->fv_btag = rman_get_bustag(sc->fv_res);
|
|
sc->fv_bhandle = rman_get_bushandle(sc->fv_res);
|
|
|
|
bus_generic_probe(dev);
|
|
bus_enumerate_hinted_children(dev);
|
|
error = bus_generic_attach(dev);
|
|
fail:
|
|
return(error);
|
|
}
|
|
|
|
static int
|
|
fvmdio_detach(device_t dev)
|
|
{
|
|
return(0);
|
|
}
|
|
#endif
|
|
|
|
#ifdef FV_DEBUG
|
|
void
|
|
dump_txdesc(struct fv_softc *sc, int pos)
|
|
{
|
|
struct fv_desc *desc;
|
|
|
|
desc = &sc->fv_rdata.fv_tx_ring[pos];
|
|
device_printf(sc->fv_dev, "CSR_TXLIST %08x\n", CSR_READ_4(sc, CSR_TXLIST));
|
|
device_printf(sc->fv_dev, "%d TDES0:%08x TDES1:%08x TDES2:%08x TDES3:%08x\n",
|
|
pos, desc->fv_stat, desc->fv_devcs, desc->fv_addr, desc->fv_link);
|
|
}
|
|
|
|
void
|
|
dump_status_reg(struct fv_softc *sc)
|
|
{
|
|
uint32_t status;
|
|
|
|
/* mask out interrupts */
|
|
|
|
status = CSR_READ_4(sc, CSR_STATUS);
|
|
device_printf(sc->fv_dev, "CSR5 Status Register EB:%d TS:%d RS:%d NIS:%d AIS:%d ER:%d SE:%d LNF:%d TM:%d RWT:%d RPS:%d RU:%d RI:%d UNF:%d LNP/ANC:%d TJT:%d TU:%d TPS:%d TI:%d\n",
|
|
(status >> 23 ) & 7,
|
|
(status >> 20 ) & 7,
|
|
(status >> 17 ) & 7,
|
|
(status >> 16 ) & 1,
|
|
(status >> 15 ) & 1,
|
|
(status >> 14 ) & 1,
|
|
(status >> 13 ) & 1,
|
|
(status >> 12 ) & 1,
|
|
(status >> 11 ) & 1,
|
|
(status >> 9 ) & 1,
|
|
(status >> 8 ) & 1,
|
|
(status >> 7 ) & 1,
|
|
(status >> 6 ) & 1,
|
|
(status >> 5 ) & 1,
|
|
(status >> 4 ) & 1,
|
|
(status >> 3 ) & 1,
|
|
(status >> 2 ) & 1,
|
|
(status >> 1 ) & 1,
|
|
(status >> 0 ) & 1);
|
|
|
|
}
|
|
#endif
|