Module Name: src Committed By: jmcneill Date: Sun Jan 4 03:53:02 UTC 2015
Modified Files: src/sys/arch/arm/rockchip: files.rockchip Added Files: src/sys/arch/arm/rockchip: rockchip_emac.c rockchip_emacreg.h Log Message: Add Rockchip ethernet driver, untested. To generate a diff of this commit: cvs rdiff -u -r1.8 -r1.9 src/sys/arch/arm/rockchip/files.rockchip cvs rdiff -u -r0 -r1.1 src/sys/arch/arm/rockchip/rockchip_emac.c \ src/sys/arch/arm/rockchip/rockchip_emacreg.h Please note that diffs are not public domain; they are subject to the copyright notices on the relevant files.
Modified files: Index: src/sys/arch/arm/rockchip/files.rockchip diff -u src/sys/arch/arm/rockchip/files.rockchip:1.8 src/sys/arch/arm/rockchip/files.rockchip:1.9 --- src/sys/arch/arm/rockchip/files.rockchip:1.8 Sat Jan 3 13:26:31 2015 +++ src/sys/arch/arm/rockchip/files.rockchip Sun Jan 4 03:53:02 2015 @@ -1,4 +1,4 @@ -# $NetBSD: files.rockchip,v 1.8 2015/01/03 13:26:31 jmcneill Exp $ +# $NetBSD: files.rockchip,v 1.9 2015/01/04 03:53:02 jmcneill Exp $ # # Configuration info for Rockchip ARM Peripherals # @@ -45,6 +45,11 @@ file arch/arm/rockchip/rockchip_dwcmmc.c attach dwctwo at obio with rkdwctwo file arch/arm/rockchip/rockchip_dwctwo.c rkdwctwo needs-flag +# VMAC Ethernet Controller +device rkemac: arp, ether, ifnet, mii +attach rkemac at obio +file arch/arm/rockchip/rockchip_emac.c rkemac + # Console parameters defparam opt_rockchip.h CONADDR defparam opt_rockchip.h CONSPEED @@ -56,3 +61,4 @@ defparam opt_rockchip.h MEMSIZE # Debugging parameters defflag opt_rockchip.h ROCKCHIP_CLOCK_DEBUG defflag opt_rkiic.h RKIIC_DEBUG +defflag opt_rkemac.h RKEMAC_DEBUG Added files: Index: src/sys/arch/arm/rockchip/rockchip_emac.c diff -u /dev/null src/sys/arch/arm/rockchip/rockchip_emac.c:1.1 --- /dev/null Sun Jan 4 03:53:02 2015 +++ src/sys/arch/arm/rockchip/rockchip_emac.c Sun Jan 4 03:53:02 2015 @@ -0,0 +1,849 @@ +/* $NetBSD: rockchip_emac.c,v 1.1 2015/01/04 03:53:02 jmcneill Exp $ */ + +/*- + * Copyright (c) 2015 Jared D. McNeill <jmcne...@invisible.ca> + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#include "opt_rkemac.h" + +#include <sys/cdefs.h> +__KERNEL_RCSID(0, "$NetBSD: rockchip_emac.c,v 1.1 2015/01/04 03:53:02 jmcneill Exp $"); + +#include <sys/param.h> +#include <sys/bus.h> +#include <sys/device.h> +#include <sys/intr.h> +#include <sys/systm.h> +#include <sys/kernel.h> +#include <sys/mutex.h> +#include <sys/condvar.h> +#include <sys/cprng.h> + +#include <arm/rockchip/rockchip_reg.h> +#include <arm/rockchip/rockchip_var.h> + +#include <net/if.h> +#include <net/if_ether.h> +#include <net/if_media.h> +#include <net/bpf.h> + +#include <dev/mii/miivar.h> + +#include <arm/rockchip/rockchip_emacreg.h> + +#define RKEMAC_ENABLE_INTR \ + (EMAC_ENABLE_TXINT|EMAC_ENABLE_RXINT|EMAC_ENABLE_ERR) + +#define RKEMAC_RX_RING_COUNT 128 +#define RKEMAC_TX_RING_COUNT 128 + +#define RKEMAC_MAX_PACKET 1536 +#define RKEMAC_POLLRATE 200 + +#define RX_DESC_OFFSET(n) \ + ((n) * sizeof(struct rkemac_rxdesc)) +#define RX_NEXT(n) (((n) + 1) & (RKEMAC_RX_RING_COUNT - 1)) + +#define TX_DESC_OFFSET(n) \ + ((RKEMAC_RX_RING_COUNT+(n)) * sizeof(struct rkemac_txdesc)) +#define TX_NEXT(n) (((n) + 1) & (RKEMAC_TX_RING_COUNT - 1)) + +struct rkemac_rxdata { + bus_dmamap_t rd_map; + struct mbuf *rd_m; +}; + +struct rkemac_txdata { + bus_dmamap_t td_map; + bus_dmamap_t td_active; + struct mbuf *td_m; +}; + +struct rkemac_txring { + bus_addr_t t_physaddr; + struct rkemac_txdesc *t_desc; + struct rkemac_txdata t_data[RKEMAC_TX_RING_COUNT]; + int t_cur, t_next, t_queued; +}; + +struct rkemac_rxring { + bus_addr_t r_physaddr; + struct rkemac_rxdesc *r_desc; + struct rkemac_rxdata r_data[RKEMAC_RX_RING_COUNT]; + int r_cur, r_next; +}; + +struct rkemac_softc { + device_t sc_dev; + bus_space_tag_t sc_bst; + bus_space_handle_t sc_bsh; + bus_dma_tag_t sc_dmat; + void *sc_ih; + + struct ethercom sc_ec; + struct mii_data sc_mii; + callout_t sc_mii_tick; + kmutex_t sc_lock; + + bus_dmamap_t sc_ring_dmamap; + bus_dma_segment_t sc_ring_dmaseg; + struct rkemac_txring sc_txq; + struct rkemac_rxring sc_rxq; +}; + +static int rkemac_match(device_t, cfdata_t, void *); +static void rkemac_attach(device_t, device_t, void *); + +static int rkemac_dma_init(struct rkemac_softc *); + +static int rkemac_intr(void *); + +static int rkemac_mii_readreg(device_t, int, int); +static void rkemac_mii_writereg(device_t, int, int, int); +static void rkemac_mii_statchg(struct ifnet *); + +static int rkemac_init(struct ifnet *); +static void rkemac_start(struct ifnet *); +static void rkemac_stop(struct ifnet *, int); +static int rkemac_ioctl(struct ifnet *, u_long, void *); +static void rkemac_watchdog(struct ifnet *); +static void rkemac_tick(void *); + +static int rkemac_queue(struct rkemac_softc *, struct mbuf *); +static void rkemac_txdesc_sync(struct rkemac_softc *, int, int, int); +static void rkemac_txintr(struct rkemac_softc *); +static void rkemac_rxintr(struct rkemac_softc *); + +CFATTACH_DECL_NEW(rkemac, sizeof(struct rkemac_softc), + rkemac_match, rkemac_attach, NULL, NULL); + +#define EMAC_WRITE(sc, reg, val) \ + bus_space_write_4((sc)->sc_bst, (sc)->sc_bsh, (reg), (val)) +#define EMAC_READ(sc, reg) \ + bus_space_read_4((sc)->sc_bst, (sc)->sc_bsh, (reg)) + +static int +rkemac_match(device_t parent, cfdata_t cf, void *aux) +{ + return 1; +} + +static void +rkemac_attach(device_t parent, device_t self, void *aux) +{ + struct rkemac_softc *sc = device_private(self); + struct obio_attach_args * const obio = aux; + prop_dictionary_t cfg = device_properties(self); + struct mii_data *mii = &sc->sc_mii; + struct ifnet *ifp = &sc->sc_ec.ec_if; + uint8_t enaddr[ETHER_ADDR_LEN]; + prop_data_t ea; + + sc->sc_dev = self; + sc->sc_bst = obio->obio_bst; + sc->sc_dmat = obio->obio_dmat; + bus_space_subregion(obio->obio_bst, obio->obio_bsh, obio->obio_offset, + obio->obio_size, &sc->sc_bsh); + mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_NET); + + callout_init(&sc->sc_mii_tick, 0); + callout_setfunc(&sc->sc_mii_tick, rkemac_tick, sc); + + aprint_naive("\n"); + aprint_normal(": Ethernet controller\n"); + +#ifdef RKEMAC_DEBUG + aprint_normal_dev(sc->sc_dev, "ID %#x\n", + EMAC_READ(sc, EMAC_ID_REG)); +#endif + + sc->sc_ih = intr_establish(obio->obio_intr, IPL_NET, + IST_LEVEL, rkemac_intr, sc); + if (sc->sc_ih == NULL) { + aprint_error_dev(self, "couldn't establish interrupt\n"); + return; + } + + if ((ea = prop_dictionary_get(cfg, "mac-address")) != NULL) { + KASSERT(prop_object_type(ea) == PROP_TYPE_DATA); + KASSERT(prop_data_size(ea) == ETHER_ADDR_LEN); + memcpy(enaddr, prop_data_data_nocopy(ea), ETHER_ADDR_LEN); + } else { + uint32_t addrl, addrh; + + addrl = EMAC_READ(sc, EMAC_ADDRL_REG); + addrh = EMAC_READ(sc, EMAC_ADDRH_REG); + + if (addrl == 0 && addrh == 0) { + /* fake MAC address */ + addrl = 0x00f2 | (cprng_strong32() << 16); + addrh = cprng_strong32(); + } + + enaddr[0] = addrl & 0xff; + enaddr[1] = (addrl >> 8) & 0xff; + enaddr[2] = (addrl >> 16) & 0xff; + enaddr[3] = (addrl >> 24) & 0xff; + enaddr[4] = addrh & 0xff; + enaddr[5] = (addrh >> 8) & 0xff; + } + + aprint_normal_dev(sc->sc_dev, "Ethernet address: %s\n", + ether_sprintf(enaddr)); + + EMAC_WRITE(sc, EMAC_CONTROL_REG, 0); + EMAC_WRITE(sc, EMAC_ENABLE_REG, 0); + EMAC_WRITE(sc, EMAC_STAT_REG, EMAC_READ(sc, EMAC_STAT_REG)); + + rkemac_dma_init(sc); + + ifp->if_softc = sc; + strlcpy(ifp->if_xname, device_xname(sc->sc_dev), IFNAMSIZ); + ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; + ifp->if_ioctl = rkemac_ioctl; + ifp->if_start = rkemac_start; + ifp->if_init = rkemac_init; + ifp->if_stop = rkemac_stop; + ifp->if_watchdog = rkemac_watchdog; + IFQ_SET_MAXLEN(&ifp->if_snd, IFQ_MAXLEN); + IFQ_SET_READY(&ifp->if_snd); + + sc->sc_ec.ec_mii = mii; + ifmedia_init(&mii->mii_media, 0, ether_mediachange, ether_mediastatus); + mii->mii_ifp = ifp; + mii->mii_readreg = rkemac_mii_readreg; + mii->mii_writereg = rkemac_mii_writereg; + mii->mii_statchg = rkemac_mii_statchg; + mii_attach(sc->sc_dev, mii, 0xffffffff, MII_PHY_ANY, MII_OFFSET_ANY, 0); + + if (LIST_EMPTY(&mii->mii_phys)) { + aprint_error_dev(sc->sc_dev, "no PHY found!\n"); + return; + } + + ifmedia_set(&mii->mii_media, IFM_ETHER|IFM_AUTO); + + if_attach(ifp); + ether_ifattach(ifp, enaddr); + + EMAC_WRITE(sc, EMAC_ENABLE_REG, RKEMAC_ENABLE_INTR); +} + +static int +rkemac_dma_init(struct rkemac_softc *sc) +{ + size_t descsize = RKEMAC_RX_RING_COUNT * sizeof(struct rkemac_rxdesc) + + RKEMAC_TX_RING_COUNT * sizeof(struct rkemac_txdesc); + bus_addr_t physaddr; + int error, nsegs; + void *descs; + + /* + * Allocate TX / RX descriptors + */ + error = bus_dmamap_create(sc->sc_dmat, descsize, 1, descsize, 0, + BUS_DMA_NOWAIT, &sc->sc_ring_dmamap); + if (error) + return error; + error = bus_dmamem_alloc(sc->sc_dmat, descsize, 8, 0, + &sc->sc_ring_dmaseg, 1, &nsegs, BUS_DMA_NOWAIT); + if (error) + return error; + error = bus_dmamem_map(sc->sc_dmat, &sc->sc_ring_dmaseg, nsegs, + descsize, &descs, BUS_DMA_NOWAIT | BUS_DMA_COHERENT); + if (error) + return error; + error = bus_dmamap_load(sc->sc_dmat, sc->sc_ring_dmamap, descs, + descsize, NULL, BUS_DMA_NOWAIT); + if (error) + return error; + + memset(descs, 0, descsize); + + sc->sc_rxq.r_desc = descs; + sc->sc_rxq.r_physaddr = sc->sc_ring_dmamap->dm_segs[0].ds_addr; + + sc->sc_txq.t_desc = + (struct rkemac_txdesc *)(sc->sc_rxq.r_desc + RKEMAC_RX_RING_COUNT); + sc->sc_txq.t_physaddr = sc->sc_rxq.r_physaddr + + RKEMAC_RX_RING_COUNT * sizeof(struct rkemac_rxdesc); + + /* + * Setup RX ring + */ + sc->sc_rxq.r_cur = sc->sc_rxq.r_next = 0; + for (int i = 0; i < RKEMAC_RX_RING_COUNT; i++) { + struct rkemac_rxdata *rd = &sc->sc_rxq.r_data[i]; + struct rkemac_rxdesc *rx = &sc->sc_rxq.r_desc[i]; + + MGETHDR(rd->rd_m, M_DONTWAIT, MT_DATA); + if (rd->rd_m == NULL) + return ENOMEM; + error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1, + MCLBYTES, 0, BUS_DMA_NOWAIT, &rd->rd_map); + if (error) + return error; + MCLGET(rd->rd_m, M_DONTWAIT); + if ((rd->rd_m->m_flags & M_EXT) == 0) + return ENOMEM; + error = bus_dmamap_load(sc->sc_dmat, rd->rd_map, + mtod(rd->rd_m, void *), MCLBYTES, NULL, + BUS_DMA_READ | BUS_DMA_NOWAIT); + if (error) + return error; + physaddr = rd->rd_map->dm_segs[0].ds_addr; + + rx->rx_ptr = htole32(physaddr); + rx->rx_info = htole32(EMAC_RXDESC_OWN | + __SHIFTIN(RKEMAC_MAX_PACKET, EMAC_RXDESC_RXLEN)); + } + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + 0, RKEMAC_RX_RING_COUNT * sizeof(struct rkemac_rxdesc), + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + + /* + * Setup TX ring + */ + sc->sc_txq.t_queued = sc->sc_txq.t_cur = sc->sc_txq.t_next = 0; + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + RKEMAC_RX_RING_COUNT, + RKEMAC_TX_RING_COUNT * sizeof(struct rkemac_txdesc), + BUS_DMASYNC_POSTWRITE); + for (int i = 0; i < RKEMAC_TX_RING_COUNT; i++) { + struct rkemac_txdata *td = &sc->sc_txq.t_data[i]; + struct rkemac_txdesc *tx = &sc->sc_txq.t_desc[i]; + error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, + RKEMAC_TX_RING_COUNT, MCLBYTES, 0, BUS_DMA_NOWAIT, + &td->td_map); + if (error) + return error; + tx->tx_ptr = htole32(sc->sc_txq.t_physaddr + + RKEMAC_RX_RING_COUNT + + sizeof(struct rkemac_txdesc) * i); + } + + return 0; +} + +static int +rkemac_intr(void *priv) +{ + struct rkemac_softc *sc = priv; + struct ifnet *ifp = &sc->sc_ec.ec_if; + uint32_t stat; + + stat = EMAC_READ(sc, EMAC_STAT_REG); + if (!stat) + return 0; + + EMAC_WRITE(sc, EMAC_STAT_REG, stat); + + if (stat & EMAC_STAT_TXINT) + rkemac_txintr(sc); + if (stat & EMAC_STAT_RXINT) + rkemac_rxintr(sc); + if (stat & EMAC_STAT_ERR) + ifp->if_oerrors++; + + if (stat & (EMAC_STAT_TXINT|EMAC_STAT_RXINT)) + rkemac_start(ifp); + + return 1; +} + +static int +rkemac_mii_readreg(device_t dev, int phy, int reg) +{ + struct rkemac_softc *sc = device_private(dev); + uint32_t mdio; + int retry = 1000; + + mdio = __SHIFTIN(EMAC_MDIO_DATA_SFD_SFD, EMAC_MDIO_DATA_SFD); + mdio |= __SHIFTIN(EMAC_MDIO_DATA_OP_READ, EMAC_MDIO_DATA_OP); + mdio |= __SHIFTIN(phy, EMAC_MDIO_DATA_PHYADDR); + mdio |= __SHIFTIN(reg, EMAC_MDIO_DATA_PHYREG); + mdio |= __SHIFTIN(EMAC_MDIO_DATA_TA_TA, EMAC_MDIO_DATA_TA); + + mutex_enter(&sc->sc_lock); + EMAC_WRITE(sc, EMAC_STAT_REG, EMAC_STAT_MDIO); + EMAC_WRITE(sc, EMAC_MDIO_DATA_REG, mdio); + while (--retry > 0) { + if (EMAC_READ(sc, EMAC_STAT_REG) & EMAC_STAT_MDIO) { + mdio = EMAC_READ(sc, EMAC_MDIO_DATA_REG); + break; + } + delay(10); + } + mutex_exit(&sc->sc_lock); + +#ifdef RKEMAC_DEBUG + printf("%s: phy %d reg %#x data %#x\n", __func__, phy, reg, mdio); + if (retry == 0) + printf("%s: timed out\n", __func__); +#endif + + return __SHIFTOUT(mdio, EMAC_MDIO_DATA_DATA); +} + +static void +rkemac_mii_writereg(device_t dev, int phy, int reg, int val) +{ + struct rkemac_softc *sc = device_private(dev); + uint32_t mdio; + int retry = 1000; + + mdio = __SHIFTIN(EMAC_MDIO_DATA_SFD_SFD, EMAC_MDIO_DATA_SFD); + mdio |= __SHIFTIN(EMAC_MDIO_DATA_OP_WRITE, EMAC_MDIO_DATA_OP); + mdio |= __SHIFTIN(phy, EMAC_MDIO_DATA_PHYADDR); + mdio |= __SHIFTIN(reg, EMAC_MDIO_DATA_PHYREG); + mdio |= __SHIFTIN(EMAC_MDIO_DATA_TA_TA, EMAC_MDIO_DATA_TA); + mdio |= __SHIFTIN(val, EMAC_MDIO_DATA_DATA); + +#ifdef RKEMAC_DEBUG + printf("%s: phy %d reg %#x data %#x\n", __func__, phy, reg, mdio); +#endif + + mutex_enter(&sc->sc_lock); + EMAC_WRITE(sc, EMAC_STAT_REG, EMAC_STAT_MDIO); + EMAC_WRITE(sc, EMAC_MDIO_DATA_REG, mdio); + while (--retry > 0) { + if (EMAC_READ(sc, EMAC_STAT_REG) & EMAC_STAT_MDIO) { + break; + } + delay(10); + } + mutex_exit(&sc->sc_lock); + +#ifdef RKEMAC_DEBUG + if (retry == 0) + printf("%s: timed out\n", __func__); +#endif +} + +static void +rkemac_mii_statchg(struct ifnet *ifp) +{ + struct rkemac_softc *sc = ifp->if_softc; + struct mii_data *mii = &sc->sc_mii; + uint32_t control; + + control = EMAC_READ(sc, EMAC_CONTROL_REG); + + switch (IFM_SUBTYPE(mii->mii_media_active)) { + case IFM_10_T: + break; + case IFM_100_TX: + break; + } + + if (mii->mii_media_active & IFM_FDX) { + control |= EMAC_CONTROL_ENFL; + } else { + control &= ~EMAC_CONTROL_ENFL; + } + + EMAC_WRITE(sc, EMAC_CONTROL_REG, control); +} + +static int +rkemac_init(struct ifnet *ifp) +{ + struct rkemac_softc *sc = ifp->if_softc; + struct ether_multi *em; + struct ether_multistep ems; + uint32_t control; + + if (ifp->if_flags & IFF_RUNNING) + return 0; + + rkemac_stop(ifp, 0); + + control = EMAC_READ(sc, EMAC_CONTROL_REG); + if (ifp->if_flags & IFF_PROMISC) { + control |= EMAC_CONTROL_PROM; + } else { + control &= ~EMAC_CONTROL_PROM; + } + if (ifp->if_flags & IFF_BROADCAST) { + control &= ~EMAC_CONTROL_DISBC; + } else { + control |= EMAC_CONTROL_DISBC; + } + ETHER_FIRST_MULTI(ems, &sc->sc_ec, em); + if (em) { + ifp->if_flags |= IFF_ALLMULTI; + control |= EMAC_CONTROL_PROM; + } else { + ifp->if_flags &= ~IFF_ALLMULTI; + } + + EMAC_WRITE(sc, EMAC_POLLRATE_REG, RKEMAC_POLLRATE); + EMAC_WRITE(sc, EMAC_RXRINGPTR_REG, sc->sc_rxq.r_physaddr); + EMAC_WRITE(sc, EMAC_TXRINGPTR_REG, sc->sc_txq.t_physaddr); + + control &= ~EMAC_CONTROL_RXBDTLEN; + control |= __SHIFTIN(RKEMAC_RX_RING_COUNT, EMAC_CONTROL_RXBDTLEN); + control &= ~EMAC_CONTROL_TXBDTLEN; + control |= __SHIFTIN(RKEMAC_TX_RING_COUNT, EMAC_CONTROL_TXBDTLEN); + control |= EMAC_CONTROL_RXRN; + control |= EMAC_CONTROL_TXRN; + control |= EMAC_CONTROL_EN; + EMAC_WRITE(sc, EMAC_CONTROL_REG, control); + + ifp->if_flags |= IFF_RUNNING; + ifp->if_flags &= ~IFF_OACTIVE; + + return 0; +} + +static void +rkemac_start(struct ifnet *ifp) +{ + struct rkemac_softc *sc = ifp->if_softc; + int old = sc->sc_txq.t_queued; + struct mbuf *m0; + + if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) + return; + + for (;;) { + IFQ_POLL(&ifp->if_snd, m0); + if (m0 == NULL) + break; + if (rkemac_queue(sc, m0) != 0) { + ifp->if_flags |= IFF_OACTIVE; + break; + } + IFQ_DEQUEUE(&ifp->if_snd, m0); + bpf_mtap(ifp, m0); + } + + if (sc->sc_txq.t_queued != old) { + rkemac_txdesc_sync(sc, old, sc->sc_txq.t_cur, + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + EMAC_WRITE(sc, EMAC_STAT_REG, + EMAC_READ(sc, EMAC_STAT_REG) | EMAC_STAT_TXPL); + } +} + +static void +rkemac_stop(struct ifnet *ifp, int disable) +{ + struct rkemac_softc *sc = ifp->if_softc; + uint32_t control; + + callout_halt(&sc->sc_mii_tick, NULL); + + control = EMAC_READ(sc, EMAC_CONTROL_REG); + control &= ~EMAC_CONTROL_EN; + EMAC_WRITE(sc, EMAC_CONTROL_REG, control); + + ifp->if_timer = 0; + ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); + + mii_down(&sc->sc_mii); + + /* reset TX/RX rings */ + for (int i = 0; i < RKEMAC_RX_RING_COUNT; i++) { + struct rkemac_rxdesc *rx = &sc->sc_rxq.r_desc[i]; + rx->rx_info = htole32(EMAC_RXDESC_OWN | + __SHIFTIN(RKEMAC_MAX_PACKET, EMAC_RXDESC_RXLEN)); + } + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + 0, RKEMAC_RX_RING_COUNT * sizeof(struct rkemac_rxdesc), + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + sc->sc_rxq.r_cur = sc->sc_rxq.r_next = 0; + for (int i = 0; i < RKEMAC_TX_RING_COUNT; i++) { + struct rkemac_txdata *td = &sc->sc_txq.t_data[i]; + if (td->td_m) { + bus_dmamap_sync(sc->sc_dmat, td->td_active, + 0, td->td_active->dm_mapsize, + BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(sc->sc_dmat, td->td_active); + m_freem(td->td_m); + td->td_m = NULL; + } + } + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + RKEMAC_RX_RING_COUNT, + RKEMAC_TX_RING_COUNT * sizeof(struct rkemac_txdesc), + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + sc->sc_txq.t_queued = sc->sc_txq.t_cur = sc->sc_txq.t_next = 0; +} + +static int +rkemac_ioctl(struct ifnet *ifp, u_long cmd, void *data) +{ + struct rkemac_softc *sc = ifp->if_softc; + struct ether_multi *em; + struct ether_multistep ems; + uint32_t control; + int s, error; + + s = splnet(); + + if ((error = ether_ioctl(ifp, cmd, data)) == ENETRESET) { + error = 0; + if (cmd != SIOCADDMULTI && cmd != SIOCDELMULTI) + ; + else if (ifp->if_flags & IFF_RUNNING) { + control = EMAC_READ(sc, EMAC_CONTROL_REG); + ETHER_FIRST_MULTI(ems, &sc->sc_ec, em); + if (em) { + ifp->if_flags |= IFF_ALLMULTI; + control |= EMAC_CONTROL_PROM; + } else { + ifp->if_flags &= ~IFF_ALLMULTI; + if ((ifp->if_flags & IFF_PROMISC) == 0) { + control &= ~EMAC_CONTROL_PROM; + } + } + EMAC_WRITE(sc, EMAC_CONTROL_REG, control); + } + } + + if (ifp->if_flags & IFF_UP) + rkemac_start(ifp); + + splx(s); + + return error; +} + +static void +rkemac_watchdog(struct ifnet *ifp) +{ + struct rkemac_softc *sc = ifp->if_softc; + + device_printf(sc->sc_dev, "watchdog timeout\n"); + ++ifp->if_oerrors; + rkemac_init(ifp); +} + +static void +rkemac_tick(void *priv) +{ + struct rkemac_softc *sc = priv; + int s; + + s = splnet(); + mii_tick(&sc->sc_mii); + splx(s); + + callout_schedule(&sc->sc_mii_tick, hz); +} + +static int +rkemac_queue(struct rkemac_softc *sc, struct mbuf *m0) +{ + struct rkemac_txdesc *tx; + struct rkemac_txdata *td = NULL; + bus_dmamap_t map; + uint32_t info, len; + int error, first; + + first = sc->sc_txq.t_cur; + map = sc->sc_txq.t_data[first].td_map; + info = 0; + + KASSERT(map->dm_nsegs > 0); + + error = bus_dmamap_load_mbuf(sc->sc_dmat, map, m0, + BUS_DMA_WRITE | BUS_DMA_NOWAIT); + if (error) { + device_printf(sc->sc_dev, "couldn't map mbuf: %d\n", error); + return error; + } + + if (sc->sc_txq.t_queued + map->dm_nsegs >= RKEMAC_TX_RING_COUNT - 1) { + bus_dmamap_unload(sc->sc_dmat, map); + return ENOBUFS; + } + + info = EMAC_TXDESC_FIRST; + for (int i = 0; i < map->dm_nsegs; i++) { + td = &sc->sc_txq.t_data[sc->sc_txq.t_cur]; + tx = &sc->sc_txq.t_desc[sc->sc_txq.t_cur]; + + tx->tx_ptr = htole32(map->dm_segs[i].ds_addr); + len = __SHIFTIN(map->dm_segs[i].ds_len, EMAC_TXDESC_TXLEN); + if (i == map->dm_nsegs - 1) + info |= EMAC_TXDESC_LAST; + tx->tx_info = htole32(info | len); + if (i > 0) + tx->tx_info |= EMAC_TXDESC_OWN; + + sc->sc_txq.t_queued++; + sc->sc_txq.t_cur = + (sc->sc_txq.t_cur + 1) % RKEMAC_TX_RING_COUNT; + } + + sc->sc_txq.t_desc[first].tx_info |= htole32(EMAC_TXDESC_OWN); + + td->td_m = m0; + td->td_active = map; + + bus_dmamap_sync(sc->sc_dmat, map, 0, map->dm_mapsize, + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + + return 0; +} + +static void +rkemac_txdesc_sync(struct rkemac_softc *sc, int start, int end, int ops) +{ + if (end > start) { + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + TX_DESC_OFFSET(start), + TX_DESC_OFFSET(end) - TX_DESC_OFFSET(start), ops); + return; + } + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + TX_DESC_OFFSET(start), + TX_DESC_OFFSET(RKEMAC_TX_RING_COUNT + 1) - TX_DESC_OFFSET(start), + ops); + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + TX_DESC_OFFSET(0), TX_DESC_OFFSET(end) - TX_DESC_OFFSET(0), ops); +} + +static void +rkemac_txintr(struct rkemac_softc *sc) +{ + struct ifnet *ifp = &sc->sc_ec.ec_if; + int i; + + for (i = sc->sc_txq.t_next; sc->sc_txq.t_queued > 0; + i = TX_NEXT(i), sc->sc_txq.t_queued--) { + struct rkemac_txdata *td = &sc->sc_txq.t_data[i]; + struct rkemac_txdesc *tx = &sc->sc_txq.t_desc[i]; + + rkemac_txdesc_sync(sc, i, i + 1, + BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD); + + const uint32_t info = le32toh(tx->tx_info); + if (info & EMAC_TXDESC_OWN) + break; + + if (td->td_m == NULL) + continue; + + ifp->if_opackets++; + bus_dmamap_sync(sc->sc_dmat, td->td_active, 0, + td->td_active->dm_mapsize, BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(sc->sc_dmat, td->td_active); + + m_freem(td->td_m); + td->td_m = NULL; + } + + sc->sc_txq.t_next = i; + + if (sc->sc_txq.t_queued < RKEMAC_TX_RING_COUNT) { + ifp->if_flags &= ~IFF_OACTIVE; + } +} + +static void +rkemac_rxintr(struct rkemac_softc *sc) +{ + struct ifnet *ifp = &sc->sc_ec.ec_if; + struct mbuf *m, *mnew; + int i, error; + + for (i = sc->sc_rxq.r_cur; ; i = RX_NEXT(i)) { + struct rkemac_rxdata *rd = &sc->sc_rxq.r_data[i]; + struct rkemac_rxdesc *rx = &sc->sc_rxq.r_desc[i]; + + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + RX_DESC_OFFSET(i), sizeof(struct rkemac_rxdesc), + BUS_DMASYNC_POSTWRITE | BUS_DMASYNC_POSTREAD); + + const uint32_t info = le32toh(rx->rx_info); + if (info & EMAC_RXDESC_OWN) + break; + + if (info & EMAC_RXDESC_BUFF) { + ifp->if_ierrors++; + goto skip; + } + + const u_int len = __SHIFTOUT(info, EMAC_RXDESC_LAST); + + MGETHDR(mnew, M_DONTWAIT, MT_DATA); + if (mnew == NULL) { + ifp->if_ierrors++; + goto skip; + } + MCLGET(mnew, M_DONTWAIT); + if ((mnew->m_flags & M_EXT) == 0) { + m_freem(mnew); + ifp->if_ierrors++; + goto skip; + } + + bus_dmamap_sync(sc->sc_dmat, rd->rd_map, 0, + rd->rd_map->dm_mapsize, BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(sc->sc_dmat, rd->rd_map); + + error = bus_dmamap_load(sc->sc_dmat, rd->rd_map, + mtod(mnew, void *), MCLBYTES, NULL, + BUS_DMA_READ | BUS_DMA_NOWAIT); + if (error) { + m_freem(mnew); + error = bus_dmamap_load(sc->sc_dmat, rd->rd_map, + mtod(rd->rd_m, void *), MCLBYTES, NULL, + BUS_DMA_READ | BUS_DMA_NOWAIT); + if (error) + panic("%s: could not load old rx mbuf", + device_xname(sc->sc_dev)); + ifp->if_ierrors++; + goto skip; + } + + m = rd->rd_m; + rd->rd_m = mnew; + rx->rx_ptr = htole32(rd->rd_map->dm_segs[0].ds_addr); + + m->m_pkthdr.len = m->m_len = len; + m->m_pkthdr.rcvif = ifp; + m->m_flags |= M_HASFCS; + + bpf_mtap(ifp, m); + ifp->if_ipackets++; + ifp->if_input(ifp, m); + +skip: + bus_dmamap_sync(sc->sc_dmat, rd->rd_map, 0, + rd->rd_map->dm_mapsize, BUS_DMASYNC_PREREAD); + rx->rx_info = htole32(EMAC_RXDESC_OWN | + __SHIFTIN(RKEMAC_MAX_PACKET, EMAC_RXDESC_RXLEN)); + bus_dmamap_sync(sc->sc_dmat, sc->sc_ring_dmamap, + RX_DESC_OFFSET(i), sizeof(struct rkemac_rxdesc), + BUS_DMASYNC_PREWRITE | BUS_DMASYNC_PREREAD); + } + + sc->sc_rxq.r_cur = i; +} + Index: src/sys/arch/arm/rockchip/rockchip_emacreg.h diff -u /dev/null src/sys/arch/arm/rockchip/rockchip_emacreg.h:1.1 --- /dev/null Sun Jan 4 03:53:02 2015 +++ src/sys/arch/arm/rockchip/rockchip_emacreg.h Sun Jan 4 03:53:02 2015 @@ -0,0 +1,139 @@ +/* $NetBSD: rockchip_emacreg.h,v 1.1 2015/01/04 03:53:02 jmcneill Exp $ */ + +/*- + * Copyright (c) 2015 Jared D. McNeill <jmcne...@invisible.ca> + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +#ifndef _ROCKCHIP_EMACREG_H +#define _ROCKCHIP_EMACREG_H + +#define EMAC_ID_REG 0x0000 +#define EMAC_STAT_REG 0x0004 +#define EMAC_ENABLE_REG 0x0008 +#define EMAC_CONTROL_REG 0x000c +#define EMAC_POLLRATE_REG 0x0010 +#define EMAC_RXERR_REG 0x0014 +#define EMAC_MISS_REG 0x0018 +#define EMAC_TXRINGPTR_REG 0x001c +#define EMAC_RXRINGPTR_REG 0x0020 +#define EMAC_ADDRL_REG 0x0024 +#define EMAC_ADDRH_REG 0x0028 +#define EMAC_LAFL_REG 0x002c +#define EMAC_LAFH_REG 0x0030 +#define EMAC_MDIO_DATA_REG 0x0034 +#define EMAC_TXRINGPTR_READ_REG 0x0038 +#define EMAC_RXRINGPTR_READ_REG 0x003c + +#define EMAC_ID_REVISION __BITS(23,16) +#define EMAC_ID_NOTID __BITS(13,8) +#define EMAC_ID_ID __BITS(5,0) + +#define EMAC_STAT_TXPL __BIT(31) +#define EMAC_STAT_MDIO __BIT(12) +#define EMAC_STAT_RXFL __BIT(10) +#define EMAC_STAT_RXFR __BIT(9) +#define EMAC_STAT_RXCR __BIT(8) +#define EMAC_STAT_MSER __BIT(4) +#define EMAC_STAT_TXCH __BIT(3) +#define EMAC_STAT_ERR __BIT(2) +#define EMAC_STAT_RXINT __BIT(1) +#define EMAC_STAT_TXINT __BIT(0) + +#define EMAC_ENABLE_TXPL __BIT(31) +#define EMAC_ENABLE_MDIO __BIT(12) +#define EMAC_ENABLE_RXFL __BIT(10) +#define EMAC_ENABLE_RXFR __BIT(9) +#define EMAC_ENABLE_RXCR __BIT(8) +#define EMAC_ENABLE_MSER __BIT(4) +#define EMAC_ENABLE_TXCH __BIT(3) +#define EMAC_ENABLE_ERR __BIT(2) +#define EMAC_ENABLE_RXINT __BIT(1) +#define EMAC_ENABLE_TXINT __BIT(0) + +#define EMAC_CONTROL_RXBDTLEN __BITS(31,24) +#define EMAC_CONTROL_TXBDTLEN __BITS(23,16) +#define EMAC_CONTROL_DISAD __BIT(15) +#define EMAC_CONTROL_DISRT __BIT(14) +#define EMAC_CONTROL_TEST __BIT(13) +#define EMAC_CONTROL_DIS2P __BIT(12) +#define EMAC_CONTROL_PROM __BIT(11) +#define EMAC_CONTROL_ENFL __BIT(10) +#define EMAC_CONTROL_DISBC __BIT(8) +#define EMAC_CONTROL_RXRN __BIT(4) +#define EMAC_CONTROL_TXRN __BIT(3) +#define EMAC_CONTROL_EN __BIT(0) + +#define EMAC_POLLRATE_POLLRATE __BITS(14,0) + +#define EMAC_RXERR_RXOFLOW __BITS(23,16) +#define EMAC_RXERR_RXFRAM __BITS(15,8) +#define EMAC_RXERR_RXCRC __BITS(7,0) + +#define EMAC_MISS_MISSCNTR __BITS(7,0) + +#define EMAC_MDIO_DATA_SFD __BITS(31,30) +#define EMAC_MDIO_DATA_SFD_SFD 1 +#define EMAC_MDIO_DATA_OP __BITS(29,28) +#define EMAC_MDIO_DATA_OP_WRITE 1 +#define EMAC_MDIO_DATA_OP_READ 2 +#define EMAC_MDIO_DATA_PHYADDR __BITS(27,23) +#define EMAC_MDIO_DATA_PHYREG __BITS(22,18) +#define EMAC_MDIO_DATA_TA __BITS(17,16) +#define EMAC_MDIO_DATA_TA_TA 2 +#define EMAC_MDIO_DATA_DATA __BITS(15,0) + +struct rkemac_txdesc { + uint32_t tx_info; +/* CPU bits */ +#define EMAC_TXDESC_OWN __BIT(31) +#define EMAC_TXDESC_ADCR __BIT(18) +#define EMAC_TXDESC_LAST __BIT(17) +#define EMAC_TXDESC_FIRST __BIT(16) +#define EMAC_TXDESC_TXLEN __BITS(10,0) +/* VMAC bits */ +#define EMAC_TXDESC_UFLO __BIT(29) +#define EMAC_TXDESC_LATECOL __BIT(28) +#define EMAC_TXDESC_RTRY __BITS(27,24) +#define EMAC_TXDESC_DRP __BIT(23) +#define EMAC_TXDESC_DEFR __BIT(22) +#define EMAC_TXDESC_CRLS __BIT(21) + uint32_t tx_ptr; +}; + +struct rkemac_rxdesc { + uint32_t rx_info; +/* CPU bits */ +#define EMAC_RXDESC_OWN __BIT(31) +#define EMAC_RXDESC_LAST __BIT(17) +#define EMAC_RXDESC_FIRST __BIT(16) +#define EMAC_RXDESC_RXLEN __BITS(10,0) +/* VMAC bits */ +#define EMAC_RXDESC_BUFF __BIT(30) + uint32_t rx_ptr; +}; + + + +#endif /* !_ROCKCHIP_EMACREG_H */