summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/ic/midway.c28
1 files changed, 23 insertions, 5 deletions
diff --git a/sys/dev/ic/midway.c b/sys/dev/ic/midway.c
index 95b052eb1f2..e50d73d48d1 100644
--- a/sys/dev/ic/midway.c
+++ b/sys/dev/ic/midway.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: midway.c,v 1.8 1996/07/03 17:21:19 chuck Exp $ */
-/* (sync'd to midway.c 1.57) */
+/* $OpenBSD: midway.c,v 1.9 1996/07/11 00:17:10 chuck Exp $ */
+/* (sync'd to midway.c 1.58) */
/*
*
@@ -1360,7 +1360,15 @@ struct ifnet *ifp;
#endif
#ifdef EN_MBUF_OPT
- if (M_LEADINGSPACE(m) >= MID_TBD_SIZE) {
+
+ /*
+ * note: external storage (M_EXT) can be shared between mbufs
+ * to avoid copying (see m_copym()). this means that the same
+ * data buffer could be shared by several mbufs, and thus it isn't
+ * a good idea to try and write TBDs or PDUs to M_EXT data areas.
+ */
+
+ if (M_LEADINGSPACE(m) >= MID_TBD_SIZE && (m->m_flags & M_EXT) == 0) {
m->m_data -= MID_TBD_SIZE;
m->m_len += MID_TBD_SIZE;
mlen += MID_TBD_SIZE;
@@ -1374,7 +1382,8 @@ struct ifnet *ifp;
atm_flags |= EN_OBHDR;
}
- if (toadd && M_TRAILINGSPACE(lastm) >= toadd) {
+ if (toadd && (lastm->m_flags & M_EXT) == 0 &&
+ M_TRAILINGSPACE(lastm) >= toadd) {
cp = mtod(lastm, u_int8_t *) + lastm->m_len;
lastm->m_len += toadd;
mlen += toadd;
@@ -2280,6 +2289,14 @@ defer: /* defer processing */
pdu -= (EN_RXSZ*1024);
pdu = EN_READ(sc, pdu); /* READ swaps to proper byte order */
fill = tlen - MID_RBD_SIZE - MID_PDU_LEN(pdu);
+ if (fill < 0) {
+ printf("%s: invalid AAL5 PDU length detected, dropping frame\n",
+ sc->sc_dev.dv_xname);
+ printf("%s: got %d cells (%d bytes), AAL5 len is %d bytes (pdu=0x%x)\n",
+ sc->sc_dev.dv_xname, MID_RBD_CNT(rbd), tlen - MID_RBD_SIZE,
+ MID_PDU_LEN(pdu), pdu);
+ fill = tlen;
+ }
mlen = tlen - fill;
}
@@ -2299,6 +2316,7 @@ defer: /* defer processing */
*/
m = sc->rxslot[slot].q.ifq_head;
+ drqneed = 1;
if (m) {
sav = mtod(m, u_int32_t *);
if (sav[0] != cur) {
@@ -2319,7 +2337,7 @@ defer: /* defer processing */
}
}
- if (m == NULL) {
+ if (mlen != 0 && m == NULL) {
m = en_mget(sc, mlen, &drqneed); /* allocate! */
if (m == NULL) {
fill += mlen;