diff options
Diffstat (limited to 'sys/dev')
-rw-r--r-- | sys/dev/pci/if_nfe.c | 259 | ||||
-rw-r--r-- | sys/dev/pci/if_nfereg.h | 10 | ||||
-rw-r--r-- | sys/dev/pci/if_nfevar.h | 3 |
3 files changed, 174 insertions, 98 deletions
diff --git a/sys/dev/pci/if_nfe.c b/sys/dev/pci/if_nfe.c index d72e7691e9a..056afff5793 100644 --- a/sys/dev/pci/if_nfe.c +++ b/sys/dev/pci/if_nfe.c @@ -1,5 +1,6 @@ -/* $OpenBSD: if_nfe.c,v 1.8 2006/01/20 22:02:03 brad Exp $ */ -/* +/* $OpenBSD: if_nfe.c,v 1.9 2006/01/22 21:35:08 damien Exp $ */ + +/*- * Copyright (c) 2006 Damien Bergamini <damien.bergamini@free.fr> * Copyright (c) 2005, 2006 Jonathan Gray <jsg@openbsd.org> * @@ -67,15 +68,12 @@ int nfe_match(struct device *, void *, void *); void nfe_attach(struct device *, struct device *, void *); - +void nfe_power(int, void *); +void nfe_miibus_statchg(struct device *); +int nfe_miibus_readreg(struct device *, int, int); +void nfe_miibus_writereg(struct device *, int, int, int); int nfe_intr(void *); -int nfe_alloc_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); -void nfe_reset_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); -void nfe_free_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); -int nfe_alloc_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); -void nfe_reset_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); -void nfe_free_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); - +int nfe_ioctl(struct ifnet *, u_long, caddr_t); void nfe_txdesc32_sync(struct nfe_softc *, struct nfe_desc32 *, int); void nfe_txdesc64_sync(struct nfe_softc *, struct nfe_desc64 *, int); void nfe_rxdesc32_sync(struct nfe_softc *, struct nfe_desc32 *, int); @@ -83,34 +81,33 @@ void nfe_rxdesc64_sync(struct nfe_softc *, struct nfe_desc64 *, int); void nfe_rxeof(struct nfe_softc *); void nfe_txeof(struct nfe_softc *); int nfe_encap(struct nfe_softc *, struct mbuf *); - -int nfe_ioctl(struct ifnet *, u_long, caddr_t); void nfe_start(struct ifnet *); -void nfe_stop(struct ifnet *, int); void nfe_watchdog(struct ifnet *); int nfe_init(struct ifnet *); +void nfe_stop(struct ifnet *, int); +int nfe_alloc_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); +void nfe_reset_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); +void nfe_free_rx_ring(struct nfe_softc *, struct nfe_rx_ring *); +int nfe_alloc_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); +void nfe_reset_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); +void nfe_free_tx_ring(struct nfe_softc *, struct nfe_tx_ring *); +int nfe_mediachange(struct ifnet *); +void nfe_mediastatus(struct ifnet *, struct ifmediareq *); void nfe_setmulti(struct nfe_softc *); void nfe_get_macaddr(struct nfe_softc *, uint8_t *); void nfe_set_macaddr(struct nfe_softc *, const uint8_t *); void nfe_update_promisc(struct nfe_softc *); void nfe_tick(void *); -int nfe_miibus_readreg(struct device *, int, int); -void nfe_miibus_writereg(struct device *, int, int, int); -void nfe_miibus_statchg(struct device *); -int nfe_mediachange(struct ifnet *); -void nfe_mediastatus(struct ifnet *, struct ifmediareq *); - struct cfattach nfe_ca = { - sizeof (struct nfe_softc), - nfe_match, - nfe_attach + sizeof (struct nfe_softc), nfe_match, nfe_attach }; struct cfdriver nfe_cd = { - 0, "nfe", DV_IFNET + NULL, "nfe", DV_IFNET }; +#define NFE_DEBUG #ifdef NFE_DEBUG int nfedebug = 1; @@ -157,16 +154,15 @@ nfe_attach(struct device *parent, struct device *self, void *aux) struct ifnet *ifp; bus_size_t memsize; - /* - * Map control/status registers. - */ if (pci_mapreg_map(pa, NFE_PCI_BA, PCI_MAPREG_TYPE_MEM, 0, &sc->sc_memt, &sc->sc_memh, NULL, &memsize, 0) != 0) { printf(": can't map mem space\n"); return; } - /* Allocate interrupt */ + /* + * Allocate interrupt. + */ if (pci_intr_map(pa, &ih) != 0) { printf(": couldn't map interrupt\n"); return; @@ -237,8 +233,6 @@ nfe_attach(struct device *parent, struct device *self, void *aux) ifp->if_baudrate = 1000000000; IFQ_SET_MAXLEN(&ifp->if_snd, NFE_IFQ_MAXLEN); IFQ_SET_READY(&ifp->if_snd); - - /* Set interface name */ strlcpy(ifp->if_xname, sc->sc_dev.dv_xname, IFNAMSIZ); sc->sc_mii.mii_ifp = ifp; @@ -246,7 +240,6 @@ nfe_attach(struct device *parent, struct device *self, void *aux) sc->sc_mii.mii_writereg = nfe_miibus_writereg; sc->sc_mii.mii_statchg = nfe_miibus_statchg; - /* XXX always seem to get a ghost ukphy along with eephy on nf4u */ ifmedia_init(&sc->sc_mii.mii_media, 0, nfe_mediachange, nfe_mediastatus); mii_attach(self, &sc->sc_mii, 0xffffffff, MII_PHY_ANY, @@ -262,14 +255,30 @@ nfe_attach(struct device *parent, struct device *self, void *aux) if_attach(ifp); ether_ifattach(ifp); - /* XXX powerhook */ + sc->sc_powerhook = powerhook_establish(nfe_power, sc); +} + +void +nfe_power(int why, void *arg) +{ + struct nfe_softc *sc = arg; + struct ifnet *ifp; + + if (why == PWR_RESUME) { + ifp = &sc->sc_arpcom.ac_if; + if (ifp->if_flags & IFF_UP) { + nfe_init(ifp); + if (ifp->if_flags & IFF_RUNNING) + nfe_start(ifp); + } + } } void nfe_miibus_statchg(struct device *dev) { struct nfe_softc *sc = (struct nfe_softc *)dev; - struct mii_data *mii = &sc->sc_mii; + struct mii_data *mii = &sc->sc_mii; uint32_t reg; reg = NFE_READ(sc, NFE_PHY_INT); @@ -290,40 +299,69 @@ int nfe_miibus_readreg(struct device *dev, int phy, int reg) { struct nfe_softc *sc = (struct nfe_softc *)dev; - uint32_t r; + uint32_t val; + int ntries; + + NFE_WRITE(sc, NFE_PHY_STATUS, 0xf); - r = NFE_READ(sc, NFE_PHY_CTL); - if (r & NFE_PHY_BUSY) { + if (NFE_READ(sc, NFE_PHY_CTL) & NFE_PHY_BUSY) { NFE_WRITE(sc, NFE_PHY_CTL, NFE_PHY_BUSY); - delay(100); + DELAY(100); } - NFE_WRITE(sc, NFE_PHY_CTL, reg | (phy << NFE_PHYADD_SHIFT)); - delay(1000); - r = NFE_READ(sc, NFE_PHY_DATA); - if (r != 0xffffffff && r != 0) + NFE_WRITE(sc, NFE_PHY_CTL, (phy << NFE_PHYADD_SHIFT) | reg); + + for (ntries = 0; ntries < 1000; ntries++) { + DELAY(100); + if (!(NFE_READ(sc, NFE_PHY_CTL) & NFE_PHY_BUSY)) + break; + } + if (ntries == 1000) { + DPRINTFN(2, ("timeout waiting for PHY\n")); + return 0; + } + + if (NFE_READ(sc, NFE_PHY_STATUS) & NFE_PHY_ERROR) { + DPRINTFN(2, ("could not read PHY\n")); + return 0; + } + + val = NFE_READ(sc, NFE_PHY_DATA); + if (val != 0xffffffff && val != 0) sc->phyaddr = phy; - DPRINTFN(2, ("nfe mii read phy %d reg 0x%x ret 0x%x\n", phy, reg, r)); + DPRINTFN(2, ("mii read phy %d reg 0x%x ret 0x%x\n", phy, reg, val)); - return r; + return val; } void -nfe_miibus_writereg(struct device *dev, int phy, int reg, int data) +nfe_miibus_writereg(struct device *dev, int phy, int reg, int val) { struct nfe_softc *sc = (struct nfe_softc *)dev; - uint32_t r; + uint32_t ctl; + int ntries; + + NFE_WRITE(sc, NFE_PHY_STATUS, 0xf); - r = NFE_READ(sc, NFE_PHY_CTL); - if (r & NFE_PHY_BUSY) { + if (NFE_READ(sc, NFE_PHY_CTL) & NFE_PHY_BUSY) { NFE_WRITE(sc, NFE_PHY_CTL, NFE_PHY_BUSY); - delay(100); + DELAY(100); } - NFE_WRITE(sc, NFE_PHY_DATA, data); - r = reg | (phy << NFE_PHYADD_SHIFT) | NFE_PHY_WRITE; - NFE_WRITE(sc, NFE_PHY_CTL, r); + NFE_WRITE(sc, NFE_PHY_DATA, val); + ctl = NFE_PHY_WRITE | (phy << NFE_PHYADD_SHIFT) | reg; + NFE_WRITE(sc, NFE_PHY_CTL, ctl); + + for (ntries = 0; ntries < 1000; ntries++) { + DELAY(100); + if (!(NFE_READ(sc, NFE_PHY_CTL) & NFE_PHY_BUSY)) + break; + } +#ifdef NFE_DEBUG + if (nfedebug >= 2 && ntries == 1000) + printf("could not write to PHY\n"); +#endif } int @@ -479,7 +517,7 @@ nfe_rxeof(struct nfe_softc *sc) for (;;) { data = &sc->rxq.data[sc->rxq.cur]; - if (sc->sc_flags & NFE_40BIT_ADDR) { /* const condition */ + if (sc->sc_flags & NFE_40BIT_ADDR) { desc64 = &sc->rxq.desc64[sc->rxq.cur]; nfe_rxdesc64_sync(sc, desc64, BUS_DMASYNC_POSTREAD); @@ -499,6 +537,7 @@ nfe_rxeof(struct nfe_softc *sc) if ((sc->sc_flags & (NFE_JUMBO_SUP | NFE_40BIT_ADDR)) == 0) { if (!(flags & NFE_RX_VALID_V1)) goto skip; + if ((flags & NFE_RX_FIXME_V1) == NFE_RX_FIXME_V1) { flags &= ~NFE_RX_ERROR; len--; /* fix buffer length */ @@ -506,6 +545,7 @@ nfe_rxeof(struct nfe_softc *sc) } else { if (!(flags & NFE_RX_VALID_V2)) goto skip; + if ((flags & NFE_RX_FIXME_V2) == NFE_RX_FIXME_V2) { flags &= ~NFE_RX_ERROR; len--; /* fix buffer length */ @@ -560,7 +600,7 @@ nfe_rxeof(struct nfe_softc *sc) } /* - * New mbuf successfully loaded, update Rx ring and continue + * New mbuf successfully loaded, update Rx ring and continue * processing. */ m = data->m; @@ -570,8 +610,19 @@ nfe_rxeof(struct nfe_softc *sc) m->m_pkthdr.len = m->m_len = len; m->m_pkthdr.rcvif = ifp; +#ifdef notyet + if (sc->sc_flags & NFE_HW_CSUM) { + if (flags & NFE_RX_IP_CSUMOK) + m->m_pkthdr.csum_flags |= M_IPV4_CSUM_IN_OK; + if (flags & NFE_RX_UDP_CSUMOK) + m->m_pkthdr.csum_flags |= M_UDPV4_CSUM_OUT; + if (flags & NFE_RX_TCP_CSUMOK) + m->m_pkthdr.csum_flags |= M_TCPV4_CSUM_OUT; + } +#else if ((sc->sc_flags & NFE_HW_CSUM) && (flags & NFE_RX_CSUMOK)) m->m_pkthdr.csum_flags = M_IPV4_CSUM_IN_OK; +#endif #if NBPFILTER > 0 if (ifp->if_bpf) @@ -580,7 +631,7 @@ nfe_rxeof(struct nfe_softc *sc) ifp->if_ipackets++; ether_input_mbuf(ifp, m); -skip: if (sc->sc_flags & NFE_40BIT_ADDR) { /* const condition */ +skip: if (sc->sc_flags & NFE_40BIT_ADDR) { #if defined(__amd64__) desc64->physaddr[0] = htole32(data->map->dm_segs->ds_addr >> 32); @@ -617,7 +668,7 @@ nfe_txeof(struct nfe_softc *sc) for (;;) { data = &sc->txq.data[sc->txq.next]; - if (sc->sc_flags & NFE_40BIT_ADDR) { /* const condition */ + if (sc->sc_flags & NFE_40BIT_ADDR) { desc64 = &sc->txq.desc64[sc->txq.next]; nfe_txdesc64_sync(sc, desc64, BUS_DMASYNC_POSTREAD); @@ -632,13 +683,30 @@ nfe_txeof(struct nfe_softc *sc) if (!(flags & NFE_TX_VALID)) break; - if (data->m == NULL) - goto skip; /* skip intermediate fragments */ + if ((sc->sc_flags & (NFE_JUMBO_SUP | NFE_40BIT_ADDR)) == 0) { + if (!(flags & NFE_TX_LASTFRAG_V1)) + goto skip; - if (flags & NFE_TX_ERROR) - ifp->if_oerrors++; - else - ifp->if_opackets++; + if ((flags & NFE_TX_ERROR_V1) != 0) { + DPRINTF(("tx error 0x%04x\n", flags)); + ifp->if_oerrors++; + } else + ifp->if_opackets++; + } else { + if (!(flags & NFE_TX_LASTFRAG_V2)) + goto skip; + + if ((flags & NFE_TX_ERROR_V2) != 0) { + DPRINTF(("tx error 0x%04x\n", flags)); + ifp->if_oerrors++; + } else + ifp->if_opackets++; + } + + if (data->m == NULL) { /* should not get there */ + DPRINTF(("last fragment bit w/o associated mbuf!\n")); + goto skip; + } /* last fragment of the mbuf chain transmitted */ bus_dmamap_sync(sc->sc_dmat, data->active, 0, @@ -711,6 +779,11 @@ nfe_encap(struct nfe_softc *sc, struct mbuf *m0) } } + if (sc->txq.queued + map->dm_nsegs >= NFE_TX_RING_COUNT - 1) { + bus_dmamap_unload(sc->sc_dmat, data->active); + return ENOBUFS; + } + /* h/w checksum (XXX only if HW_CSUM?) */ if (m0->m_pkthdr.csum_flags & M_IPV4_CSUM_OUT) flags |= NFE_TX_IP_CSUM; @@ -718,10 +791,9 @@ nfe_encap(struct nfe_softc *sc, struct mbuf *m0) flags |= NFE_TX_TCP_CSUM; for (i = 0; i < map->dm_nsegs; i++) { - data = &sc->txq.data[sc->txq.cur]; - if (sc->sc_flags & NFE_40BIT_ADDR) { /* const condition */ + if (sc->sc_flags & NFE_40BIT_ADDR) { desc64 = &sc->txq.desc64[sc->txq.cur]; #if defined(__amd64__) desc64->physaddr[0] = @@ -743,7 +815,7 @@ nfe_encap(struct nfe_softc *sc, struct mbuf *m0) nfe_txdesc32_sync(sc, desc32, BUS_DMASYNC_PREWRITE); } - /* csum flags belong to the first frament only */ + /* csum flags belong to the first fragment only */ if (map->dm_nsegs > 1) flags &= ~(M_TCPV4_CSUM_OUT | M_UDPV4_CSUM_OUT); @@ -826,30 +898,6 @@ nfe_watchdog(struct ifnet *ifp) ifp->if_oerrors++; } -void -nfe_stop(struct ifnet *ifp, int disable) -{ - struct nfe_softc *sc = ifp->if_softc; - - timeout_del(&sc->sc_timeout); - - ifp->if_timer = 0; - ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); - - /* abort Tx */ - NFE_WRITE(sc, NFE_TX_CTL, 0); - - /* disable Rx */ - NFE_WRITE(sc, NFE_RX_CTL, 0); - - /* disable interrupts */ - NFE_WRITE(sc, NFE_IRQ_MASK, 0); - - /* reset Tx and Rx rings */ - nfe_reset_tx_ring(sc, &sc->txq); - nfe_reset_rx_ring(sc, &sc->rxq); -} - int nfe_init(struct ifnet *ifp) { @@ -869,7 +917,7 @@ nfe_init(struct ifnet *ifp) rxtxctl |= NFE_RXTX_RXCHECK; NFE_WRITE(sc, NFE_RXTX_CTL, NFE_RXTX_RESET | rxtxctl); - delay(10); + DELAY(10); NFE_WRITE(sc, NFE_RXTX_CTL, rxtxctl); NFE_WRITE(sc, NFE_SETUP_R6, 0); @@ -887,14 +935,14 @@ nfe_init(struct ifnet *ifp) NFE_WRITE(sc, NFE_SETUP_R1, NFE_R1_MAGIC); NFE_WRITE(sc, NFE_SETUP_R2, NFE_R2_MAGIC); - NFE_WRITE(sc, NFE_TIMER_INT, 970); /* XXX Magic */ + NFE_WRITE(sc, NFE_TIMER_INT, 970); /* XXX Magic */ NFE_WRITE(sc, NFE_SETUP_R4, NFE_R4_MAGIC); NFE_WRITE(sc, NFE_WOL_CTL, NFE_WOL_MAGIC); rxtxctl &= ~NFE_RXTX_BIT2; NFE_WRITE(sc, NFE_RXTX_CTL, rxtxctl); - delay(10); + DELAY(10); NFE_WRITE(sc, NFE_RXTX_CTL, NFE_RXTX_BIT1 | rxtxctl); /* enable Rx */ @@ -905,6 +953,8 @@ nfe_init(struct ifnet *ifp) nfe_setmulti(sc); + NFE_WRITE(sc, NFE_PHY_STATUS, 0xf); + /* enable interrupts */ NFE_WRITE(sc, NFE_IRQ_MASK, NFE_IRQ_WANTED); @@ -918,6 +968,30 @@ nfe_init(struct ifnet *ifp) return 0; } +void +nfe_stop(struct ifnet *ifp, int disable) +{ + struct nfe_softc *sc = ifp->if_softc; + + timeout_del(&sc->sc_timeout); + + ifp->if_timer = 0; + ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); + + /* abort Tx */ + NFE_WRITE(sc, NFE_TX_CTL, 0); + + /* disable Rx */ + NFE_WRITE(sc, NFE_RX_CTL, 0); + + /* disable interrupts */ + NFE_WRITE(sc, NFE_IRQ_MASK, 0); + + /* reset Tx and Rx rings */ + nfe_reset_tx_ring(sc, &sc->txq); + nfe_reset_rx_ring(sc, &sc->rxq); +} + int nfe_alloc_rx_ring(struct nfe_softc *sc, struct nfe_rx_ring *ring) { @@ -975,7 +1049,6 @@ nfe_alloc_rx_ring(struct nfe_softc *sc, struct nfe_rx_ring *ring) /* * Pre-allocate Rx buffers and populate Rx ring. */ - for (i = 0; i < NFE_RX_RING_COUNT; i++) { data = &sc->rxq.data[i]; @@ -1320,9 +1393,9 @@ void nfe_set_macaddr(struct nfe_softc *sc, const uint8_t *addr) { NFE_WRITE(sc, NFE_MACADDR_LO, - addr[0] << 8 | addr[1]); + addr[5] << 8 | addr[4]); NFE_WRITE(sc, NFE_MACADDR_HI, - addr[2] << 24 | addr[3] << 16 | addr[4] << 8 | addr[5]); + addr[3] << 24 | addr[2] << 16 | addr[1] << 8 | addr[0]); } void diff --git a/sys/dev/pci/if_nfereg.h b/sys/dev/pci/if_nfereg.h index e73f2883c27..94a786107d2 100644 --- a/sys/dev/pci/if_nfereg.h +++ b/sys/dev/pci/if_nfereg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_nfereg.h,v 1.3 2006/01/18 20:44:51 damien Exp $ */ +/* $OpenBSD: if_nfereg.h,v 1.4 2006/01/22 21:35:08 damien Exp $ */ /* * Copyright (c) 2005 Jonathan Gray <jsg@openbsd.org> * @@ -17,7 +17,6 @@ #define NFE_PCI_BA 0x10 -#define NFE_PHYADD_SHIFT 5 #define NFE_RX_RING_COUNT 128 #define NFE_TX_RING_COUNT 64 @@ -53,7 +52,7 @@ #define NFE_SETUP_R3 0x13C #define NFE_SETUP_R7 0x140 #define NFE_RXTX_CTL 0x144 -#define NFE_MEDIA_STATUS 0x180 +#define NFE_PHY_STATUS 0x180 #define NFE_SETUP_R4 0x184 #define NFE_STATUS 0x188 #define NFE_PHY_SPEED 0x18c @@ -65,8 +64,10 @@ #define NFE_PWR_CAP 0x268 #define NFE_PWR_STATE 0x26c +#define NFE_PHY_ERROR 0x00001 #define NFE_PHY_WRITE 0x00400 #define NFE_PHY_BUSY 0x08000 +#define NFE_PHYADD_SHIFT 5 #define NFE_R1_MAGIC 0x16070f #define NFE_R2_MAGIC 0x16 @@ -134,8 +135,9 @@ struct nfe_desc64 { #define NFE_TX_TCP_CSUM (1 << 10) #define NFE_TX_IP_CSUM (1 << 11) #define NFE_TX_LASTFRAG_V2 (1 << 13) -#define NFE_TX_ERROR (1 << 14) #define NFE_TX_VALID (1 << 15) +#define NFE_TX_ERROR_V1 0x7808 +#define NFE_TX_ERROR_V2 0x5c04 } __packed; #define NFE_READ(sc, reg) \ diff --git a/sys/dev/pci/if_nfevar.h b/sys/dev/pci/if_nfevar.h index 84589eb2b3e..fa786b56d98 100644 --- a/sys/dev/pci/if_nfevar.h +++ b/sys/dev/pci/if_nfevar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_nfevar.h,v 1.4 2006/01/18 20:44:51 damien Exp $ */ +/* $OpenBSD: if_nfevar.h,v 1.5 2006/01/22 21:35:08 damien Exp $ */ /* * Copyright (c) 2005 Jonathan Gray <jsg@openbsd.org> * @@ -60,6 +60,7 @@ struct nfe_softc { bus_dma_tag_t sc_dmat; struct mii_data sc_mii; struct timeout sc_timeout; + void *sc_powerhook; u_int sc_flags; #define NFE_JUMBO_SUP 0x01 |