Source-Changes-HG archive

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

[src/trunk]: src/sys/kern change m_defrag() to coalesce the chain to single m...



details:   https://anonhg.NetBSD.org/src/rev/cd54640c67e9
branches:  trunk
changeset: 931240:cd54640c67e9
user:      jdolecek <jdolecek%NetBSD.org@localhost>
date:      Fri Apr 24 22:07:12 2020 +0000

description:
change m_defrag() to coalesce the chain to single mbuf if it's short enough
and first mbuf doesn't use external storage

most fragmented packets end up with first short mbuf containing
frame + protocol header only, and second mbuf containing the data;
m_defrag() previously always returned chain of at least two mbufs,
now it should actually return all data in single mbuf for typical
mbuf chain with length < MCLBYTES

diffstat:

 sys/kern/uipc_mbuf.c |  38 ++++++++++++++++++++++++++++++++++++--
 1 files changed, 36 insertions(+), 2 deletions(-)

diffs (59 lines):

diff -r c7460450d274 -r cd54640c67e9 sys/kern/uipc_mbuf.c
--- a/sys/kern/uipc_mbuf.c      Fri Apr 24 19:47:03 2020 +0000
+++ b/sys/kern/uipc_mbuf.c      Fri Apr 24 22:07:12 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: uipc_mbuf.c,v 1.237 2020/03/15 23:14:41 thorpej Exp $  */
+/*     $NetBSD: uipc_mbuf.c,v 1.238 2020/04/24 22:07:12 jdolecek Exp $ */
 
 /*
  * Copyright (c) 1999, 2001, 2018 The NetBSD Foundation, Inc.
@@ -62,7 +62,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: uipc_mbuf.c,v 1.237 2020/03/15 23:14:41 thorpej Exp $");
+__KERNEL_RCSID(0, "$NetBSD: uipc_mbuf.c,v 1.238 2020/04/24 22:07:12 jdolecek Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_mbuftrace.h"
@@ -1672,6 +1672,40 @@
        if (m->m_next == NULL)
                return m;
 
+       /* Defrag to single mbuf if at all possible */
+       if ((m->m_flags & M_EXT) == 0) {
+               if (m->m_pkthdr.len <= MHLEN) {
+                       if (M_TRAILINGSPACE(m) < (m->m_pkthdr.len - m->m_len)) {
+                               KASSERT(M_LEADINGSPACE(m) >=
+                                   (m->m_pkthdr.len - m->m_len));
+                               memmove(m->m_pktdat, m->m_data, m->m_len);
+                               m->m_data = m->m_pktdat;
+                       }
+
+                       KASSERT(M_TRAILINGSPACE(m) >=
+                           (m->m_pkthdr.len - m->m_len));
+                       if (__predict_false(!m_ensure_contig(&m,
+                           m->m_pkthdr.len))) {
+                               panic("m_ensure_contig(%d) failed\n",
+                                   m->m_pkthdr.len);
+                       }
+                       return m;
+               } else if (m->m_pkthdr.len <= MCLBYTES) {
+                       void *odata = m->m_data;
+
+                       MCLGET(m, how);
+                       if ((m->m_flags & M_EXT) == 0)
+                               return NULL;
+                       memcpy(m->m_data, odata, m->m_len);
+                       if (m_pulldown(m, m->m_len, m->m_pkthdr.len - m->m_len,
+                           NULL) == NULL) {
+                               panic("m_pulldown(%d, %d) failed\n",
+                                   m->m_len, m->m_pkthdr.len - m->m_len);
+                       }
+                       return m;
+               }
+       }
+
        m0 = m_get(how, MT_DATA);
        if (m0 == NULL)
                return NULL;



Home | Main Index | Thread Index | Old Index