149ed6e97SBjoern A. Zeeb /*- 243ffd0f5SBjoern A. Zeeb * Copyright (c) 2020-2025 The FreeBSD Foundation 36baea331SBjoern A. Zeeb * Copyright (c) 2021-2022 Bjoern A. Zeeb 449ed6e97SBjoern A. Zeeb * 549ed6e97SBjoern A. Zeeb * This software was developed by Björn Zeeb under sponsorship from 649ed6e97SBjoern A. Zeeb * the FreeBSD Foundation. 749ed6e97SBjoern A. Zeeb * 849ed6e97SBjoern A. Zeeb * Redistribution and use in source and binary forms, with or without 949ed6e97SBjoern A. Zeeb * modification, are permitted provided that the following conditions 1049ed6e97SBjoern A. Zeeb * are met: 1149ed6e97SBjoern A. Zeeb * 1. Redistributions of source code must retain the above copyright 1249ed6e97SBjoern A. Zeeb * notice, this list of conditions and the following disclaimer. 1349ed6e97SBjoern A. Zeeb * 2. Redistributions in binary form must reproduce the above copyright 1449ed6e97SBjoern A. Zeeb * notice, this list of conditions and the following disclaimer in the 1549ed6e97SBjoern A. Zeeb * documentation and/or other materials provided with the distribution. 1649ed6e97SBjoern A. Zeeb * 1749ed6e97SBjoern A. Zeeb * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 1849ed6e97SBjoern A. Zeeb * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 1949ed6e97SBjoern A. Zeeb * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 2049ed6e97SBjoern A. Zeeb * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 2149ed6e97SBjoern A. Zeeb * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 2249ed6e97SBjoern A. Zeeb * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 2349ed6e97SBjoern A. Zeeb * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2449ed6e97SBjoern A. Zeeb * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2549ed6e97SBjoern A. Zeeb * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2649ed6e97SBjoern A. Zeeb * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 2749ed6e97SBjoern A. Zeeb * SUCH DAMAGE. 2849ed6e97SBjoern A. Zeeb */ 2949ed6e97SBjoern A. Zeeb 3049ed6e97SBjoern A. Zeeb /* 3149ed6e97SBjoern A. Zeeb * NOTE: this socket buffer compatibility code is highly EXPERIMENTAL. 3249ed6e97SBjoern A. Zeeb * Do not rely on the internals of this implementation. They are highly 3349ed6e97SBjoern A. Zeeb * likely to change as we will improve the integration to FreeBSD mbufs. 3449ed6e97SBjoern A. Zeeb */ 3549ed6e97SBjoern A. Zeeb 3649ed6e97SBjoern A. Zeeb #include <sys/cdefs.h> 376baea331SBjoern A. Zeeb #include "opt_ddb.h" 386baea331SBjoern A. Zeeb 3949ed6e97SBjoern A. Zeeb #include <sys/param.h> 4049ed6e97SBjoern A. Zeeb #include <sys/types.h> 4149ed6e97SBjoern A. Zeeb #include <sys/kernel.h> 4249ed6e97SBjoern A. Zeeb #include <sys/malloc.h> 436baea331SBjoern A. Zeeb #include <sys/sysctl.h> 446baea331SBjoern A. Zeeb 4543ffd0f5SBjoern A. Zeeb #include <vm/uma.h> 4643ffd0f5SBjoern A. Zeeb 476baea331SBjoern A. Zeeb #ifdef DDB 486baea331SBjoern A. Zeeb #include <ddb/ddb.h> 496baea331SBjoern A. Zeeb #endif 5049ed6e97SBjoern A. Zeeb 5149ed6e97SBjoern A. Zeeb #include <linux/skbuff.h> 5249ed6e97SBjoern A. Zeeb #include <linux/slab.h> 536baea331SBjoern A. Zeeb #include <linux/gfp.h> 546a501570SBjoern A. Zeeb #ifdef __LP64__ 556a501570SBjoern A. Zeeb #include <linux/log2.h> 566a501570SBjoern A. Zeeb #endif 576baea331SBjoern A. Zeeb 586baea331SBjoern A. Zeeb SYSCTL_DECL(_compat_linuxkpi); 596baea331SBjoern A. Zeeb SYSCTL_NODE(_compat_linuxkpi, OID_AUTO, skb, CTLFLAG_RW | CTLFLAG_MPSAFE, 0, 606baea331SBjoern A. Zeeb "LinuxKPI skbuff"); 616baea331SBjoern A. Zeeb 626a501570SBjoern A. Zeeb #ifdef SKB_DEBUG 636baea331SBjoern A. Zeeb int linuxkpi_debug_skb; 646baea331SBjoern A. Zeeb SYSCTL_INT(_compat_linuxkpi_skb, OID_AUTO, debug, CTLFLAG_RWTUN, 656baea331SBjoern A. Zeeb &linuxkpi_debug_skb, 0, "SKB debug level"); 666baea331SBjoern A. Zeeb #endif 6749ed6e97SBjoern A. Zeeb 6843ffd0f5SBjoern A. Zeeb static uma_zone_t skbzone; 6943ffd0f5SBjoern A. Zeeb 70*e6c91f8eSBjoern A. Zeeb #define SKB_DMA32_MALLOC 71*e6c91f8eSBjoern A. Zeeb #ifdef SKB_DMA32_MALLOC 726a501570SBjoern A. Zeeb /* 736a501570SBjoern A. Zeeb * Realtek wireless drivers (e.g., rtw88) require 32bit DMA in a single segment. 746a501570SBjoern A. Zeeb * busdma(9) has a hard time providing this currently for 3-ish pages at large 756a501570SBjoern A. Zeeb * quantities (see lkpi_pci_nseg1_fail in linux_pci.c). 766a501570SBjoern A. Zeeb * Work around this for now by allowing a tunable to enforce physical addresses 77*e6c91f8eSBjoern A. Zeeb * allocation limits using "old-school" contigmalloc(9) to avoid bouncing. 78*e6c91f8eSBjoern A. Zeeb * Note: with the malloc/contigmalloc + kmalloc changes also providing physical 79*e6c91f8eSBjoern A. Zeeb * contiguous memory, and the nseg=1 limit for bouncing we should in theory be 80*e6c91f8eSBjoern A. Zeeb * fine now and not need any of this anymore, however busdma still has troubles 81*e6c91f8eSBjoern A. Zeeb * boncing three contiguous pages so for now this stays. 826a501570SBjoern A. Zeeb */ 836a501570SBjoern A. Zeeb static int linuxkpi_skb_memlimit; 846a501570SBjoern A. Zeeb SYSCTL_INT(_compat_linuxkpi_skb, OID_AUTO, mem_limit, CTLFLAG_RDTUN, 856a501570SBjoern A. Zeeb &linuxkpi_skb_memlimit, 0, "SKB memory limit: 0=no limit, " 866a501570SBjoern A. Zeeb "1=32bit, 2=36bit, other=undef (currently 32bit)"); 876a501570SBjoern A. Zeeb 8849ed6e97SBjoern A. Zeeb static MALLOC_DEFINE(M_LKPISKB, "lkpiskb", "Linux KPI skbuff compat"); 89*e6c91f8eSBjoern A. Zeeb #endif 9049ed6e97SBjoern A. Zeeb 9149ed6e97SBjoern A. Zeeb struct sk_buff * 9249ed6e97SBjoern A. Zeeb linuxkpi_alloc_skb(size_t size, gfp_t gfp) 9349ed6e97SBjoern A. Zeeb { 9449ed6e97SBjoern A. Zeeb struct sk_buff *skb; 9543ffd0f5SBjoern A. Zeeb void *p; 9649ed6e97SBjoern A. Zeeb size_t len; 9749ed6e97SBjoern A. Zeeb 9843ffd0f5SBjoern A. Zeeb skb = uma_zalloc(skbzone, linux_check_m_flags(gfp) | M_ZERO); 9943ffd0f5SBjoern A. Zeeb if (skb == NULL) 10043ffd0f5SBjoern A. Zeeb return (NULL); 10143ffd0f5SBjoern A. Zeeb 10243ffd0f5SBjoern A. Zeeb skb->prev = skb->next = skb; 10343ffd0f5SBjoern A. Zeeb skb->truesize = size; 10443ffd0f5SBjoern A. Zeeb skb->shinfo = (struct skb_shared_info *)(skb + 1); 10543ffd0f5SBjoern A. Zeeb 10643ffd0f5SBjoern A. Zeeb if (size == 0) 10743ffd0f5SBjoern A. Zeeb return (skb); 10843ffd0f5SBjoern A. Zeeb 10943ffd0f5SBjoern A. Zeeb len = size; 110*e6c91f8eSBjoern A. Zeeb #ifdef SKB_DMA32_MALLOC 11149ed6e97SBjoern A. Zeeb /* 1126baea331SBjoern A. Zeeb * Using our own type here not backing my kmalloc. 11349ed6e97SBjoern A. Zeeb * We assume no one calls kfree directly on the skb. 11449ed6e97SBjoern A. Zeeb */ 115*e6c91f8eSBjoern A. Zeeb if (__predict_false(linuxkpi_skb_memlimit != 0)) { 1166a501570SBjoern A. Zeeb vm_paddr_t high; 1176a501570SBjoern A. Zeeb 1186a501570SBjoern A. Zeeb switch (linuxkpi_skb_memlimit) { 1196a501570SBjoern A. Zeeb case 2: 1206a501570SBjoern A. Zeeb high = (0xfffffffff); /* 1<<36 really. */ 1216a501570SBjoern A. Zeeb break; 1226a501570SBjoern A. Zeeb case 1: 1236a501570SBjoern A. Zeeb default: 1246a501570SBjoern A. Zeeb high = (0xffffffff); /* 1<<32 really. */ 1256a501570SBjoern A. Zeeb break; 1266a501570SBjoern A. Zeeb } 1276a501570SBjoern A. Zeeb len = roundup_pow_of_two(len); 12843ffd0f5SBjoern A. Zeeb p = contigmalloc(len, M_LKPISKB, 1296a501570SBjoern A. Zeeb linux_check_m_flags(gfp) | M_ZERO, 0, high, PAGE_SIZE, 0); 130*e6c91f8eSBjoern A. Zeeb } else 1316a501570SBjoern A. Zeeb #endif 132*e6c91f8eSBjoern A. Zeeb p = __kmalloc(len, linux_check_m_flags(gfp) | M_ZERO); 13343ffd0f5SBjoern A. Zeeb if (p == NULL) { 13443ffd0f5SBjoern A. Zeeb uma_zfree(skbzone, skb); 13543ffd0f5SBjoern A. Zeeb return (NULL); 13643ffd0f5SBjoern A. Zeeb } 13749ed6e97SBjoern A. Zeeb 13843ffd0f5SBjoern A. Zeeb skb->head = skb->data = (uint8_t *)p; 13943ffd0f5SBjoern A. Zeeb skb_reset_tail_pointer(skb); 14049ed6e97SBjoern A. Zeeb skb->end = skb->head + size; 14149ed6e97SBjoern A. Zeeb 1429df5f29cSBjoern A. Zeeb SKB_TRACE_FMT(skb, "data %p size %zu", (skb) ? skb->data : NULL, size); 1439df5f29cSBjoern A. Zeeb return (skb); 1449df5f29cSBjoern A. Zeeb } 1459df5f29cSBjoern A. Zeeb 1469df5f29cSBjoern A. Zeeb struct sk_buff * 1479df5f29cSBjoern A. Zeeb linuxkpi_dev_alloc_skb(size_t size, gfp_t gfp) 1489df5f29cSBjoern A. Zeeb { 1499df5f29cSBjoern A. Zeeb struct sk_buff *skb; 1509df5f29cSBjoern A. Zeeb size_t len; 1519df5f29cSBjoern A. Zeeb 1529df5f29cSBjoern A. Zeeb len = size + NET_SKB_PAD; 1539df5f29cSBjoern A. Zeeb skb = linuxkpi_alloc_skb(len, gfp); 1549df5f29cSBjoern A. Zeeb 1559df5f29cSBjoern A. Zeeb if (skb != NULL) 1569df5f29cSBjoern A. Zeeb skb_reserve(skb, NET_SKB_PAD); 1579df5f29cSBjoern A. Zeeb 1589df5f29cSBjoern A. Zeeb SKB_TRACE_FMT(skb, "data %p size %zu len %zu", 1599df5f29cSBjoern A. Zeeb (skb) ? skb->data : NULL, size, len); 16049ed6e97SBjoern A. Zeeb return (skb); 16149ed6e97SBjoern A. Zeeb } 16249ed6e97SBjoern A. Zeeb 163349b042bSBjoern A. Zeeb struct sk_buff * 1645504bd59SBjoern A. Zeeb linuxkpi_build_skb(void *data, size_t fragsz) 1655504bd59SBjoern A. Zeeb { 1665504bd59SBjoern A. Zeeb struct sk_buff *skb; 1675504bd59SBjoern A. Zeeb 1685504bd59SBjoern A. Zeeb if (data == NULL || fragsz == 0) 1695504bd59SBjoern A. Zeeb return (NULL); 1705504bd59SBjoern A. Zeeb 1715504bd59SBjoern A. Zeeb /* Just allocate a skb without data area. */ 1725504bd59SBjoern A. Zeeb skb = linuxkpi_alloc_skb(0, GFP_KERNEL); 1735504bd59SBjoern A. Zeeb if (skb == NULL) 1745504bd59SBjoern A. Zeeb return (NULL); 1755504bd59SBjoern A. Zeeb 1765504bd59SBjoern A. Zeeb skb->_flags |= _SKB_FLAGS_SKBEXTFRAG; 1775504bd59SBjoern A. Zeeb skb->truesize = fragsz; 1785504bd59SBjoern A. Zeeb skb->head = skb->data = data; 17943ffd0f5SBjoern A. Zeeb skb_reset_tail_pointer(skb); 18043ffd0f5SBjoern A. Zeeb skb->end = skb->head + fragsz; 1815504bd59SBjoern A. Zeeb 1825504bd59SBjoern A. Zeeb return (skb); 1835504bd59SBjoern A. Zeeb } 1845504bd59SBjoern A. Zeeb 1855504bd59SBjoern A. Zeeb struct sk_buff * 1862ab4a419SBjoern A. Zeeb linuxkpi_skb_copy(const struct sk_buff *skb, gfp_t gfp) 187349b042bSBjoern A. Zeeb { 188349b042bSBjoern A. Zeeb struct sk_buff *new; 189349b042bSBjoern A. Zeeb struct skb_shared_info *shinfo; 190349b042bSBjoern A. Zeeb size_t len; 191349b042bSBjoern A. Zeeb unsigned int headroom; 192349b042bSBjoern A. Zeeb 193349b042bSBjoern A. Zeeb /* Full buffer size + any fragments. */ 194349b042bSBjoern A. Zeeb len = skb->end - skb->head + skb->data_len; 195349b042bSBjoern A. Zeeb 196349b042bSBjoern A. Zeeb new = linuxkpi_alloc_skb(len, gfp); 197349b042bSBjoern A. Zeeb if (new == NULL) 198349b042bSBjoern A. Zeeb return (NULL); 199349b042bSBjoern A. Zeeb 200349b042bSBjoern A. Zeeb headroom = skb_headroom(skb); 201349b042bSBjoern A. Zeeb /* Fixup head and end. */ 202349b042bSBjoern A. Zeeb skb_reserve(new, headroom); /* data and tail move headroom forward. */ 203349b042bSBjoern A. Zeeb skb_put(new, skb->len); /* tail and len get adjusted */ 204349b042bSBjoern A. Zeeb 205349b042bSBjoern A. Zeeb /* Copy data. */ 206349b042bSBjoern A. Zeeb memcpy(new->head, skb->data - headroom, headroom + skb->len); 207349b042bSBjoern A. Zeeb 208349b042bSBjoern A. Zeeb /* Deal with fragments. */ 209349b042bSBjoern A. Zeeb shinfo = skb->shinfo; 210349b042bSBjoern A. Zeeb if (shinfo->nr_frags > 0) { 211349b042bSBjoern A. Zeeb printf("%s:%d: NOT YET SUPPORTED; missing %d frags\n", 212349b042bSBjoern A. Zeeb __func__, __LINE__, shinfo->nr_frags); 213349b042bSBjoern A. Zeeb SKB_TODO(); 214349b042bSBjoern A. Zeeb } 215349b042bSBjoern A. Zeeb 216349b042bSBjoern A. Zeeb /* Deal with header fields. */ 217349b042bSBjoern A. Zeeb memcpy(new->cb, skb->cb, sizeof(skb->cb)); 218349b042bSBjoern A. Zeeb SKB_IMPROVE("more header fields to copy?"); 219349b042bSBjoern A. Zeeb 220349b042bSBjoern A. Zeeb return (new); 221349b042bSBjoern A. Zeeb } 222349b042bSBjoern A. Zeeb 22349ed6e97SBjoern A. Zeeb void 22449ed6e97SBjoern A. Zeeb linuxkpi_kfree_skb(struct sk_buff *skb) 22549ed6e97SBjoern A. Zeeb { 22649ed6e97SBjoern A. Zeeb struct skb_shared_info *shinfo; 2276baea331SBjoern A. Zeeb uint16_t fragno, count; 22849ed6e97SBjoern A. Zeeb 22949ed6e97SBjoern A. Zeeb SKB_TRACE(skb); 23049ed6e97SBjoern A. Zeeb if (skb == NULL) 23149ed6e97SBjoern A. Zeeb return; 23249ed6e97SBjoern A. Zeeb 23349ed6e97SBjoern A. Zeeb /* 23449ed6e97SBjoern A. Zeeb * XXX TODO this will go away once we have skb backed by mbuf. 23549ed6e97SBjoern A. Zeeb * currently we allow the mbuf to stay around and use a private 23649ed6e97SBjoern A. Zeeb * free function to allow secondary resources to be freed along. 23749ed6e97SBjoern A. Zeeb */ 23849ed6e97SBjoern A. Zeeb if (skb->m != NULL) { 23949ed6e97SBjoern A. Zeeb void *m; 24049ed6e97SBjoern A. Zeeb 24149ed6e97SBjoern A. Zeeb m = skb->m; 24249ed6e97SBjoern A. Zeeb skb->m = NULL; 24349ed6e97SBjoern A. Zeeb 24449ed6e97SBjoern A. Zeeb KASSERT(skb->m_free_func != NULL, ("%s: skb %p has m %p but no " 24549ed6e97SBjoern A. Zeeb "m_free_func %p\n", __func__, skb, m, skb->m_free_func)); 24649ed6e97SBjoern A. Zeeb skb->m_free_func(m); 24749ed6e97SBjoern A. Zeeb } 24849ed6e97SBjoern A. Zeeb KASSERT(skb->m == NULL, 24949ed6e97SBjoern A. Zeeb ("%s: skb %p m %p != NULL\n", __func__, skb, skb->m)); 25049ed6e97SBjoern A. Zeeb 25149ed6e97SBjoern A. Zeeb shinfo = skb->shinfo; 2526baea331SBjoern A. Zeeb for (count = fragno = 0; 2536baea331SBjoern A. Zeeb count < shinfo->nr_frags && fragno < nitems(shinfo->frags); 2546baea331SBjoern A. Zeeb fragno++) { 25549ed6e97SBjoern A. Zeeb 2566baea331SBjoern A. Zeeb if (shinfo->frags[fragno].page != NULL) { 2576baea331SBjoern A. Zeeb struct page *p; 2586baea331SBjoern A. Zeeb 2596baea331SBjoern A. Zeeb p = shinfo->frags[fragno].page; 2606baea331SBjoern A. Zeeb shinfo->frags[fragno].size = 0; 2616baea331SBjoern A. Zeeb shinfo->frags[fragno].offset = 0; 2626baea331SBjoern A. Zeeb shinfo->frags[fragno].page = NULL; 2636baea331SBjoern A. Zeeb __free_page(p); 2646baea331SBjoern A. Zeeb count++; 2656baea331SBjoern A. Zeeb } 26649ed6e97SBjoern A. Zeeb } 26749ed6e97SBjoern A. Zeeb 2685504bd59SBjoern A. Zeeb if ((skb->_flags & _SKB_FLAGS_SKBEXTFRAG) != 0) { 2695504bd59SBjoern A. Zeeb void *p; 2705504bd59SBjoern A. Zeeb 2715504bd59SBjoern A. Zeeb p = skb->head; 2725504bd59SBjoern A. Zeeb skb_free_frag(p); 27343ffd0f5SBjoern A. Zeeb skb->head = NULL; 2745504bd59SBjoern A. Zeeb } 2755504bd59SBjoern A. Zeeb 276*e6c91f8eSBjoern A. Zeeb #ifdef SKB_DMA32_MALLOC 277*e6c91f8eSBjoern A. Zeeb if (__predict_false(linuxkpi_skb_memlimit != 0)) 27843ffd0f5SBjoern A. Zeeb free(skb->head, M_LKPISKB); 279*e6c91f8eSBjoern A. Zeeb else 280*e6c91f8eSBjoern A. Zeeb #endif 281*e6c91f8eSBjoern A. Zeeb kfree(skb->head); 28243ffd0f5SBjoern A. Zeeb uma_zfree(skbzone, skb); 28349ed6e97SBjoern A. Zeeb } 2846baea331SBjoern A. Zeeb 28543ffd0f5SBjoern A. Zeeb static void 28643ffd0f5SBjoern A. Zeeb lkpi_skbuff_init(void *arg __unused) 28743ffd0f5SBjoern A. Zeeb { 28843ffd0f5SBjoern A. Zeeb skbzone = uma_zcreate("skbuff", 28943ffd0f5SBjoern A. Zeeb sizeof(struct sk_buff) + sizeof(struct skb_shared_info), 29043ffd0f5SBjoern A. Zeeb NULL, NULL, NULL, NULL, UMA_ALIGN_PTR, 0); 29143ffd0f5SBjoern A. Zeeb /* Do we need to apply limits? */ 29243ffd0f5SBjoern A. Zeeb } 29343ffd0f5SBjoern A. Zeeb SYSINIT(linuxkpi_skbuff, SI_SUB_DRIVERS, SI_ORDER_FIRST, lkpi_skbuff_init, NULL); 29443ffd0f5SBjoern A. Zeeb 29543ffd0f5SBjoern A. Zeeb static void 29643ffd0f5SBjoern A. Zeeb lkpi_skbuff_destroy(void *arg __unused) 29743ffd0f5SBjoern A. Zeeb { 29843ffd0f5SBjoern A. Zeeb uma_zdestroy(skbzone); 29943ffd0f5SBjoern A. Zeeb } 30043ffd0f5SBjoern A. Zeeb SYSUNINIT(linuxkpi_skbuff, SI_SUB_DRIVERS, SI_ORDER_SECOND, lkpi_skbuff_destroy, NULL); 30143ffd0f5SBjoern A. Zeeb 3026baea331SBjoern A. Zeeb #ifdef DDB 3036baea331SBjoern A. Zeeb DB_SHOW_COMMAND(skb, db_show_skb) 3046baea331SBjoern A. Zeeb { 3056baea331SBjoern A. Zeeb struct sk_buff *skb; 3066baea331SBjoern A. Zeeb int i; 3076baea331SBjoern A. Zeeb 3086baea331SBjoern A. Zeeb if (!have_addr) { 3096baea331SBjoern A. Zeeb db_printf("usage: show skb <addr>\n"); 3106baea331SBjoern A. Zeeb return; 3116baea331SBjoern A. Zeeb } 3126baea331SBjoern A. Zeeb 3136baea331SBjoern A. Zeeb skb = (struct sk_buff *)addr; 3146baea331SBjoern A. Zeeb 3156baea331SBjoern A. Zeeb db_printf("skb %p\n", skb); 3166baea331SBjoern A. Zeeb db_printf("\tnext %p prev %p\n", skb->next, skb->prev); 31789c32dafSBjoern A. Zeeb db_printf("\tlist %p\n", &skb->list); 31873e3969fSBjoern A. Zeeb db_printf("\tlen %u data_len %u truesize %u mac_len %u\n", 31973e3969fSBjoern A. Zeeb skb->len, skb->data_len, skb->truesize, skb->mac_len); 3206baea331SBjoern A. Zeeb db_printf("\tcsum %#06x l3hdroff %u l4hdroff %u priority %u qmap %u\n", 3216baea331SBjoern A. Zeeb skb->csum, skb->l3hdroff, skb->l4hdroff, skb->priority, skb->qmap); 3226baea331SBjoern A. Zeeb db_printf("\tpkt_type %d dev %p sk %p\n", 3236baea331SBjoern A. Zeeb skb->pkt_type, skb->dev, skb->sk); 3246baea331SBjoern A. Zeeb db_printf("\tcsum_offset %d csum_start %d ip_summed %d protocol %d\n", 3256baea331SBjoern A. Zeeb skb->csum_offset, skb->csum_start, skb->ip_summed, skb->protocol); 3265504bd59SBjoern A. Zeeb db_printf("\t_flags %#06x\n", skb->_flags); /* XXX-BZ print names? */ 3276baea331SBjoern A. Zeeb db_printf("\thead %p data %p tail %p end %p\n", 3286baea331SBjoern A. Zeeb skb->head, skb->data, skb->tail, skb->end); 3296baea331SBjoern A. Zeeb db_printf("\tshinfo %p m %p m_free_func %p\n", 3306baea331SBjoern A. Zeeb skb->shinfo, skb->m, skb->m_free_func); 3316baea331SBjoern A. Zeeb 3326baea331SBjoern A. Zeeb if (skb->shinfo != NULL) { 3336baea331SBjoern A. Zeeb struct skb_shared_info *shinfo; 3346baea331SBjoern A. Zeeb 3356baea331SBjoern A. Zeeb shinfo = skb->shinfo; 3366baea331SBjoern A. Zeeb db_printf("\t\tgso_type %d gso_size %u nr_frags %u\n", 3376baea331SBjoern A. Zeeb shinfo->gso_type, shinfo->gso_size, shinfo->nr_frags); 3386baea331SBjoern A. Zeeb for (i = 0; i < nitems(shinfo->frags); i++) { 3396baea331SBjoern A. Zeeb struct skb_frag *frag; 3406baea331SBjoern A. Zeeb 3416baea331SBjoern A. Zeeb frag = &shinfo->frags[i]; 3426baea331SBjoern A. Zeeb if (frag == NULL || frag->page == NULL) 3436baea331SBjoern A. Zeeb continue; 3446baea331SBjoern A. Zeeb db_printf("\t\t\tfrag %p fragno %d page %p %p " 3456baea331SBjoern A. Zeeb "offset %ju size %zu\n", 3466baea331SBjoern A. Zeeb frag, i, frag->page, linux_page_address(frag->page), 3476baea331SBjoern A. Zeeb (uintmax_t)frag->offset, frag->size); 3486baea331SBjoern A. Zeeb } 3496baea331SBjoern A. Zeeb } 3506baea331SBjoern A. Zeeb db_printf("\tcb[] %p {", skb->cb); 3516baea331SBjoern A. Zeeb for (i = 0; i < nitems(skb->cb); i++) { 3526baea331SBjoern A. Zeeb db_printf("%#04x%s", 3536baea331SBjoern A. Zeeb skb->cb[i], (i < (nitems(skb->cb)-1)) ? ", " : ""); 3546baea331SBjoern A. Zeeb } 3556baea331SBjoern A. Zeeb db_printf("}\n"); 3566baea331SBjoern A. Zeeb 3575504bd59SBjoern A. Zeeb db_printf("\t__scratch[0] %p\n", skb->__scratch); 3586baea331SBjoern A. Zeeb }; 3596baea331SBjoern A. Zeeb #endif 360