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
70e6c91f8eSBjoern A. Zeeb #define SKB_DMA32_MALLOC
71e6c91f8eSBjoern 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
77e6c91f8eSBjoern A. Zeeb * allocation limits using "old-school" contigmalloc(9) to avoid bouncing.
78e6c91f8eSBjoern A. Zeeb * Note: with the malloc/contigmalloc + kmalloc changes also providing physical
79e6c91f8eSBjoern A. Zeeb * contiguous memory, and the nseg=1 limit for bouncing we should in theory be
80e6c91f8eSBjoern A. Zeeb * fine now and not need any of this anymore, however busdma still has troubles
81e6c91f8eSBjoern 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");
89e6c91f8eSBjoern A. Zeeb #endif
9049ed6e97SBjoern A. Zeeb
9149ed6e97SBjoern A. Zeeb struct sk_buff *
linuxkpi_alloc_skb(size_t size,gfp_t gfp)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;
110e6c91f8eSBjoern 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 */
115e6c91f8eSBjoern 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) {
119*cd649cfcSBjoern A. Zeeb #ifdef __LP64__
1206a501570SBjoern A. Zeeb case 2:
1216a501570SBjoern A. Zeeb high = (0xfffffffff); /* 1<<36 really. */
1226a501570SBjoern A. Zeeb break;
123*cd649cfcSBjoern A. Zeeb #endif
1246a501570SBjoern A. Zeeb case 1:
1256a501570SBjoern A. Zeeb default:
1266a501570SBjoern A. Zeeb high = (0xffffffff); /* 1<<32 really. */
1276a501570SBjoern A. Zeeb break;
1286a501570SBjoern A. Zeeb }
1296a501570SBjoern A. Zeeb len = roundup_pow_of_two(len);
13043ffd0f5SBjoern A. Zeeb p = contigmalloc(len, M_LKPISKB,
1316a501570SBjoern A. Zeeb linux_check_m_flags(gfp) | M_ZERO, 0, high, PAGE_SIZE, 0);
132e6c91f8eSBjoern A. Zeeb } else
1336a501570SBjoern A. Zeeb #endif
134e6c91f8eSBjoern A. Zeeb p = __kmalloc(len, linux_check_m_flags(gfp) | M_ZERO);
13543ffd0f5SBjoern A. Zeeb if (p == NULL) {
13643ffd0f5SBjoern A. Zeeb uma_zfree(skbzone, skb);
13743ffd0f5SBjoern A. Zeeb return (NULL);
13843ffd0f5SBjoern A. Zeeb }
13949ed6e97SBjoern A. Zeeb
14043ffd0f5SBjoern A. Zeeb skb->head = skb->data = (uint8_t *)p;
14143ffd0f5SBjoern A. Zeeb skb_reset_tail_pointer(skb);
14249ed6e97SBjoern A. Zeeb skb->end = skb->head + size;
14349ed6e97SBjoern A. Zeeb
1449df5f29cSBjoern A. Zeeb SKB_TRACE_FMT(skb, "data %p size %zu", (skb) ? skb->data : NULL, size);
1459df5f29cSBjoern A. Zeeb return (skb);
1469df5f29cSBjoern A. Zeeb }
1479df5f29cSBjoern A. Zeeb
1489df5f29cSBjoern A. Zeeb struct sk_buff *
linuxkpi_dev_alloc_skb(size_t size,gfp_t gfp)1499df5f29cSBjoern A. Zeeb linuxkpi_dev_alloc_skb(size_t size, gfp_t gfp)
1509df5f29cSBjoern A. Zeeb {
1519df5f29cSBjoern A. Zeeb struct sk_buff *skb;
1529df5f29cSBjoern A. Zeeb size_t len;
1539df5f29cSBjoern A. Zeeb
1549df5f29cSBjoern A. Zeeb len = size + NET_SKB_PAD;
1559df5f29cSBjoern A. Zeeb skb = linuxkpi_alloc_skb(len, gfp);
1569df5f29cSBjoern A. Zeeb
1579df5f29cSBjoern A. Zeeb if (skb != NULL)
1589df5f29cSBjoern A. Zeeb skb_reserve(skb, NET_SKB_PAD);
1599df5f29cSBjoern A. Zeeb
1609df5f29cSBjoern A. Zeeb SKB_TRACE_FMT(skb, "data %p size %zu len %zu",
1619df5f29cSBjoern A. Zeeb (skb) ? skb->data : NULL, size, len);
16249ed6e97SBjoern A. Zeeb return (skb);
16349ed6e97SBjoern A. Zeeb }
16449ed6e97SBjoern A. Zeeb
165349b042bSBjoern A. Zeeb struct sk_buff *
linuxkpi_build_skb(void * data,size_t fragsz)1665504bd59SBjoern A. Zeeb linuxkpi_build_skb(void *data, size_t fragsz)
1675504bd59SBjoern A. Zeeb {
1685504bd59SBjoern A. Zeeb struct sk_buff *skb;
1695504bd59SBjoern A. Zeeb
1705504bd59SBjoern A. Zeeb if (data == NULL || fragsz == 0)
1715504bd59SBjoern A. Zeeb return (NULL);
1725504bd59SBjoern A. Zeeb
1735504bd59SBjoern A. Zeeb /* Just allocate a skb without data area. */
1745504bd59SBjoern A. Zeeb skb = linuxkpi_alloc_skb(0, GFP_KERNEL);
1755504bd59SBjoern A. Zeeb if (skb == NULL)
1765504bd59SBjoern A. Zeeb return (NULL);
1775504bd59SBjoern A. Zeeb
1785504bd59SBjoern A. Zeeb skb->_flags |= _SKB_FLAGS_SKBEXTFRAG;
1795504bd59SBjoern A. Zeeb skb->truesize = fragsz;
1805504bd59SBjoern A. Zeeb skb->head = skb->data = data;
18143ffd0f5SBjoern A. Zeeb skb_reset_tail_pointer(skb);
18243ffd0f5SBjoern A. Zeeb skb->end = skb->head + fragsz;
1835504bd59SBjoern A. Zeeb
1845504bd59SBjoern A. Zeeb return (skb);
1855504bd59SBjoern A. Zeeb }
1865504bd59SBjoern A. Zeeb
1875504bd59SBjoern A. Zeeb struct sk_buff *
linuxkpi_skb_copy(const struct sk_buff * skb,gfp_t gfp)1882ab4a419SBjoern A. Zeeb linuxkpi_skb_copy(const struct sk_buff *skb, gfp_t gfp)
189349b042bSBjoern A. Zeeb {
190349b042bSBjoern A. Zeeb struct sk_buff *new;
191349b042bSBjoern A. Zeeb struct skb_shared_info *shinfo;
192349b042bSBjoern A. Zeeb size_t len;
193349b042bSBjoern A. Zeeb unsigned int headroom;
194349b042bSBjoern A. Zeeb
195349b042bSBjoern A. Zeeb /* Full buffer size + any fragments. */
196349b042bSBjoern A. Zeeb len = skb->end - skb->head + skb->data_len;
197349b042bSBjoern A. Zeeb
198349b042bSBjoern A. Zeeb new = linuxkpi_alloc_skb(len, gfp);
199349b042bSBjoern A. Zeeb if (new == NULL)
200349b042bSBjoern A. Zeeb return (NULL);
201349b042bSBjoern A. Zeeb
202349b042bSBjoern A. Zeeb headroom = skb_headroom(skb);
203349b042bSBjoern A. Zeeb /* Fixup head and end. */
204349b042bSBjoern A. Zeeb skb_reserve(new, headroom); /* data and tail move headroom forward. */
205349b042bSBjoern A. Zeeb skb_put(new, skb->len); /* tail and len get adjusted */
206349b042bSBjoern A. Zeeb
207349b042bSBjoern A. Zeeb /* Copy data. */
208349b042bSBjoern A. Zeeb memcpy(new->head, skb->data - headroom, headroom + skb->len);
209349b042bSBjoern A. Zeeb
210349b042bSBjoern A. Zeeb /* Deal with fragments. */
211349b042bSBjoern A. Zeeb shinfo = skb->shinfo;
212349b042bSBjoern A. Zeeb if (shinfo->nr_frags > 0) {
213349b042bSBjoern A. Zeeb printf("%s:%d: NOT YET SUPPORTED; missing %d frags\n",
214349b042bSBjoern A. Zeeb __func__, __LINE__, shinfo->nr_frags);
215349b042bSBjoern A. Zeeb SKB_TODO();
216349b042bSBjoern A. Zeeb }
217349b042bSBjoern A. Zeeb
218349b042bSBjoern A. Zeeb /* Deal with header fields. */
219349b042bSBjoern A. Zeeb memcpy(new->cb, skb->cb, sizeof(skb->cb));
220349b042bSBjoern A. Zeeb SKB_IMPROVE("more header fields to copy?");
221349b042bSBjoern A. Zeeb
222349b042bSBjoern A. Zeeb return (new);
223349b042bSBjoern A. Zeeb }
224349b042bSBjoern A. Zeeb
22549ed6e97SBjoern A. Zeeb void
linuxkpi_kfree_skb(struct sk_buff * skb)22649ed6e97SBjoern A. Zeeb linuxkpi_kfree_skb(struct sk_buff *skb)
22749ed6e97SBjoern A. Zeeb {
22849ed6e97SBjoern A. Zeeb struct skb_shared_info *shinfo;
2296baea331SBjoern A. Zeeb uint16_t fragno, count;
23049ed6e97SBjoern A. Zeeb
23149ed6e97SBjoern A. Zeeb SKB_TRACE(skb);
23249ed6e97SBjoern A. Zeeb if (skb == NULL)
23349ed6e97SBjoern A. Zeeb return;
23449ed6e97SBjoern A. Zeeb
23549ed6e97SBjoern A. Zeeb /*
23649ed6e97SBjoern A. Zeeb * XXX TODO this will go away once we have skb backed by mbuf.
23749ed6e97SBjoern A. Zeeb * currently we allow the mbuf to stay around and use a private
23849ed6e97SBjoern A. Zeeb * free function to allow secondary resources to be freed along.
23949ed6e97SBjoern A. Zeeb */
24049ed6e97SBjoern A. Zeeb if (skb->m != NULL) {
24149ed6e97SBjoern A. Zeeb void *m;
24249ed6e97SBjoern A. Zeeb
24349ed6e97SBjoern A. Zeeb m = skb->m;
24449ed6e97SBjoern A. Zeeb skb->m = NULL;
24549ed6e97SBjoern A. Zeeb
24649ed6e97SBjoern A. Zeeb KASSERT(skb->m_free_func != NULL, ("%s: skb %p has m %p but no "
24749ed6e97SBjoern A. Zeeb "m_free_func %p\n", __func__, skb, m, skb->m_free_func));
24849ed6e97SBjoern A. Zeeb skb->m_free_func(m);
24949ed6e97SBjoern A. Zeeb }
25049ed6e97SBjoern A. Zeeb KASSERT(skb->m == NULL,
25149ed6e97SBjoern A. Zeeb ("%s: skb %p m %p != NULL\n", __func__, skb, skb->m));
25249ed6e97SBjoern A. Zeeb
25349ed6e97SBjoern A. Zeeb shinfo = skb->shinfo;
2546baea331SBjoern A. Zeeb for (count = fragno = 0;
2556baea331SBjoern A. Zeeb count < shinfo->nr_frags && fragno < nitems(shinfo->frags);
2566baea331SBjoern A. Zeeb fragno++) {
25749ed6e97SBjoern A. Zeeb
2586baea331SBjoern A. Zeeb if (shinfo->frags[fragno].page != NULL) {
2596baea331SBjoern A. Zeeb struct page *p;
2606baea331SBjoern A. Zeeb
2616baea331SBjoern A. Zeeb p = shinfo->frags[fragno].page;
2626baea331SBjoern A. Zeeb shinfo->frags[fragno].size = 0;
2636baea331SBjoern A. Zeeb shinfo->frags[fragno].offset = 0;
2646baea331SBjoern A. Zeeb shinfo->frags[fragno].page = NULL;
2656baea331SBjoern A. Zeeb __free_page(p);
2666baea331SBjoern A. Zeeb count++;
2676baea331SBjoern A. Zeeb }
26849ed6e97SBjoern A. Zeeb }
26949ed6e97SBjoern A. Zeeb
2705504bd59SBjoern A. Zeeb if ((skb->_flags & _SKB_FLAGS_SKBEXTFRAG) != 0) {
2715504bd59SBjoern A. Zeeb void *p;
2725504bd59SBjoern A. Zeeb
2735504bd59SBjoern A. Zeeb p = skb->head;
2745504bd59SBjoern A. Zeeb skb_free_frag(p);
27543ffd0f5SBjoern A. Zeeb skb->head = NULL;
2765504bd59SBjoern A. Zeeb }
2775504bd59SBjoern A. Zeeb
278e6c91f8eSBjoern A. Zeeb #ifdef SKB_DMA32_MALLOC
279e6c91f8eSBjoern A. Zeeb if (__predict_false(linuxkpi_skb_memlimit != 0))
28043ffd0f5SBjoern A. Zeeb free(skb->head, M_LKPISKB);
281e6c91f8eSBjoern A. Zeeb else
282e6c91f8eSBjoern A. Zeeb #endif
283e6c91f8eSBjoern A. Zeeb kfree(skb->head);
28443ffd0f5SBjoern A. Zeeb uma_zfree(skbzone, skb);
28549ed6e97SBjoern A. Zeeb }
2866baea331SBjoern A. Zeeb
28743ffd0f5SBjoern A. Zeeb static void
lkpi_skbuff_init(void * arg __unused)28843ffd0f5SBjoern A. Zeeb lkpi_skbuff_init(void *arg __unused)
28943ffd0f5SBjoern A. Zeeb {
29043ffd0f5SBjoern A. Zeeb skbzone = uma_zcreate("skbuff",
29143ffd0f5SBjoern A. Zeeb sizeof(struct sk_buff) + sizeof(struct skb_shared_info),
29243ffd0f5SBjoern A. Zeeb NULL, NULL, NULL, NULL, UMA_ALIGN_PTR, 0);
29343ffd0f5SBjoern A. Zeeb /* Do we need to apply limits? */
29443ffd0f5SBjoern A. Zeeb }
29543ffd0f5SBjoern A. Zeeb SYSINIT(linuxkpi_skbuff, SI_SUB_DRIVERS, SI_ORDER_FIRST, lkpi_skbuff_init, NULL);
29643ffd0f5SBjoern A. Zeeb
29743ffd0f5SBjoern A. Zeeb static void
lkpi_skbuff_destroy(void * arg __unused)29843ffd0f5SBjoern A. Zeeb lkpi_skbuff_destroy(void *arg __unused)
29943ffd0f5SBjoern A. Zeeb {
30043ffd0f5SBjoern A. Zeeb uma_zdestroy(skbzone);
30143ffd0f5SBjoern A. Zeeb }
30243ffd0f5SBjoern A. Zeeb SYSUNINIT(linuxkpi_skbuff, SI_SUB_DRIVERS, SI_ORDER_SECOND, lkpi_skbuff_destroy, NULL);
30343ffd0f5SBjoern A. Zeeb
3046baea331SBjoern A. Zeeb #ifdef DDB
DB_SHOW_COMMAND(skb,db_show_skb)3056baea331SBjoern A. Zeeb DB_SHOW_COMMAND(skb, db_show_skb)
3066baea331SBjoern A. Zeeb {
3076baea331SBjoern A. Zeeb struct sk_buff *skb;
3086baea331SBjoern A. Zeeb int i;
3096baea331SBjoern A. Zeeb
3106baea331SBjoern A. Zeeb if (!have_addr) {
3116baea331SBjoern A. Zeeb db_printf("usage: show skb <addr>\n");
3126baea331SBjoern A. Zeeb return;
3136baea331SBjoern A. Zeeb }
3146baea331SBjoern A. Zeeb
3156baea331SBjoern A. Zeeb skb = (struct sk_buff *)addr;
3166baea331SBjoern A. Zeeb
3176baea331SBjoern A. Zeeb db_printf("skb %p\n", skb);
3186baea331SBjoern A. Zeeb db_printf("\tnext %p prev %p\n", skb->next, skb->prev);
31989c32dafSBjoern A. Zeeb db_printf("\tlist %p\n", &skb->list);
32073e3969fSBjoern A. Zeeb db_printf("\tlen %u data_len %u truesize %u mac_len %u\n",
32173e3969fSBjoern A. Zeeb skb->len, skb->data_len, skb->truesize, skb->mac_len);
3226baea331SBjoern A. Zeeb db_printf("\tcsum %#06x l3hdroff %u l4hdroff %u priority %u qmap %u\n",
3236baea331SBjoern A. Zeeb skb->csum, skb->l3hdroff, skb->l4hdroff, skb->priority, skb->qmap);
3246baea331SBjoern A. Zeeb db_printf("\tpkt_type %d dev %p sk %p\n",
3256baea331SBjoern A. Zeeb skb->pkt_type, skb->dev, skb->sk);
3266baea331SBjoern A. Zeeb db_printf("\tcsum_offset %d csum_start %d ip_summed %d protocol %d\n",
3276baea331SBjoern A. Zeeb skb->csum_offset, skb->csum_start, skb->ip_summed, skb->protocol);
3285504bd59SBjoern A. Zeeb db_printf("\t_flags %#06x\n", skb->_flags); /* XXX-BZ print names? */
3296baea331SBjoern A. Zeeb db_printf("\thead %p data %p tail %p end %p\n",
3306baea331SBjoern A. Zeeb skb->head, skb->data, skb->tail, skb->end);
3316baea331SBjoern A. Zeeb db_printf("\tshinfo %p m %p m_free_func %p\n",
3326baea331SBjoern A. Zeeb skb->shinfo, skb->m, skb->m_free_func);
3336baea331SBjoern A. Zeeb
3346baea331SBjoern A. Zeeb if (skb->shinfo != NULL) {
3356baea331SBjoern A. Zeeb struct skb_shared_info *shinfo;
3366baea331SBjoern A. Zeeb
3376baea331SBjoern A. Zeeb shinfo = skb->shinfo;
3386baea331SBjoern A. Zeeb db_printf("\t\tgso_type %d gso_size %u nr_frags %u\n",
3396baea331SBjoern A. Zeeb shinfo->gso_type, shinfo->gso_size, shinfo->nr_frags);
3406baea331SBjoern A. Zeeb for (i = 0; i < nitems(shinfo->frags); i++) {
3416baea331SBjoern A. Zeeb struct skb_frag *frag;
3426baea331SBjoern A. Zeeb
3436baea331SBjoern A. Zeeb frag = &shinfo->frags[i];
3446baea331SBjoern A. Zeeb if (frag == NULL || frag->page == NULL)
3456baea331SBjoern A. Zeeb continue;
3466baea331SBjoern A. Zeeb db_printf("\t\t\tfrag %p fragno %d page %p %p "
3476baea331SBjoern A. Zeeb "offset %ju size %zu\n",
3486baea331SBjoern A. Zeeb frag, i, frag->page, linux_page_address(frag->page),
3496baea331SBjoern A. Zeeb (uintmax_t)frag->offset, frag->size);
3506baea331SBjoern A. Zeeb }
3516baea331SBjoern A. Zeeb }
3526baea331SBjoern A. Zeeb db_printf("\tcb[] %p {", skb->cb);
3536baea331SBjoern A. Zeeb for (i = 0; i < nitems(skb->cb); i++) {
3546baea331SBjoern A. Zeeb db_printf("%#04x%s",
3556baea331SBjoern A. Zeeb skb->cb[i], (i < (nitems(skb->cb)-1)) ? ", " : "");
3566baea331SBjoern A. Zeeb }
3576baea331SBjoern A. Zeeb db_printf("}\n");
3586baea331SBjoern A. Zeeb
3595504bd59SBjoern A. Zeeb db_printf("\t__scratch[0] %p\n", skb->__scratch);
3606baea331SBjoern A. Zeeb };
3616baea331SBjoern A. Zeeb #endif
362