diff options
Diffstat (limited to 'sys/net')
-rw-r--r-- | sys/net/if_sl.c | 138 | ||||
-rw-r--r-- | sys/net/if_slvar.h | 5 |
2 files changed, 79 insertions, 64 deletions
diff --git a/sys/net/if_sl.c b/sys/net/if_sl.c index c1ac7a7fa4a..a0784101bfe 100644 --- a/sys/net/if_sl.c +++ b/sys/net/if_sl.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_sl.c,v 1.10 2001/03/13 05:16:06 mickey Exp $ */ +/* $OpenBSD: if_sl.c,v 1.11 2001/05/17 18:41:46 provos Exp $ */ /* $NetBSD: if_sl.c,v 1.39.4.1 1996/06/02 16:26:31 thorpej Exp $ */ /* @@ -227,21 +227,23 @@ static int slinit(sc) register struct sl_softc *sc; { - register caddr_t p; - if (sc->sc_ep == (u_char *) 0) { - MCLALLOC(p, M_WAIT); - if (p) - sc->sc_ep = (u_char *)p + SLBUFSIZE; - else { + MGETHDR(sc->sc_mbuf, M_WAIT, MT_DATA); + if (sc->sc_mbuf) + MCLGET(sc->sc_mbuf, M_WAIT); + if (sc->sc_mbuf == NULL || sc->sc_mbuf->m_ext.ext_buf == NULL) { printf("sl%d: can't allocate buffer\n", sc->sc_unit); sc->sc_if.if_flags &= ~IFF_UP; return (0); } } - sc->sc_buf = sc->sc_ep - SLMAX; - sc->sc_mp = sc->sc_buf; + sc->sc_ep = (u_char *) sc->sc_mbuf->m_ext.ext_buf + + sc->sc_mbuf->m_ext.ext_size; + sc->sc_mp = sc->sc_pktstart = (u_char *) sc->sc_mbuf->m_ext.ext_buf + + BUFOFFSET; + sl_compress_init(&sc->sc_comp); + return (1); } @@ -317,26 +319,28 @@ slclose(tp) int s; ttywflush(tp); - s = splimp(); /* actually, max(spltty, splsoftnet) */ tp->t_line = 0; sc = (struct sl_softc *)tp->t_sc; if (sc != NULL) { + s = splimp(); /* actually, max(spltty, splsoftnet) */ + if_down(&sc->sc_if); sc->sc_ttyp = NULL; tp->t_sc = NULL; - MCLFREE((caddr_t)(sc->sc_ep - SLBUFSIZE)); - sc->sc_ep = 0; - sc->sc_mp = 0; - sc->sc_buf = 0; - } + + m_freem(sc->sc_mbuf); + sc->sc_mbuf = NULL; + sc->sc_ep = sc->sc_mp = sc->sc_pktstart = NULL; + #if defined(__NetBSD__) || defined(__OpenBSD__) - /* if necessary, install a new outq buffer of the appropriate size */ - if (sc->sc_oldbufsize != 0) { - clfree(&tp->t_outq); - clalloc(&tp->t_outq, sc->sc_oldbufsize, sc->sc_oldbufquot); - } + /* if necessary, install a new outq buffer of the appropriate size */ + if (sc->sc_oldbufsize != 0) { + clfree(&tp->t_outq); + clalloc(&tp->t_outq, sc->sc_oldbufsize, sc->sc_oldbufquot); + } #endif - splx(s); + splx(s); + } } /* @@ -652,34 +656,31 @@ sl_btom(sc, len) register struct sl_softc *sc; register int len; { - register struct mbuf *m; - - MGETHDR(m, M_DONTWAIT, MT_DATA); - if (m == NULL) - return (NULL); + struct mbuf *m; /* - * If we have more than MHLEN bytes, it's cheaper to - * queue the cluster we just filled & allocate a new one - * for the input buffer. Otherwise, fill the mbuf we - * allocated above. Note that code in the input routine - * guarantees that packet will fit in a cluster. + * Allocate a new input buffer and swap. */ - if (len >= MHLEN) { - MCLGET(m, M_DONTWAIT); - if ((m->m_flags & M_EXT) == 0) { - /* - * we couldn't get a cluster - if memory's this - * low, it's time to start dropping packets. - */ - (void) m_free(m); - return (NULL); - } - sc->sc_ep = mtod(m, u_char *) + SLBUFSIZE; - m->m_data = (caddr_t)sc->sc_buf; - m->m_ext.ext_buf = (caddr_t)((long)sc->sc_buf &~ MCLOFSET); - } else - bcopy((caddr_t)sc->sc_buf, mtod(m, caddr_t), len); + m = sc->sc_mbuf; + MGETHDR(sc->sc_mbuf, M_DONTWAIT, MT_DATA); + if (sc->sc_mbuf == NULL) { + sc->sc_mbuf = m; + return (NULL); + } + MCLGET(sc->sc_mbuf, M_DONTWAIT); + if ((sc->sc_mbuf->m_flags & M_EXT) == 0) { + /* + * we couldn't get a cluster - if memory's this + * low, it's time to start dropping packets. + */ + m_freem(sc->sc_mbuf); + sc->sc_mbuf = m; + return (NULL); + } + sc->sc_ep = (u_char *) sc->sc_mbuf->m_ext.ext_buf + + sc->sc_mbuf->m_ext.ext_size; + + m->m_data = sc->sc_pktstart; m->m_len = len; m->m_pkthdr.len = len; @@ -763,7 +764,7 @@ slinput(c, tp) sc->sc_flags &= ~SC_ERROR; goto newpack; } - len = sc->sc_mp - sc->sc_buf; + len = sc->sc_mp - sc->sc_pktstart; if (len < 3) /* less than min length packet - ignore */ goto newpack; @@ -778,15 +779,15 @@ slinput(c, tp) * where the buffer started so we can * compute the new header length. */ - bcopy(sc->sc_buf, chdr, CHDR_LEN); + bcopy(sc->sc_pktstart, chdr, CHDR_LEN); } #endif - if ((c = (*sc->sc_buf & 0xf0)) != (IPVERSION << 4)) { + if ((c = (*sc->sc_pktstart & 0xf0)) != (IPVERSION << 4)) { if (c & 0x80) c = TYPE_COMPRESSED_TCP; else if (c == TYPE_UNCOMPRESSED_TCP) - *sc->sc_buf &= 0x4f; /* XXX */ + *sc->sc_pktstart &= 0x4f; /* XXX */ /* * We've got something that's not an IP packet. * If compression is enabled, try to decompress it. @@ -795,13 +796,13 @@ slinput(c, tp) * enable compression. Otherwise, drop it. */ if (sc->sc_if.if_flags & SC_COMPRESS) { - len = sl_uncompress_tcp(&sc->sc_buf, len, + len = sl_uncompress_tcp(&sc->sc_pktstart, len, (u_int)c, &sc->sc_comp); if (len <= 0) goto error; } else if ((sc->sc_if.if_flags & SC_AUTOCOMP) && c == TYPE_UNCOMPRESSED_TCP && len >= 40) { - len = sl_uncompress_tcp(&sc->sc_buf, len, + len = sl_uncompress_tcp(&sc->sc_pktstart, len, (u_int)c, &sc->sc_comp); if (len <= 0) goto error; @@ -809,24 +810,36 @@ slinput(c, tp) } else goto error; } + + m = sl_btom(sc, len); + if (m == NULL) + goto error; + #if NBPFILTER > 0 if (sc->sc_bpf) { /* * Put the SLIP pseudo-"link header" in place. - * We couldn't do this any earlier since - * decompression probably moved the buffer - * pointer. Then, invoke BPF. + * Note this M_PREPEND() should bever fail, + * since we know we always have enough space + * in the input buffer. */ - register u_char *hp = sc->sc_buf - SLIP_HDRLEN; + u_char *hp; + + M_PREPEND(m, SLIP_HDRLEN, M_DONTWAIT); + if (m == NULL) + goto error; + hp = mtod(m, u_char *); hp[SLX_DIR] = SLIPDIR_IN; - bcopy(chdr, &hp[SLX_CHDR], CHDR_LEN); - bpf_tap(sc->sc_bpf, hp, len + SLIP_HDRLEN); + memcpy(&hp[SLX_CHDR], chdr, CHDR_LEN); + + s = splnet(); + bpf_mtap(sc->sc_bpf, m); + splx(s); + + m_adj(m, SLIP_HDRLEN); } #endif - m = sl_btom(sc, len); - if (m == NULL) - goto error; sc->sc_if.if_ipackets++; sc->sc_if.if_lastchange = time; @@ -855,7 +868,8 @@ slinput(c, tp) error: sc->sc_if.if_ierrors++; newpack: - sc->sc_mp = sc->sc_buf = sc->sc_ep - SLMAX; + sc->sc_mp = sc->sc_pktstart = (u_char *) sc->sc_mbuf->m_ext.ext_buf + + BUFOFFSET; sc->sc_escape = 0; } diff --git a/sys/net/if_slvar.h b/sys/net/if_slvar.h index 9f52e09b92b..75c21d64e5e 100644 --- a/sys/net/if_slvar.h +++ b/sys/net/if_slvar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_slvar.h,v 1.5 2001/03/13 05:16:07 mickey Exp $ */ +/* $OpenBSD: if_slvar.h,v 1.6 2001/05/17 18:41:46 provos Exp $ */ /* $NetBSD: if_slvar.h,v 1.16 1996/05/07 02:40:46 thorpej Exp $ */ /*- @@ -49,7 +49,8 @@ struct sl_softc { struct tty *sc_ttyp; /* pointer to tty structure */ u_char *sc_mp; /* pointer to next available buf char */ u_char *sc_ep; /* pointer to last available buf char */ - u_char *sc_buf; /* input buffer */ + u_char *sc_pktstart; /* pointer to beginning of packet */ + struct mbuf *sc_mbuf; /* input buffer */ u_int sc_flags; /* see below */ u_int sc_escape; /* =1 if last char input was FRAME_ESCAPE */ long sc_lasttime; /* last time a char arrived */ |