Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/is-mlppp]: src/sys/net first part of defragmentation code. No dropping/s...



details:   https://anonhg.NetBSD.org/src/rev/1a8398f5e444
branches:  is-mlppp
changeset: 970676:1a8398f5e444
user:      is <is%NetBSD.org@localhost>
date:      Fri Apr 10 17:28:37 2020 +0000

description:
first part of defragmentation code. No dropping/sequence error statistics
yet, and no MRRU negotiation so not active.

diffstat:

 sys/net/if_spppsubr.c |  135 +++++++++++++++++++++++++++++++++++++++++++++++++-
 sys/net/if_spppvar.h  |    4 +-
 2 files changed, 136 insertions(+), 3 deletions(-)

diffs (195 lines):

diff -r 5a3918f18f72 -r 1a8398f5e444 sys/net/if_spppsubr.c
--- a/sys/net/if_spppsubr.c     Tue Apr 07 19:26:44 2020 +0000
+++ b/sys/net/if_spppsubr.c     Fri Apr 10 17:28:37 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_spppsubr.c,v 1.187.2.3 2020/04/07 19:26:44 is Exp $  */
+/*     $NetBSD: if_spppsubr.c,v 1.187.2.4 2020/04/10 17:28:37 is Exp $  */
 
 /*
  * Synchronous PPP/Cisco link level subroutines.
@@ -41,7 +41,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_spppsubr.c,v 1.187.2.3 2020/04/07 19:26:44 is Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_spppsubr.c,v 1.187.2.4 2020/04/10 17:28:37 is Exp $");
 
 #if defined(_KERNEL_OPT)
 #include "opt_inet.h"
@@ -336,6 +336,9 @@
 static void sppp_lcp_check_and_close(struct sppp *sp);
 static int sppp_ncp_check(struct sppp *sp);
 
+static struct mbuf *sppp_ml_defrag(struct sppp *sp, struct mbuf *m,
+                                   u_int16_t *protocolp);
+
 static void sppp_ipcp_init(struct sppp *sp);
 static void sppp_ipcp_up(struct sppp *sp);
 static void sppp_ipcp_down(struct sppp *sp);
@@ -517,6 +520,123 @@
        }
 }
 
+
+/*
+ * Defragment an MP packet. 
+ *
+ * Returns NULL or an assembled packet or the original,
+ * and adjusts the passed protocol to the inner one.
+ *
+ * Called with and returns a packet without PPP header, 
+ * but with MP fragment header.
+ *
+ * Called and returns with lock held.
+ */
+static struct mbuf *
+sppp_ml_defrag(struct sppp *sp, struct mbuf *m, u_int16_t *protocolp)
+{
+       u_int8_t *p;
+       u_int8_t flags;
+       u_int32_t seqid;
+       u_int16_t protocol;
+       u_int16_t *m_protop;
+       int newflen;
+
+       if (*protocolp != PPP_MP)
+               return m;       /* not ours */
+
+       if (sp->lcp.mrru == 0)
+               return m;       /*
+                                * ours, but we're not ready.
+                                * sppp_input will arrange for rejection.
+                                */
+
+       if (m->m_len < 4) {
+               m = m_pullup(m, 4);
+               if (m == NULL) {
+                       if_statadd2(ifp, if_ierrors, 1, if_iqdrops, 1);
+                       return NULL;
+               }
+       }
+       p = mtod(m, u_int8_t *);
+       flags = *p; 
+       seqid = (p[1]<<16) + (p[2]<<8) + p[3];
+
+       m_adj(m, 4);
+
+       /* We're manipulating the defragmentation state below: */
+       SPPP_UPGRADE(sp);
+
+       if (flags & 0x80) {
+               /* Beginning fragment. */
+               sp->lcp.ml_seq_expected=seqid+1;        /* next expected */
+               
+               /* TODO: if prefix, count dropped? */
+
+               m_freem(sp->lcp.ml_prefix);
+               sp->lcp.ml_prefix = m;
+
+       } else if (seqid == sp->lcp.ml_seq_expected) {
+               sp->lcp.ml_seq_expected=seqid+1;        /* next expected */
+               if (sp->lcp.ml_prefix == 0) {
+                       /* didn't see B frame.  */
+                       /* TODO: count as dropped. */
+                       m_freem(m);
+                       return NULL;
+               }
+               /*
+                * m_cat might free the first mbuf (with pkthdr)
+                * in 2nd chain; therefore:
+                */
+               newflen = m->m_pkthdr.len;
+               m_cat(sp->lcp.ml_prefix, m);
+               lcp.ml_prefix->m_pkthdr.len += newflen;
+
+       } else {
+               /*
+                * sequence error. 
+                *
+                * For now, only drop this fragment, and don't touch state-
+                * might be from the long past or future, and we could still
+                * finish our current prefix.
+                *
+                * TODO: count as dropped.
+                */
+                m_freem(sp->lcp.ml_prefix);
+                return NULL;
+       }
+       /* Successfully got the Beginning or appended a non-B packet.*/
+       if (flags & 0x40) {
+               /* B/next was E packet. Unwrap gift and deliver. */
+               m = sp->lcp.ml_prefix;
+               sp->lcp.ml_prefix = NULL;
+
+               if (m->m_len < 2) {
+                       m = m_pullup(m,2)
+                       if (m == NULL) {
+                               * TODO: count as dropped.
+                               return NULL;
+                       }
+               }
+               /* RFC 1990 2.: don't assume no protocol field compression */
+               p = mtod(m, u_int8_t);
+               protocol = *p;
+               
+               if (protocol & 1) {
+                       m_adj(m, 1);
+               } else {
+                       protocol = (protocol << 8) + p[1];
+                       m_adj(m, 2);
+               }
+               *protocolp = protocol;
+
+               return m;
+       }
+
+       return NULL;
+}
+
+
 /*
  * Exported functions, comprising our interface to the lower layer.
  */
