123417Smckusick /* 263952Sbostic * Copyright (c) 1982, 1986, 1988, 1991, 1993 363952Sbostic * The Regents of the University of California. All rights reserved. 423417Smckusick * 544448Sbostic * %sccs.include.redist.c% 633185Sbostic * 7*68171Scgd * @(#)uipc_mbuf.c 8.3 (Berkeley) 01/09/95 823417Smckusick */ 94585Swnj 1056517Sbostic #include <sys/param.h> 1156517Sbostic #include <sys/systm.h> 1256517Sbostic #include <sys/proc.h> 1356517Sbostic #include <sys/malloc.h> 1456517Sbostic #include <sys/map.h> 1538962Skarels #define MBTYPES 1656517Sbostic #include <sys/mbuf.h> 1756517Sbostic #include <sys/kernel.h> 1856517Sbostic #include <sys/syslog.h> 1956517Sbostic #include <sys/domain.h> 2056517Sbostic #include <sys/protosw.h> 214585Swnj 2256517Sbostic #include <vm/vm.h> 2356517Sbostic 2448445Skarels extern vm_map_t mb_map; 2548445Skarels struct mbuf *mbutl; 2648445Skarels char *mclrefcnt; 2745736Smckusick 285228Swnj mbinit() 295228Swnj { 3021107Skarels int s; 315228Swnj 3245736Smckusick #if CLBYTES < 4096 3341998Smckusick #define NCL_INIT (4096/CLBYTES) 3432659Skarels #else 3532659Skarels #define NCL_INIT 1 3632659Skarels #endif 3721107Skarels s = splimp(); 3838962Skarels if (m_clalloc(NCL_INIT, M_DONTWAIT) == 0) 395228Swnj goto bad; 4021107Skarels splx(s); 415228Swnj return; 425228Swnj bad: 435228Swnj panic("mbinit"); 445228Swnj } 455228Swnj 4621107Skarels /* 4738962Skarels * Allocate some number of mbuf clusters 4838962Skarels * and place on cluster free list. 4921107Skarels * Must be called at splimp. 5021107Skarels */ 5130442Skarels /* ARGSUSED */ 5252038Smckusick m_clalloc(ncl, nowait) 535228Swnj register int ncl; 5452038Smckusick int nowait; 555228Swnj { 5665439Sbostic static int logged; 5738962Skarels register caddr_t p; 585228Swnj register int i; 5965439Sbostic int npg; 605228Swnj 615228Swnj npg = ncl * CLSIZE; 6252038Smckusick p = (caddr_t)kmem_malloc(mb_map, ctob(npg), !nowait); 6345736Smckusick if (p == NULL) { 6430442Skarels if (logged == 0) { 6530442Skarels logged++; 6645736Smckusick log(LOG_ERR, "mb_map full\n"); 6730442Skarels } 685228Swnj return (0); 6921107Skarels } 7038962Skarels ncl = ncl * CLBYTES / MCLBYTES; 7138962Skarels for (i = 0; i < ncl; i++) { 7238962Skarels ((union mcluster *)p)->mcl_next = mclfree; 7338962Skarels mclfree = (union mcluster *)p; 7438962Skarels p += MCLBYTES; 7538962Skarels mbstat.m_clfree++; 7638962Skarels } 7738962Skarels mbstat.m_clusters += ncl; 7838962Skarels return (1); 7938962Skarels } 805228Swnj 8138962Skarels /* 8238962Skarels * When MGET failes, ask protocols to free space when short of memory, 8338962Skarels * then re-attempt to allocate an mbuf. 8438962Skarels */ 8538962Skarels struct mbuf * 8638962Skarels m_retry(i, t) 8738962Skarels int i, t; 8838962Skarels { 8938962Skarels register struct mbuf *m; 905228Swnj 9138962Skarels m_reclaim(); 9238962Skarels #define m_retry(i, t) (struct mbuf *)0 9338962Skarels MGET(m, i, t); 9438962Skarels #undef m_retry 9538962Skarels return (m); 965228Swnj } 975228Swnj 9821107Skarels /* 9938962Skarels * As above; retry an MGETHDR. 10021107Skarels */ 10138962Skarels struct mbuf * 10238962Skarels m_retryhdr(i, t) 10338962Skarels int i, t; 1045228Swnj { 10538962Skarels register struct mbuf *m; 10638962Skarels 10738962Skarels m_reclaim(); 10838962Skarels #define m_retryhdr(i, t) (struct mbuf *)0 10938962Skarels MGETHDR(m, i, t); 11038962Skarels #undef m_retryhdr 11138962Skarels return (m); 11238962Skarels } 11338962Skarels 11438962Skarels m_reclaim() 11538962Skarels { 11630442Skarels register struct domain *dp; 11730442Skarels register struct protosw *pr; 11838962Skarels int s = splimp(); 1195228Swnj 12038962Skarels for (dp = domains; dp; dp = dp->dom_next) 12138962Skarels for (pr = dp->dom_protosw; pr < dp->dom_protoswNPROTOSW; pr++) 12238962Skarels if (pr->pr_drain) 12338962Skarels (*pr->pr_drain)(); 12438962Skarels splx(s); 12538962Skarels mbstat.m_drain++; 1265228Swnj } 1275228Swnj 1285228Swnj /* 1295228Swnj * Space allocation routines. 1305228Swnj * These are also available as macros 1315228Swnj * for critical paths. 1325228Swnj */ 1334585Swnj struct mbuf * 13452038Smckusick m_get(nowait, type) 13552038Smckusick int nowait, type; 1364585Swnj { 1374585Swnj register struct mbuf *m; 1384585Swnj 13952038Smckusick MGET(m, nowait, type); 1404585Swnj return (m); 1414585Swnj } 1424585Swnj 1434585Swnj struct mbuf * 14452038Smckusick m_gethdr(nowait, type) 14552038Smckusick int nowait, type; 14638962Skarels { 14738962Skarels register struct mbuf *m; 14838962Skarels 14952038Smckusick MGETHDR(m, nowait, type); 15038962Skarels return (m); 15138962Skarels } 15238962Skarels 15338962Skarels struct mbuf * 15452038Smckusick m_getclr(nowait, type) 15552038Smckusick int nowait, type; 1564890Swnj { 1574890Swnj register struct mbuf *m; 1584890Swnj 15952038Smckusick MGET(m, nowait, type); 1604890Swnj if (m == 0) 1614890Swnj return (0); 1624890Swnj bzero(mtod(m, caddr_t), MLEN); 1634890Swnj return (m); 1644890Swnj } 1654890Swnj 1664890Swnj struct mbuf * 1674585Swnj m_free(m) 1684585Swnj struct mbuf *m; 1694585Swnj { 1704585Swnj register struct mbuf *n; 1714585Swnj 1724585Swnj MFREE(m, n); 1734585Swnj return (n); 1744585Swnj } 1754585Swnj 17652394Storek void 1774669Swnj m_freem(m) 1784585Swnj register struct mbuf *m; 1794585Swnj { 1804585Swnj register struct mbuf *n; 1814585Swnj 1824585Swnj if (m == NULL) 1834916Swnj return; 1844585Swnj do { 1854585Swnj MFREE(m, n); 1864585Swnj } while (m = n); 1874585Swnj } 1884585Swnj 1895228Swnj /* 1905228Swnj * Mbuffer utility routines. 1915228Swnj */ 19221107Skarels 19321107Skarels /* 19438962Skarels * Lesser-used path for M_PREPEND: 19538962Skarels * allocate new mbuf to prepend to chain, 19638962Skarels * copy junk along. 19738962Skarels */ 19838962Skarels struct mbuf * 19938962Skarels m_prepend(m, len, how) 20038962Skarels register struct mbuf *m; 20138962Skarels int len, how; 20238962Skarels { 20338962Skarels struct mbuf *mn; 20438962Skarels 20538962Skarels MGET(mn, how, m->m_type); 20638962Skarels if (mn == (struct mbuf *)NULL) { 20738962Skarels m_freem(m); 20838962Skarels return ((struct mbuf *)NULL); 20938962Skarels } 21038962Skarels if (m->m_flags & M_PKTHDR) { 21138962Skarels M_COPY_PKTHDR(mn, m); 21238962Skarels m->m_flags &= ~M_PKTHDR; 21338962Skarels } 21438962Skarels mn->m_next = m; 21538962Skarels m = mn; 21638962Skarels if (len < MHLEN) 21738962Skarels MH_ALIGN(m, len); 21838962Skarels m->m_len = len; 21938962Skarels return (m); 22038962Skarels } 22138962Skarels 22233606Skarels /* 22338962Skarels * Make a copy of an mbuf chain starting "off0" bytes from the beginning, 22421107Skarels * continuing for "len" bytes. If len is M_COPYALL, copy to end of mbuf. 22538962Skarels * The wait parameter is a choice of M_WAIT/M_DONTWAIT from caller. 22621107Skarels */ 22741999Smckusick int MCFail; 22841999Smckusick 2294927Swnj struct mbuf * 23038962Skarels m_copym(m, off0, len, wait) 2314927Swnj register struct mbuf *m; 23238962Skarels int off0, wait; 2334927Swnj register int len; 2344927Swnj { 2354927Swnj register struct mbuf *n, **np; 23638962Skarels register int off = off0; 23738962Skarels struct mbuf *top; 23838962Skarels int copyhdr = 0; 2394927Swnj 2404927Swnj if (off < 0 || len < 0) 24138962Skarels panic("m_copym"); 24238962Skarels if (off == 0 && m->m_flags & M_PKTHDR) 24338962Skarels copyhdr = 1; 2444927Swnj while (off > 0) { 2454927Swnj if (m == 0) 24638962Skarels panic("m_copym"); 2474927Swnj if (off < m->m_len) 2484927Swnj break; 2494927Swnj off -= m->m_len; 2504927Swnj m = m->m_next; 2514927Swnj } 2524927Swnj np = ⊤ 2534927Swnj top = 0; 2544927Swnj while (len > 0) { 2555609Swnj if (m == 0) { 2565609Swnj if (len != M_COPYALL) 25738962Skarels panic("m_copym"); 2585609Swnj break; 2595609Swnj } 26038962Skarels MGET(n, wait, m->m_type); 2614927Swnj *np = n; 2624927Swnj if (n == 0) 2634927Swnj goto nospace; 26438962Skarels if (copyhdr) { 26538962Skarels M_COPY_PKTHDR(n, m); 26638962Skarels if (len == M_COPYALL) 26738962Skarels n->m_pkthdr.len -= off0; 26838962Skarels else 26938962Skarels n->m_pkthdr.len = len; 27038962Skarels copyhdr = 0; 27138962Skarels } 27255058Spendry n->m_len = min(len, m->m_len - off); 27338962Skarels if (m->m_flags & M_EXT) { 27438962Skarels n->m_data = m->m_data + off; 27538962Skarels mclrefcnt[mtocl(m->m_ext.ext_buf)]++; 27638962Skarels n->m_ext = m->m_ext; 27738962Skarels n->m_flags |= M_EXT; 2788318Sroot } else 2794927Swnj bcopy(mtod(m, caddr_t)+off, mtod(n, caddr_t), 2804927Swnj (unsigned)n->m_len); 2815609Swnj if (len != M_COPYALL) 2825609Swnj len -= n->m_len; 2834927Swnj off = 0; 2844927Swnj m = m->m_next; 2854927Swnj np = &n->m_next; 2864927Swnj } 28741999Smckusick if (top == 0) 28841999Smckusick MCFail++; 2894927Swnj return (top); 2904927Swnj nospace: 2914927Swnj m_freem(top); 29241999Smckusick MCFail++; 2934927Swnj return (0); 2944927Swnj } 2954927Swnj 29633606Skarels /* 29733606Skarels * Copy data from an mbuf chain starting "off" bytes from the beginning, 29833606Skarels * continuing for "len" bytes, into the indicated buffer. 29933606Skarels */ 30033606Skarels m_copydata(m, off, len, cp) 30133606Skarels register struct mbuf *m; 30233988Skarels register int off; 30333606Skarels register int len; 30433988Skarels caddr_t cp; 30533606Skarels { 30633606Skarels register unsigned count; 30733606Skarels 30833606Skarels if (off < 0 || len < 0) 30933606Skarels panic("m_copydata"); 31033606Skarels while (off > 0) { 31133606Skarels if (m == 0) 31233606Skarels panic("m_copydata"); 31333606Skarels if (off < m->m_len) 31433606Skarels break; 31533606Skarels off -= m->m_len; 31633606Skarels m = m->m_next; 31733606Skarels } 31833606Skarels while (len > 0) { 31933606Skarels if (m == 0) 32033606Skarels panic("m_copydata"); 32155058Spendry count = min(m->m_len - off, len); 32233606Skarels bcopy(mtod(m, caddr_t) + off, cp, count); 32333606Skarels len -= count; 32433988Skarels cp += count; 32533606Skarels off = 0; 32633606Skarels m = m->m_next; 32733606Skarels } 32833606Skarels } 32933606Skarels 33038962Skarels /* 33138962Skarels * Concatenate mbuf chain n to m. 33238962Skarels * Both chains must be of the same type (e.g. MT_DATA). 33338962Skarels * Any m_pkthdr is not updated. 33438962Skarels */ 3354669Swnj m_cat(m, n) 3364669Swnj register struct mbuf *m, *n; 3374669Swnj { 3384669Swnj while (m->m_next) 3394669Swnj m = m->m_next; 3406091Sroot while (n) { 34138962Skarels if (m->m_flags & M_EXT || 34238962Skarels m->m_data + m->m_len + n->m_len >= &m->m_dat[MLEN]) { 3436091Sroot /* just join the two chains */ 3444669Swnj m->m_next = n; 3456091Sroot return; 3464669Swnj } 3476091Sroot /* splat the data from one into the other */ 3486091Sroot bcopy(mtod(n, caddr_t), mtod(m, caddr_t) + m->m_len, 3496091Sroot (u_int)n->m_len); 3506091Sroot m->m_len += n->m_len; 3516091Sroot n = m_free(n); 3526091Sroot } 3534669Swnj } 3544669Swnj 355*68171Scgd void 35638962Skarels m_adj(mp, req_len) 3574585Swnj struct mbuf *mp; 35852394Storek int req_len; 3594585Swnj { 36038962Skarels register int len = req_len; 36124764Skarels register struct mbuf *m; 36224764Skarels register count; 3634585Swnj 3644585Swnj if ((m = mp) == NULL) 3654585Swnj return; 3664822Swnj if (len >= 0) { 36738962Skarels /* 36838962Skarels * Trim from head. 36938962Skarels */ 3704585Swnj while (m != NULL && len > 0) { 3714822Swnj if (m->m_len <= len) { 3724585Swnj len -= m->m_len; 3734585Swnj m->m_len = 0; 3744585Swnj m = m->m_next; 3754822Swnj } else { 3764585Swnj m->m_len -= len; 37738962Skarels m->m_data += len; 37838962Skarels len = 0; 3794585Swnj } 3804585Swnj } 38138962Skarels m = mp; 38238962Skarels if (mp->m_flags & M_PKTHDR) 38338962Skarels m->m_pkthdr.len -= (req_len - len); 3844822Swnj } else { 38524764Skarels /* 38624764Skarels * Trim from tail. Scan the mbuf chain, 38724764Skarels * calculating its length and finding the last mbuf. 38824764Skarels * If the adjustment only affects this mbuf, then just 38924764Skarels * adjust and return. Otherwise, rescan and truncate 39024764Skarels * after the remaining size. 39124764Skarels */ 3924585Swnj len = -len; 39324764Skarels count = 0; 39424764Skarels for (;;) { 39524764Skarels count += m->m_len; 39624764Skarels if (m->m_next == (struct mbuf *)0) 3974585Swnj break; 39824764Skarels m = m->m_next; 39924764Skarels } 40024764Skarels if (m->m_len >= len) { 40124764Skarels m->m_len -= len; 40263951Smckusick if (mp->m_flags & M_PKTHDR) 40363951Smckusick mp->m_pkthdr.len -= len; 40424764Skarels return; 40524764Skarels } 40624764Skarels count -= len; 40738962Skarels if (count < 0) 40838962Skarels count = 0; 40924764Skarels /* 41024764Skarels * Correct length for chain is "count". 41124764Skarels * Find the mbuf with last data, adjust its length, 41224764Skarels * and toss data from remaining mbufs on chain. 41324764Skarels */ 41438962Skarels m = mp; 41538962Skarels if (m->m_flags & M_PKTHDR) 41638962Skarels m->m_pkthdr.len = count; 41738962Skarels for (; m; m = m->m_next) { 41824764Skarels if (m->m_len >= count) { 41924764Skarels m->m_len = count; 42024764Skarels break; 4214585Swnj } 42224764Skarels count -= m->m_len; 4234585Swnj } 42424764Skarels while (m = m->m_next) 42524764Skarels m->m_len = 0; 4264585Swnj } 4274585Swnj } 4285228Swnj 42923816Skarels /* 43023816Skarels * Rearange an mbuf chain so that len bytes are contiguous 43123816Skarels * and in the data area of an mbuf (so that mtod and dtom 43224764Skarels * will work for a structure of size len). Returns the resulting 43324764Skarels * mbuf chain on success, frees it and returns null on failure. 43438962Skarels * If there is room, it will add up to max_protohdr-len extra bytes to the 43524764Skarels * contiguous region in an attempt to avoid being called next time. 43623816Skarels */ 43741999Smckusick int MPFail; 43841999Smckusick 4395310Sroot struct mbuf * 44024764Skarels m_pullup(n, len) 44124764Skarels register struct mbuf *n; 4425228Swnj int len; 4435228Swnj { 44424764Skarels register struct mbuf *m; 44524764Skarels register int count; 44624764Skarels int space; 4475228Swnj 44838962Skarels /* 44938962Skarels * If first mbuf has no cluster, and has room for len bytes 45038962Skarels * without shifting current data, pullup into it, 45138962Skarels * otherwise allocate a new mbuf to prepend to the chain. 45238962Skarels */ 45338962Skarels if ((n->m_flags & M_EXT) == 0 && 45438962Skarels n->m_data + len < &n->m_dat[MLEN] && n->m_next) { 45538962Skarels if (n->m_len >= len) 45638962Skarels return (n); 45724764Skarels m = n; 45824764Skarels n = n->m_next; 45924764Skarels len -= m->m_len; 46024764Skarels } else { 46138962Skarels if (len > MHLEN) 46224764Skarels goto bad; 46324764Skarels MGET(m, M_DONTWAIT, n->m_type); 46424764Skarels if (m == 0) 46524764Skarels goto bad; 46624764Skarels m->m_len = 0; 46744969Skarels if (n->m_flags & M_PKTHDR) { 46838962Skarels M_COPY_PKTHDR(m, n); 46944969Skarels n->m_flags &= ~M_PKTHDR; 47044969Skarels } 47124764Skarels } 47238962Skarels space = &m->m_dat[MLEN] - (m->m_data + m->m_len); 4735310Sroot do { 47438962Skarels count = min(min(max(len, max_protohdr), space), n->m_len); 47538962Skarels bcopy(mtod(n, caddr_t), mtod(m, caddr_t) + m->m_len, 4766164Ssam (unsigned)count); 4776164Ssam len -= count; 4786164Ssam m->m_len += count; 4796164Ssam n->m_len -= count; 48038962Skarels space -= count; 4815310Sroot if (n->m_len) 48238962Skarels n->m_data += count; 48323816Skarels else 48423816Skarels n = m_free(n); 48524764Skarels } while (len > 0 && n); 48624764Skarels if (len > 0) { 4875310Sroot (void) m_free(m); 4885310Sroot goto bad; 4895310Sroot } 4905310Sroot m->m_next = n; 4915310Sroot return (m); 4925310Sroot bad: 4935732Sroot m_freem(n); 49441999Smckusick MPFail++; 4955228Swnj return (0); 4965228Swnj } 49750941Ssklower 49850941Ssklower /* 49950941Ssklower * Partition an mbuf chain in two pieces, returning the tail -- 50050941Ssklower * all but the first len0 bytes. In case of failure, it returns NULL and 50150941Ssklower * attempts to restore the chain to its original state. 50250941Ssklower */ 50350941Ssklower struct mbuf * 50450941Ssklower m_split(m0, len0, wait) 50552394Storek register struct mbuf *m0; 50652394Storek int len0, wait; 50750941Ssklower { 50850941Ssklower register struct mbuf *m, *n; 50950941Ssklower unsigned len = len0, remain; 51050941Ssklower 51150941Ssklower for (m = m0; m && len > m->m_len; m = m->m_next) 51250941Ssklower len -= m->m_len; 51350941Ssklower if (m == 0) 51450941Ssklower return (0); 51550941Ssklower remain = m->m_len - len; 51650941Ssklower if (m0->m_flags & M_PKTHDR) { 51750941Ssklower MGETHDR(n, wait, m0->m_type); 51850941Ssklower if (n == 0) 51950941Ssklower return (0); 52050941Ssklower n->m_pkthdr.rcvif = m0->m_pkthdr.rcvif; 52150941Ssklower n->m_pkthdr.len = m0->m_pkthdr.len - len0; 52250941Ssklower m0->m_pkthdr.len = len0; 52350941Ssklower if (m->m_flags & M_EXT) 52450941Ssklower goto extpacket; 52550941Ssklower if (remain > MHLEN) { 52650941Ssklower /* m can't be the lead packet */ 52750941Ssklower MH_ALIGN(n, 0); 52850941Ssklower n->m_next = m_split(m, len, wait); 52950941Ssklower if (n->m_next == 0) { 53050941Ssklower (void) m_free(n); 53150941Ssklower return (0); 53250941Ssklower } else 53350941Ssklower return (n); 53450941Ssklower } else 53550941Ssklower MH_ALIGN(n, remain); 53650941Ssklower } else if (remain == 0) { 53750941Ssklower n = m->m_next; 53850941Ssklower m->m_next = 0; 53950941Ssklower return (n); 54050941Ssklower } else { 54150941Ssklower MGET(n, wait, m->m_type); 54250941Ssklower if (n == 0) 54350941Ssklower return (0); 54450941Ssklower M_ALIGN(n, remain); 54550941Ssklower } 54650941Ssklower extpacket: 54750941Ssklower if (m->m_flags & M_EXT) { 54850941Ssklower n->m_flags |= M_EXT; 54950941Ssklower n->m_ext = m->m_ext; 55050941Ssklower mclrefcnt[mtocl(m->m_ext.ext_buf)]++; 55150941Ssklower m->m_ext.ext_size = 0; /* For Accounting XXXXXX danger */ 55250941Ssklower n->m_data = m->m_data + len; 55350941Ssklower } else { 55450941Ssklower bcopy(mtod(m, caddr_t) + len, mtod(n, caddr_t), remain); 55550941Ssklower } 55650941Ssklower n->m_len = remain; 55750941Ssklower m->m_len = len; 55850941Ssklower n->m_next = m->m_next; 55950941Ssklower m->m_next = 0; 56050941Ssklower return (n); 56150941Ssklower } 56250941Ssklower /* 56350941Ssklower * Routine to copy from device local memory into mbufs. 56450941Ssklower */ 56550941Ssklower struct mbuf * 56650941Ssklower m_devget(buf, totlen, off0, ifp, copy) 56750941Ssklower char *buf; 56850941Ssklower int totlen, off0; 56950941Ssklower struct ifnet *ifp; 57050941Ssklower void (*copy)(); 57150941Ssklower { 57250941Ssklower register struct mbuf *m; 57350941Ssklower struct mbuf *top = 0, **mp = ⊤ 57450941Ssklower register int off = off0, len; 57550941Ssklower register char *cp; 57650941Ssklower char *epkt; 57750941Ssklower 57850941Ssklower cp = buf; 57950941Ssklower epkt = cp + totlen; 58050941Ssklower if (off) { 58150941Ssklower cp += off + 2 * sizeof(u_short); 58250941Ssklower totlen -= 2 * sizeof(u_short); 58350941Ssklower } 58450941Ssklower MGETHDR(m, M_DONTWAIT, MT_DATA); 58550941Ssklower if (m == 0) 58650941Ssklower return (0); 58750941Ssklower m->m_pkthdr.rcvif = ifp; 58850941Ssklower m->m_pkthdr.len = totlen; 58950941Ssklower m->m_len = MHLEN; 59050941Ssklower 59150941Ssklower while (totlen > 0) { 59250941Ssklower if (top) { 59350941Ssklower MGET(m, M_DONTWAIT, MT_DATA); 59450941Ssklower if (m == 0) { 59550941Ssklower m_freem(top); 59650941Ssklower return (0); 59750941Ssklower } 59850941Ssklower m->m_len = MLEN; 59950941Ssklower } 60050941Ssklower len = min(totlen, epkt - cp); 60150941Ssklower if (len >= MINCLSIZE) { 60250941Ssklower MCLGET(m, M_DONTWAIT); 60350941Ssklower if (m->m_flags & M_EXT) 60450941Ssklower m->m_len = len = min(len, MCLBYTES); 60550941Ssklower else 60650941Ssklower len = m->m_len; 60750941Ssklower } else { 60850941Ssklower /* 60950941Ssklower * Place initial small packet/header at end of mbuf. 61050941Ssklower */ 61150941Ssklower if (len < m->m_len) { 61250941Ssklower if (top == 0 && len + max_linkhdr <= m->m_len) 61350941Ssklower m->m_data += max_linkhdr; 61450941Ssklower m->m_len = len; 61550941Ssklower } else 61650941Ssklower len = m->m_len; 61750941Ssklower } 61850941Ssklower if (copy) 61950941Ssklower copy(cp, mtod(m, caddr_t), (unsigned)len); 62050941Ssklower else 62150941Ssklower bcopy(cp, mtod(m, caddr_t), (unsigned)len); 62250941Ssklower cp += len; 62350941Ssklower *mp = m; 62450941Ssklower mp = &m->m_next; 62550941Ssklower totlen -= len; 62650941Ssklower if (cp == epkt) 62750941Ssklower cp = buf; 62850941Ssklower } 62950941Ssklower return (top); 63050941Ssklower } 631