summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/ic/i82596.c57
1 files changed, 31 insertions, 26 deletions
diff --git a/sys/dev/ic/i82596.c b/sys/dev/ic/i82596.c
index 5c025ab8afb..2592888c356 100644
--- a/sys/dev/ic/i82596.c
+++ b/sys/dev/ic/i82596.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: i82596.c,v 1.9 2002/03/14 01:26:54 millert Exp $ */
+/* $OpenBSD: i82596.c,v 1.10 2002/07/29 23:41:24 fgsch Exp $ */
/* $NetBSD: i82586.c,v 1.18 1998/08/15 04:42:42 mycroft Exp $ */
/*-
@@ -998,18 +998,20 @@ i82596_chk_rx_ring(sc)
* operation considerably. (Provided that it works, of course.)
*/
static __inline__ struct mbuf *
-i82596_get(struct ie_softc *sc, struct ether_header *ehp, int head, int totlen)
+i82596_get(struct ie_softc *sc, int head, int totlen)
{
struct mbuf *top, **mp, *m;
int off, len, resid;
int thisrboff, thismboff;
+ struct ether_header eh;
/*
* Snarf the Ethernet header.
*/
- (sc->memcopyin)(sc, ehp, IE_RBUF_ADDR(sc,head), sizeof *ehp);
+ (sc->memcopyin)(sc, &eh, IE_RBUF_ADDR(sc, head),
+ sizeof(struct ether_header));
- resid = totlen -= (thisrboff = sizeof *ehp);
+ resid = totlen;
MGETHDR(m, M_DONTWAIT, MT_DATA);
if (m == 0)
@@ -1041,6 +1043,13 @@ i82596_get(struct ie_softc *sc, struct ether_header *ehp, int head, int totlen)
}
len = MCLBYTES;
}
+ if (mp == top) {
+ caddr_t newdata = (caddr_t)
+ ALIGN(m->m_data + sizeof(struct ether_header)) -
+ sizeof(struct ether_header);
+ len -= newdata - m->m_data;
+ m->m_data = newdata;
+ }
m->m_len = len = min(totlen, len);
totlen -= len;
*mp = m;
@@ -1051,6 +1060,14 @@ i82596_get(struct ie_softc *sc, struct ether_header *ehp, int head, int totlen)
thismboff = 0;
/*
+ * Copy the Ethernet header into the mbuf chain.
+ */
+ bcopy(&eh, mtod(m, caddr_t), sizeof(struct ether_header));
+ thismboff = sizeof(struct ether_header);
+ thisrboff = sizeof(struct ether_header);
+ resid -= sizeof(struct ether_header);
+
+ /*
* Now we take the mbuf chain (hopefully only one mbuf most of the
* time) and stuff the data into it. There are no possible failures
* at or after this point.
@@ -1103,7 +1120,6 @@ i82596_readframe(sc, num)
{
struct ifnet *ifp = &sc->sc_arpcom.ac_if;
struct mbuf *m;
- struct ether_header eh;
u_int16_t bstart, bend;
int pktlen;
@@ -1112,7 +1128,7 @@ i82596_readframe(sc, num)
return (1);
}
- m = i82596_get(sc, &eh, bstart, pktlen);
+ m = i82596_get(sc, bstart, pktlen);
i82596_release_rbd_list(sc, bstart, bend);
if (m == 0) {
@@ -1121,38 +1137,27 @@ i82596_readframe(sc, num)
}
#ifdef I82596_DEBUG
- if (sc->sc_debug & IED_READFRAME)
+ if (sc->sc_debug & IED_READFRAME) {
+ struct ether_header *eh = mtod(m, struct ether_header *);
+
printf("%s: frame from ether %s type 0x%x len %d\n",
- sc->sc_dev.dv_xname,
- ether_sprintf(eh.ether_shost),
- (u_int)eh.ether_type, pktlen);
+ sc->sc_dev.dv_xname, ether_sprintf(eh->ether_shost),
+ (u_int)eh->ether_type, pktlen);
+ }
#endif
#if NBPFILTER > 0
- /*
- * Check for a BPF filter; if so, hand it up.
- * Note that we have to stick an extra mbuf up front, because bpf_mtap
- * expects to have the ether header at the front.
- * It doesn't matter that this results in an ill-formatted mbuf chain,
- * since BPF just looks at the data. (It doesn't try to free the mbuf,
- * tho' it will make a copy for tcpdump.)
- */
+ /* Check for a BPF filter; if so, hand it up. */
if (ifp->if_bpf) {
- struct mbuf m0;
- m0.m_len = sizeof eh;
- m0.m_data = (caddr_t)&eh;
- m0.m_next = m;
-
/* Pass it up. */
- bpf_mtap(ifp->if_bpf, &m0);
-
+ bpf_mtap(ifp->if_bpf, m);
}
#endif /* NBPFILTER > 0 */
/*
* Finally pass this packet up to higher layers.
*/
- ether_input(ifp, &eh, m);
+ ether_input_mbuf(ifp, m);
ifp->if_ipackets++;
return (0);
}