@@ -627,6 +747,12 @@
                protocol = ntohs(h->protocol);
        }
 
+       m = sppp_ml_defrag(sp, m, &protocol);
+       if (m == 0) {
+               SPPP_UNLOCK(sp);
+               return;
+       }
+
        switch (protocol) {
        default:
                if (sp->state[IDX_LCP] == STATE_OPENED) {
@@ -2222,6 +2348,11 @@
        /* Initialize activity timestamp: opening a connection is an activity */
        sp->pp_last_receive = sp->pp_last_activity = time_uptime;
 
+       /* Initialize mlppp state */
+       sp->lcp.mrru = sp->lcp.their_mrru = 0;
+       sp->lcp.ml_prefix = NULL;
+       sp->lcp.ml_seq_expected = 0;
+
        /*
         * If this interface is passive or dial-on-demand, and we are
         * still in Initial state, it means we've got an incoming
diff -r 5a3918f18f72 -r 1a8398f5e444 sys/net/if_spppvar.h
--- a/sys/net/if_spppvar.h      Tue Apr 07 19:26:44 2020 +0000
+++ b/sys/net/if_spppvar.h      Fri Apr 10 17:28:37 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: if_spppvar.h,v 1.22.12.2 2020/04/07 18:47:43 is Exp $  */
+/*     $NetBSD: if_spppvar.h,v 1.22.12.3 2020/04/10 17:28:37 is Exp $  */
 
 #ifndef _NET_IF_SPPPVAR_H_
 #define _NET_IF_SPPPVAR_H_
@@ -48,6 +48,8 @@
        /* multilink variables */
        u_long  mrru;           /* our   max received reconstructed unit */
        u_long  their_mrru;     /* their max received reconstructed unit */
+       u_long  ml_seq_xpctd;   /* next seqid we expect to receive */
+       struct mbuf *ml_prefix; /* reassembled packet so far */
 };
 
 #define IDX_IPCP 1             /* idx into state table */



Home | Main Index | Thread Index | Old Index