Module Name: src
Committed By: matt
Date: Thu Jun 5 03:48:08 UTC 2014
Modified Files:
src/sys/arch/arm/allwinner: awin_eth.c
Log Message:
Start of the 10/100 driver.
To generate a diff of this commit:
cvs rdiff -u -r1.3 -r1.4 src/sys/arch/arm/allwinner/awin_eth.c
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/allwinner/awin_eth.c
diff -u src/sys/arch/arm/allwinner/awin_eth.c:1.3 src/sys/arch/arm/allwinner/awin_eth.c:1.4
--- src/sys/arch/arm/allwinner/awin_eth.c:1.3 Sun Sep 8 04:06:44 2013
+++ src/sys/arch/arm/allwinner/awin_eth.c Thu Jun 5 03:48:08 2014
@@ -31,12 +31,15 @@
#include <sys/cdefs.h>
-__KERNEL_RCSID(1, "$NetBSD: awin_eth.c,v 1.3 2013/09/08 04:06:44 matt Exp $");
+__KERNEL_RCSID(1, "$NetBSD: awin_eth.c,v 1.4 2014/06/05 03:48:08 matt Exp $");
+#include <sys/param.h>
#include <sys/bus.h>
#include <sys/device.h>
#include <sys/intr.h>
+#include <sys/ioctl.h>
#include <sys/mutex.h>
+#include <sys/rnd.h>
#include <sys/systm.h>
#include <net/if.h>
@@ -51,10 +54,47 @@ __KERNEL_RCSID(1, "$NetBSD: awin_eth.c,v
static int awin_eth_match(device_t, cfdata_t, void *);
static void awin_eth_attach(device_t, device_t, void *);
+static int awin_eth_intr(void *);
+
static int awin_eth_miibus_read_reg(device_t, int, int);
static void awin_eth_miibus_write_reg(device_t, int, int, int);
static void awin_eth_miibus_statchg(struct ifnet *);
+static void awin_eth_ifstart(struct ifnet *);
+static int awin_eth_ifioctl(struct ifnet *, u_long, void *);
+static int awin_eth_ifinit(struct ifnet *);
+static void awin_eth_ifstop(struct ifnet *, int);
+static void awin_eth_ifwatchdog(struct ifnet *);
+static void awin_eth_ifdrain(struct ifnet *);
+
+struct awin_eth_softc;
+static void awin_eth_rx_hash(struct awin_eth_softc *);
+
+struct awin_eth_regs {
+ uint32_t reg_ctl;
+ uint32_t reg_ts_mode;
+ uint32_t reg_ts_flow;
+ uint32_t reg_ts_ctl[2];
+ uint32_t reg_ts_ins;
+ uint32_t reg_ts_pl[2];
+ uint32_t reg_ts_sta;
+ uint32_t reg_rx_ctl;
+ uint32_t reg_rx_hash[2];
+ uint32_t reg_rx_sta;
+ uint32_t reg_rx_fbc;
+ uint32_t reg_int_ctl;
+ uint32_t reg_mac_ctl0;
+ uint32_t reg_mac_ctl1;
+ uint32_t reg_mac_ipgt;
+ uint32_t reg_mac_ipgr;
+ uint32_t reg_mac_clrt;
+ uint32_t reg_mac_maxf;
+ uint32_t reg_mac_supp;
+ uint32_t reg_mac_a0;
+ uint32_t reg_mac_a1;
+ uint32_t reg_mac_a2;
+};
+
struct awin_eth_softc {
device_t sc_dev;
bus_space_tag_t sc_bst;
@@ -62,7 +102,12 @@ struct awin_eth_softc {
bus_dma_tag_t sc_dmat;
struct ethercom sc_ec;
struct mii_data sc_mii;
+ krndsource_t sc_rnd_source; /* random source */
+ kmutex_t sc_intr_lock;
kmutex_t sc_mdio_lock;
+ uint8_t sc_tx_active;
+ struct awin_eth_regs sc_reg;
+ void *sc_ih;
};
CFATTACH_DECL_NEW(awin_eth, sizeof(struct awin_eth_softc),
@@ -74,17 +119,25 @@ static const struct awin_gpio_pinset awi
};
static inline uint32_t
-awin_eth_read_4(struct awin_eth_softc *sc, bus_size_t o)
+awin_eth_read(struct awin_eth_softc *sc, bus_size_t o)
{
return bus_space_read_4(sc->sc_bst, sc->sc_bsh, o);
}
static inline void
-awin_eth_write_4(struct awin_eth_softc *sc, bus_size_t o, uint32_t v)
+awin_eth_write(struct awin_eth_softc *sc, bus_size_t o, uint32_t v)
{
return bus_space_write_4(sc->sc_bst, sc->sc_bsh, o, v);
}
+static inline void
+awin_eth_clear_set(struct awin_eth_softc *sc, bus_size_t o, uint32_t c,
+ uint32_t s)
+{
+ uint32_t v = bus_space_read_4(sc->sc_bst, sc->sc_bsh, o);
+ return bus_space_write_4(sc->sc_bst, sc->sc_bsh, o, (v & ~c) | s);
+}
+
static int
awin_eth_match(device_t parent, cfdata_t cf, void *aux)
{
@@ -116,11 +169,22 @@ awin_eth_attach(device_t parent, device_
&awin_eth_pinsets[cf->cf_flags & 1];
struct ifnet * const ifp = &sc->sc_ec.ec_if;
struct mii_data * const mii = &sc->sc_mii;
+ char enaddr[ETHER_ADDR_LEN];
sc->sc_dev = self;
awin_gpio_pinset_acquire(pinset);
+ awin_reg_set_clear(aio->aio_core_bst, aio->aio_ccm_bsh,
+ AWIN_AHB_GATING0_REG, AWIN_AHB_GATING0_EMAC, 0);
+ /*
+ * Give 13KB of SRAM to EMAC
+ */
+ awin_reg_set_clear(aio->aio_core_bst, aio->aio_core_bsh,
+ AWIN_SRAM_OFFSET + AWIN_SRAM_CTL1_REG,
+ __SHIFTIN(AWIN_SRAM_CTL1_A3_A4_EMAC, AWIN_SRAM_CTL1_A3_A4),
+ AWIN_SRAM_CTL1_A3_A4);
+ mutex_init(&sc->sc_intr_lock, MUTEX_DEFAULT, IPL_NET);
mutex_init(&sc->sc_mdio_lock, MUTEX_DEFAULT, IPL_NET);
sc->sc_bst = aio->aio_core_bst;
@@ -131,7 +195,40 @@ awin_eth_attach(device_t parent, device_
aprint_naive("\n");
aprint_normal(": 10/100 Ethernet Controller\n");
+ /*
+ * Diable and then clear all interrupts
+ */
+ awin_eth_write(sc, AWIN_EMAC_INT_CTL_REG, 0);
+ awin_eth_write(sc, AWIN_EMAC_INT_STA_REG,
+ awin_eth_read(sc, AWIN_EMAC_INT_STA_REG));
+
+ sc->sc_ih = intr_establish(loc->loc_intr, IPL_NET, IST_LEVEL,
+ awin_eth_intr, sc);
+ if (sc->sc_ih == NULL) {
+ aprint_error_dev(self, "failed to establish interrupt %d\n",
+ loc->loc_intr);
+ return;
+ }
+ aprint_normal_dev(self, "interrupting on irq %d\n", loc->loc_intr);
+
+ uint32_t a1 = awin_eth_read(sc, AWIN_EMAC_MAC_A1_REG);
+ uint32_t a0 = awin_eth_read(sc, AWIN_EMAC_MAC_A0_REG);
+ if (a0 != 0 || a1 != 0) {
+ enaddr[0] = a1 >> 16;
+ enaddr[1] = a1 >> 8;
+ enaddr[2] = a1 >> 0;
+ enaddr[3] = a0 >> 16;
+ enaddr[4] = a0 >> 8;
+ enaddr[5] = a0 >> 0;
+ }
+
ifp->if_softc = sc;
+ ifp->if_start = awin_eth_ifstart;
+ ifp->if_ioctl = awin_eth_ifioctl;
+ ifp->if_init = awin_eth_ifinit;
+ ifp->if_stop = awin_eth_ifstop;
+ ifp->if_watchdog = awin_eth_ifwatchdog;
+ ifp->if_drain = awin_eth_ifdrain;
ifmedia_init(&mii->mii_media, 0, ether_mediachange, ether_mediastatus);
@@ -140,9 +237,8 @@ awin_eth_attach(device_t parent, device_
mii->mii_writereg = awin_eth_miibus_write_reg;
mii->mii_statchg = awin_eth_miibus_statchg;
- int mii_flags = 0;
-
- mii_attach(self, mii, 0xffffffff, MII_PHY_ANY, MII_OFFSET_ANY, mii_flags);
+ mii_attach(self, mii, 0xffffffff, MII_PHY_ANY, MII_OFFSET_ANY,
+ MIIF_DOPAUSE);
if (LIST_EMPTY(&mii->mii_phys)) {
aprint_error_dev(self, "no PHY found!\n");
@@ -151,6 +247,14 @@ awin_eth_attach(device_t parent, device_
} else {
ifmedia_set(&mii->mii_media, IFM_ETHER|IFM_AUTO);
}
+
+ /*
+ * Attach the interface.
+ */
+ if_attach(ifp);
+ ether_ifattach(ifp, enaddr);
+ rnd_attach_source(&sc->sc_rnd_source, device_xname(self),
+ RND_TYPE_NET, 0);
}
int
@@ -160,13 +264,13 @@ awin_eth_miibus_read_reg(device_t self,
mutex_enter(&sc->sc_mdio_lock);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MADR_REG, (phy << 8) | reg);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MCMD_REG, 1);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MADR_REG, (phy << 8) | reg);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MCMD_REG, 1);
delay(100);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MCMD_REG, 0);
- const uint32_t rv = awin_eth_read_4(sc, AWIN_EMAC_MAC_MRDD_REG);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MCMD_REG, 0);
+ const uint32_t rv = awin_eth_read(sc, AWIN_EMAC_MAC_MRDD_REG);
mutex_exit(&sc->sc_mdio_lock);
@@ -180,13 +284,13 @@ awin_eth_miibus_write_reg(device_t self,
mutex_enter(&sc->sc_mdio_lock);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MADR_REG, (phy << 8) | reg);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MCMD_REG, 1);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MADR_REG, (phy << 8) | reg);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MCMD_REG, 1);
delay(100);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MCMD_REG, 0);
- awin_eth_write_4(sc, AWIN_EMAC_MAC_MWTD_REG, val);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MCMD_REG, 0);
+ awin_eth_write(sc, AWIN_EMAC_MAC_MWTD_REG, val);
mutex_exit(&sc->sc_mdio_lock);
}
@@ -196,14 +300,472 @@ awin_eth_miibus_statchg(struct ifnet *if
{
struct awin_eth_softc * const sc = ifp->if_softc;
struct mii_data * const mii = &sc->sc_mii;
+ const u_int media = mii->mii_media_active;
/*
- * Set MII or GMII interface based on the speed
+ * Set MII interface based on the speed
* negotiated by the PHY.
*/
- switch (IFM_SUBTYPE(mii->mii_media_active)) {
+ switch (IFM_SUBTYPE(media)) {
case IFM_10_T:
+ sc->sc_reg.reg_mac_supp &= ~AWIN_EMAC_MAC_SUPP_100M;
+ break;
case IFM_100_TX:
- ;
+ sc->sc_reg.reg_mac_supp |= AWIN_EMAC_MAC_SUPP_100M;
+ break;
+ }
+
+ sc->sc_reg.reg_mac_ctl0 &=
+ ~(AWIN_EMAC_MAC_CTL0_TFC|AWIN_EMAC_MAC_CTL0_RFC);
+ if (media & IFM_FLOW) {
+ if (media & IFM_ETH_TXPAUSE) {
+ sc->sc_reg.reg_mac_ctl0 |= AWIN_EMAC_MAC_CTL0_TFC;
+ }
+ if (media & IFM_ETH_RXPAUSE) {
+ sc->sc_reg.reg_mac_ctl0 |= AWIN_EMAC_MAC_CTL0_RFC;
+ }
+ }
+
+ sc->sc_reg.reg_mac_ctl1 &= ~AWIN_EMAC_MAC_CTL1_FD;
+ if (media & IFM_FDX) {
+ sc->sc_reg.reg_mac_ctl1 |= AWIN_EMAC_MAC_CTL1_FD;
+ }
+}
+
+static inline void
+awin_eth_int_enable(struct awin_eth_softc *sc)
+{
+ awin_eth_clear_set(sc, AWIN_EMAC_INT_CTL_REG, 0,
+ AWIN_EMAC_INT_RX|AWIN_EMAC_INT_TX0|AWIN_EMAC_INT_TX1);
+}
+
+static inline void
+awin_eth_rxfifo_flush(struct awin_eth_softc *sc)
+{
+ uint32_t rxctl_reg = awin_eth_read(sc, AWIN_EMAC_RX_CTL_REG);
+
+ if (rxctl_reg & AWIN_EMAC_RX_CTL_DMA) {
+ awin_eth_write(sc, AWIN_EMAC_RX_CTL_REG,
+ rxctl_reg & ~AWIN_EMAC_RX_CTL_DMA);
+ }
+ awin_eth_write(sc, AWIN_EMAC_RX_CTL_REG,
+ (rxctl_reg & ~AWIN_EMAC_RX_CTL_DMA)
+ | AWIN_EMAC_RX_CTL_FIFO_RESET);
+
+ for (;;) {
+ uint32_t v0 = awin_eth_read(sc, AWIN_EMAC_RX_CTL_REG);
+ if ((v0 & AWIN_EMAC_RX_CTL_FIFO_RESET) == 0)
+ break;
+ }
+ awin_eth_write(sc, AWIN_EMAC_RX_CTL_REG, rxctl_reg);
+}
+
+static void
+awin_eth_rxfifo_consume(struct awin_eth_softc *sc, size_t len)
+{
+ for (len = (len + 3) >> 2; len > 0; len--) {
+ (void) awin_eth_read(sc, AWIN_EMAC_RX_IO_DATA_REG);
+ }
+}
+
+static void
+awin_eth_rxfifo_transfer(struct awin_eth_softc *sc, struct mbuf *m)
+{
+ uint32_t *dp32 = (uint32_t *)(m->m_data);
+
+ bus_space_read_multi_4(sc->sc_bst, sc->sc_bsh,
+ AWIN_EMAC_RX_IO_DATA_REG, dp32, m->m_len >> 2);
+
+ /*
+ * Now we have to copy any remaining bytes that if it wasn't
+ * a multiple of 4 bytes.
+ */
+ if (m->m_len & 3) {
+ const uint32_t v = awin_eth_read(sc, AWIN_EMAC_RX_IO_DATA_REG);
+ memcpy(m->m_data, &v, m->m_len & 3);
+ }
+
+ /*
+ * Now all we have is the CRC to read and discard. We may have read
+ * to three bytes of it but there is at least one more byte to read
+ * discard.
+ */
+ (void) awin_eth_read(sc, AWIN_EMAC_RX_IO_DATA_REG);
+}
+
+static struct mbuf *
+awin_eth_mgethdr(struct awin_eth_softc *sc, size_t rxlen)
+{
+ struct mbuf *m = m_gethdr(M_DONTWAIT, MT_DATA);
+
+ if (rxlen + 2 > MHLEN) {
+ MCLGET(m, M_DONTWAIT);
+ if ((m->m_flags & M_EXT) == 0) {
+ m_free(m);
+ return NULL;
+ }
+ }
+
+ m->m_data += 2;
+ m->m_len = rxlen;
+ m->m_pkthdr.len = rxlen;
+ m->m_pkthdr.rcvif = &sc->sc_ec.ec_if;
+
+ return m;
+}
+
+static void
+awin_eth_if_input(struct awin_eth_softc *sc, struct mbuf *m)
+{
+ struct ifnet * const ifp = &sc->sc_ec.ec_if;
+
+ (*ifp->if_input)(ifp, m);
+}
+
+static void
+awin_eth_rx_intr(struct awin_eth_softc *sc)
+{
+ for (;;) {
+ uint32_t rx_count = awin_eth_read(sc, AWIN_EMAC_RX_FBC_REG);
+ struct mbuf *m;
+
+ if (rx_count == 0) {
+ awin_eth_int_enable(sc);
+ rx_count = awin_eth_read(sc, AWIN_EMAC_RX_FBC_REG);
+ if (rx_count == 0)
+ return;
+ }
+
+ uint32_t v = awin_eth_read(sc, AWIN_EMAC_RX_IO_DATA_REG);
+ if (v != AWIN_EMAC_RX_MAGIC) {
+ awin_eth_rxfifo_flush(sc);
+ awin_eth_int_enable(sc);
+ return;
+ }
+
+ uint32_t rxhdr = awin_eth_read(sc, AWIN_EMAC_RX_IO_DATA_REG);
+ uint32_t rxlen = __SHIFTOUT(rxhdr, AWIN_EMAC_RXHDR_LEN);
+ uint32_t rxsts = __SHIFTOUT(rxhdr, AWIN_EMAC_RXHDR_STS);
+ bool drop = false;
+
+ if (rxlen < ETHER_MIN_LEN) {
+ drop = true;
+ } else if (rxsts & (AWIN_EMAC_RX_STA_CRCERR | AWIN_EMAC_RX_STA_LENERR | AWIN_EMAC_RX_STA_ALNERR)) {
+ drop = true;
+ sc->sc_ec.ec_if.if_ierrors++;
+ }
+
+ if (!drop) {
+ m = awin_eth_mgethdr(sc, rxlen - 4);
+ drop = (m == NULL);
+ }
+
+ if (drop) {
+ awin_eth_rxfifo_consume(sc, rxlen);
+ awin_eth_int_enable(sc);
+ return;
+ }
+
+ awin_eth_rxfifo_transfer(sc, m);
+ awin_eth_if_input(sc, m);
+ }
+}
+
+static void
+awin_eth_txfifo_transfer(struct awin_eth_softc *sc, struct mbuf *m, u_int slot)
+{
+ bus_size_t const io_data_reg = AWIN_EMAC_TX_IO_DATA_REG(0 /*slot*/);
+ u_int leftover = 0;
+ uint32_t v = 0;
+ signed int pad = ETHER_MIN_LEN - ETHER_CRC_LEN - m->m_pkthdr.len;
+
+ for (; m != NULL; m = m->m_next) {
+ uint8_t *dp = m->m_data;
+ size_t len = m->m_len;
+ if (len == 0)
+ continue;
+
+ if (leftover > 0) {
+ /*
+ * We do a memcpy instead of dereferencing an uint32_t
+ * in case these are the last bytes of a page which is
+ * followed by an unmapped page.
+ */
+ uint32_t uv = 0;
+ memcpy(&uv, dp, 32 - leftover / 8);
+ if (leftover + len * 8 < 32) {
+ v |= ((uv & ~0L) << (len * 8)) << leftover;
+ leftover += len * 8;
+ continue;
+ }
+ dp += (32 - leftover) / 8;
+ len -= (32 - leftover) / 8;
+ awin_eth_write(sc, io_data_reg, v | (uv << leftover));
+ v = 0;
+ leftover = 0;
+ if (len == 0)
+ continue;
+ }
+ KASSERT(len > 0);
+ KASSERT(leftover == 0);
+ bus_space_write_multi_4(sc->sc_bst, sc->sc_bsh, io_data_reg,
+ (const uint32_t *)dp, len >> 2);
+
+ dp += len & -4;
+ len &= 3;
+
+ if (len == 0)
+ continue;
+
+ /*
+ * We do a memcpy instead of dereferencing an uint32_t in case
+ * these are the last bytes of a page which is followed by an
+ * unmapped page.
+ */
+ uint32_t uv = 0;
+ memcpy(&uv, dp, len);
+ const u_int bits = len * 8;
+ if (bits + leftover < 32) {
+ v |= ((uv & ~0U) << bits) << leftover;
+ leftover += bits;
+ continue;
+ }
+ awin_eth_write(sc, io_data_reg, v | (uv << leftover));
+ if (bits + leftover == 32) {
+ v = 0;
+ leftover = 0;
+ } else {
+ v = uv >> (32 - leftover);
+ leftover = (leftover + bits) & 31;
+ v &= ~0U << leftover;
+ }
+ }
+
+ /*
+ * If we have data yet to be written, write it now (padding by 0).
+ * Be sure to subtract the extra zeroes from the pad bytes.
+ */
+ if (leftover) {
+ awin_eth_write(sc, io_data_reg, v);
+ pad -= 32 - leftover / 8;
+ }
+
+ /*
+ * Pad the packet out to minimum packet size.
+ */
+ for (; pad > 4; pad -= 4) {
+ awin_eth_write(sc, io_data_reg, 0);
+ }
+}
+
+static void
+awin_eth_tx_enqueue(struct awin_eth_softc *sc, struct mbuf *m, u_int slot)
+{
+ awin_eth_write(sc, AWIN_EMAC_TX_INS_REG, slot);
+
+ awin_eth_txfifo_transfer(sc, m, slot);
+
+ bus_size_t const pl_reg = AWIN_EMAC_TX_PL_REG(slot);
+ bus_size_t const ctl_reg = AWIN_EMAC_TX_CTL_REG(slot);
+
+ awin_eth_write(sc, pl_reg, m->m_pkthdr.len);
+ awin_eth_clear_set(sc, ctl_reg, 0, AWIN_EMAC_TX_CTL_START);
+
+ m_freem(m);
+}
+
+static void
+awin_eth_tx_intr(struct awin_eth_softc *sc, u_int slot)
+{
+ struct ifnet * const ifp = &sc->sc_ec.ec_if;
+ struct mbuf *m;
+
+ IF_DEQUEUE(&ifp->if_snd, m);
+
+ if (m == NULL) {
+ awin_eth_tx_enqueue(sc, m, slot);
+ } else {
+ sc->sc_tx_active &= ~__BIT(slot);
+ ifp->if_flags &= ~IFF_OACTIVE;
+ }
+}
+
+int
+awin_eth_intr(void *arg)
+{
+ struct awin_eth_softc * const sc = arg;
+ int rv = 0;
+
+ mutex_enter(&sc->sc_intr_lock);
+
+ for (;;) {
+ uint32_t sts = awin_eth_read(sc, AWIN_EMAC_INT_STA_REG);
+ awin_eth_write(sc, AWIN_EMAC_INT_STA_REG, sts);
+ rnd_add_uint32(&sc->sc_rnd_source, sts);
+
+ if ((sts & (AWIN_EMAC_INT_RX|AWIN_EMAC_INT_TX0|AWIN_EMAC_INT_TX1)) == 0)
+ break;
+
+ rv = 1;
+ if (sts & AWIN_EMAC_INT_RX) {
+ awin_eth_rx_intr(sc);
+ }
+ if (sts & AWIN_EMAC_INT_TX0) {
+ awin_eth_tx_intr(sc, 0);
+ }
+ if (sts & AWIN_EMAC_INT_TX1) {
+ awin_eth_tx_intr(sc, 1);
+ }
+ }
+ mutex_exit(&sc->sc_intr_lock);
+
+ return rv;
+}
+
+void
+awin_eth_ifstart(struct ifnet *ifp)
+{
+ struct awin_eth_softc * const sc = ifp->if_softc;
+
+ mutex_enter(&sc->sc_intr_lock);
+
+ if ((sc->sc_tx_active & 1) == 0) {
+ struct mbuf *m;
+ IF_DEQUEUE(&ifp->if_snd, m);
+ if (m == NULL) {
+ mutex_exit(&sc->sc_intr_lock);
+ return;
+ }
+ awin_eth_tx_enqueue(sc, m, 0);
+ sc->sc_tx_active |= 1;
+ }
+
+ if ((sc->sc_tx_active & 2) == 0) {
+ struct mbuf *m;
+ IF_DEQUEUE(&ifp->if_snd, m);
+ if (m == NULL) {
+ mutex_exit(&sc->sc_intr_lock);
+ return;
+ }
+ awin_eth_tx_enqueue(sc, m, 1);
+ sc->sc_tx_active |= 2;
+ }
+
+ if (sc->sc_tx_active == 3)
+ ifp->if_flags |= IFF_OACTIVE;
+
+ mutex_exit(&sc->sc_intr_lock);
+}
+
+
+static int
+awin_eth_ifioctl(struct ifnet *ifp, u_long cmd, void *data)
+{
+ struct awin_eth_softc * const sc = ifp->if_softc;
+ struct ifreq *ifr = (struct ifreq *)data;
+ int error;
+
+ mutex_enter(&sc->sc_intr_lock);
+
+ switch (cmd) {
+ case SIOCGIFMEDIA:
+ case SIOCSIFMEDIA:
+ error = ifmedia_ioctl(ifp, ifr, &sc->sc_mii.mii_media, cmd);
+ break;
+ default:
+ if ((error = ether_ioctl(ifp, cmd, data)) != ENETRESET)
+ break;
+ error = 0;
+ if (cmd != SIOCADDMULTI && cmd != SIOCDELMULTI)
+ break;
+ if (ifp->if_flags & IFF_RUNNING) {
+ /*
+ * Multicast list has changed; set the hardware filter
+ * accordingly.
+ */
+ awin_eth_ifstop(ifp, 0);
+ error = awin_eth_ifinit(ifp);
+ }
+ break;
+ }
+
+ mutex_exit(&sc->sc_intr_lock);
+
+ return error;
+}
+
+static void
+awin_eth_ifstop(struct ifnet *ifp, int discard)
+{
+ struct awin_eth_softc * const sc = ifp->if_softc;
+
+ KASSERT(mutex_owned(&sc->sc_intr_lock));
+ awin_eth_write(sc, AWIN_EMAC_CTL_REG, 0);
+ delay(200);
+ awin_eth_write(sc, AWIN_EMAC_CTL_REG, 1);
+ delay(200);
+}
+
+int
+awin_eth_ifinit(struct ifnet *ifp)
+{
+ struct awin_eth_softc * const sc = ifp->if_softc;
+
+ awin_eth_rx_hash(sc);
+ return 0;
+}
+
+void
+awin_eth_ifwatchdog(struct ifnet *ifp)
+{
+}
+
+void
+awin_eth_ifdrain(struct ifnet *ifp)
+{
+}
+
+static void
+awin_eth_rx_hash(struct awin_eth_softc *sc)
+{
+ struct ifnet * const ifp = &sc->sc_ec.ec_if;
+ struct ether_multistep step;
+ struct ether_multi *enm;
+
+ if (ifp->if_flags & IFF_PROMISC) {
+ sc->sc_reg.reg_rx_hash[0] = sc->sc_reg.reg_rx_hash[1] = ~0;
+ ifp->if_flags |= IFF_ALLMULTI;
+ return;
+ }
+
+ ETHER_FIRST_MULTI(step, &sc->sc_ec, enm);
+ while (enm != NULL) {
+ if (memcmp(enm->enm_addrlo, enm->enm_addrhi, ETHER_ADDR_LEN)) {
+ /*
+ * We must listen to a range of multicast addresses.
+ * For now, just accept all multicasts, rather than
+ * trying to set only those filter bits needed to match
+ * the range. (At this time, the only use of address
+ * ranges is for IP multicast routing, for which the
+ * range is big enough to require all bits set.)
+ */
+ sc->sc_reg.reg_rx_hash[0] = sc->sc_reg.reg_rx_hash[1] = ~0;
+ ifp->if_flags |= IFF_ALLMULTI;
+ return;
+ }
+
+#if 0
+ u_int crc = ether_crc32_le(enm->enm_addrlo, ETHER_ADDR_LEN);
+#else
+ u_int crc = ether_crc32_be(enm->enm_addrlo, ETHER_ADDR_LEN);
+#endif
+
+ /* Just want the 6 most significant bits. */
+ crc >>= 26;
+
+ /* Set the corresponding bit in the filter. */
+ sc->sc_reg.reg_rx_hash[crc >> 5] |= __BIT(crc & 31);
+ ETHER_NEXT_MULTI(step, enm);
}
+ ifp->if_flags &= ~IFF_ALLMULTI;
}