2023-10-09 10:41:51 -04:00
// SPDX-License-Identifier: GPL-2.0-or-later
# include <kunit/test.h>
# include <linux/skbuff.h>
static const char hdr [ ] = " abcdefgh " ;
2023-10-12 14:08:56 +02:00
# define GSO_TEST_SIZE 1000
2023-10-09 10:41:51 -04:00
static void __init_skb ( struct sk_buff * skb )
{
skb_reset_mac_header ( skb ) ;
memcpy ( skb_mac_header ( skb ) , hdr , sizeof ( hdr ) ) ;
/* skb_segment expects skb->data at start of payload */
skb_pull ( skb , sizeof ( hdr ) ) ;
skb_reset_network_header ( skb ) ;
skb_reset_transport_header ( skb ) ;
/* proto is arbitrary, as long as not ETH_P_TEB or vlan */
skb - > protocol = htons ( ETH_P_ATALK ) ;
2023-10-12 14:08:56 +02:00
skb_shinfo ( skb ) - > gso_size = GSO_TEST_SIZE ;
2023-10-09 10:41:51 -04:00
}
2023-10-09 10:41:52 -04:00
enum gso_test_nr {
GSO_TEST_LINEAR ,
GSO_TEST_NO_GSO ,
GSO_TEST_FRAGS ,
GSO_TEST_FRAGS_PURE ,
GSO_TEST_GSO_PARTIAL ,
2023-10-09 10:41:53 -04:00
GSO_TEST_FRAG_LIST ,
GSO_TEST_FRAG_LIST_PURE ,
GSO_TEST_FRAG_LIST_NON_UNIFORM ,
GSO_TEST_GSO_BY_FRAGS ,
2023-10-09 10:41:52 -04:00
} ;
struct gso_test_case {
enum gso_test_nr id ;
const char * name ;
/* input */
unsigned int linear_len ;
unsigned int nr_frags ;
const unsigned int * frags ;
2023-10-09 10:41:53 -04:00
unsigned int nr_frag_skbs ;
const unsigned int * frag_skbs ;
2023-10-09 10:41:52 -04:00
/* output as expected */
unsigned int nr_segs ;
const unsigned int * segs ;
} ;
static struct gso_test_case cases [ ] = {
{
. id = GSO_TEST_NO_GSO ,
. name = " no_gso " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE ,
2023-10-09 10:41:52 -04:00
. nr_segs = 1 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE } ,
2023-10-09 10:41:52 -04:00
} ,
{
. id = GSO_TEST_LINEAR ,
. name = " linear " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE + GSO_TEST_SIZE + 1 ,
2023-10-09 10:41:52 -04:00
. nr_segs = 3 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , 1 } ,
2023-10-09 10:41:52 -04:00
} ,
{
. id = GSO_TEST_FRAGS ,
. name = " frags " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE ,
2023-10-09 10:41:52 -04:00
. nr_frags = 2 ,
2023-10-12 14:08:56 +02:00
. frags = ( const unsigned int [ ] ) { GSO_TEST_SIZE , 1 } ,
2023-10-09 10:41:52 -04:00
. nr_segs = 3 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , 1 } ,
2023-10-09 10:41:52 -04:00
} ,
{
. id = GSO_TEST_FRAGS_PURE ,
. name = " frags_pure " ,
. nr_frags = 3 ,
2023-10-12 14:08:56 +02:00
. frags = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , 2 } ,
2023-10-09 10:41:52 -04:00
. nr_segs = 3 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , 2 } ,
2023-10-09 10:41:52 -04:00
} ,
{
. id = GSO_TEST_GSO_PARTIAL ,
. name = " gso_partial " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE ,
2023-10-09 10:41:52 -04:00
. nr_frags = 2 ,
2023-10-12 14:08:56 +02:00
. frags = ( const unsigned int [ ] ) { GSO_TEST_SIZE , 3 } ,
2023-10-09 10:41:52 -04:00
. nr_segs = 2 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { 2 * GSO_TEST_SIZE , 3 } ,
2023-10-09 10:41:52 -04:00
} ,
2023-10-09 10:41:53 -04:00
{
/* commit 89319d3801d1: frag_list on mss boundaries */
. id = GSO_TEST_FRAG_LIST ,
. name = " frag_list " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE ,
2023-10-09 10:41:53 -04:00
. nr_frag_skbs = 2 ,
2023-10-12 14:08:56 +02:00
. frag_skbs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE } ,
2023-10-09 10:41:53 -04:00
. nr_segs = 3 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , GSO_TEST_SIZE } ,
2023-10-09 10:41:53 -04:00
} ,
{
. id = GSO_TEST_FRAG_LIST_PURE ,
. name = " frag_list_pure " ,
. nr_frag_skbs = 2 ,
2023-10-12 14:08:56 +02:00
. frag_skbs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE } ,
2023-10-09 10:41:53 -04:00
. nr_segs = 2 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE } ,
2023-10-09 10:41:53 -04:00
} ,
{
/* commit 43170c4e0ba7: GRO of frag_list trains */
. id = GSO_TEST_FRAG_LIST_NON_UNIFORM ,
. name = " frag_list_non_uniform " ,
2023-10-12 14:08:56 +02:00
. linear_len = GSO_TEST_SIZE ,
2023-10-09 10:41:53 -04:00
. nr_frag_skbs = 4 ,
2023-10-12 14:08:56 +02:00
. frag_skbs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , 1 , GSO_TEST_SIZE , 2 } ,
2023-10-09 10:41:53 -04:00
. nr_segs = 4 ,
2023-10-12 14:08:56 +02:00
. segs = ( const unsigned int [ ] ) { GSO_TEST_SIZE , GSO_TEST_SIZE , GSO_TEST_SIZE , 3 } ,
2023-10-09 10:41:53 -04:00
} ,
{
/* commit 3953c46c3ac7 ("sk_buff: allow segmenting based on frag sizes") and
* commit 90017 accff61 ( " sctp: Add GSO support " )
*
* " there will be a cover skb with protocol headers and
* children ones containing the actual segments "
*/
. id = GSO_TEST_GSO_BY_FRAGS ,
. name = " gso_by_frags " ,
. nr_frag_skbs = 4 ,
. frag_skbs = ( const unsigned int [ ] ) { 100 , 200 , 300 , 400 } ,
. nr_segs = 4 ,
. segs = ( const unsigned int [ ] ) { 100 , 200 , 300 , 400 } ,
} ,
2023-10-09 10:41:52 -04:00
} ;
static void gso_test_case_to_desc ( struct gso_test_case * t , char * desc )
{
sprintf ( desc , " %s " , t - > name ) ;
}
KUNIT_ARRAY_PARAM ( gso_test , cases , gso_test_case_to_desc ) ;
2023-10-09 10:41:51 -04:00
static void gso_test_func ( struct kunit * test )
{
const int shinfo_size = SKB_DATA_ALIGN ( sizeof ( struct skb_shared_info ) ) ;
2023-10-12 14:02:37 +02:00
struct sk_buff * skb , * segs , * cur , * next , * last ;
2023-10-09 10:41:52 -04:00
const struct gso_test_case * tcase ;
netdev_features_t features ;
2023-10-09 10:41:51 -04:00
struct page * page ;
2023-10-09 10:41:52 -04:00
int i ;
tcase = test - > param_value ;
2023-10-09 10:41:51 -04:00
page = alloc_page ( GFP_KERNEL ) ;
KUNIT_ASSERT_NOT_NULL ( test , page ) ;
2023-10-09 10:41:52 -04:00
skb = build_skb ( page_address ( page ) , sizeof ( hdr ) + tcase - > linear_len + shinfo_size ) ;
2023-10-09 10:41:51 -04:00
KUNIT_ASSERT_NOT_NULL ( test , skb ) ;
2023-10-09 10:41:52 -04:00
__skb_put ( skb , sizeof ( hdr ) + tcase - > linear_len ) ;
2023-10-09 10:41:51 -04:00
__init_skb ( skb ) ;
2023-10-09 10:41:52 -04:00
if ( tcase - > nr_frags ) {
unsigned int pg_off = 0 ;
page = alloc_page ( GFP_KERNEL ) ;
KUNIT_ASSERT_NOT_NULL ( test , page ) ;
page_ref_add ( page , tcase - > nr_frags - 1 ) ;
for ( i = 0 ; i < tcase - > nr_frags ; i + + ) {
skb_fill_page_desc ( skb , i , page , pg_off , tcase - > frags [ i ] ) ;
pg_off + = tcase - > frags [ i ] ;
}
KUNIT_ASSERT_LE ( test , pg_off , PAGE_SIZE ) ;
skb - > data_len = pg_off ;
skb - > len + = skb - > data_len ;
skb - > truesize + = skb - > data_len ;
}
2023-10-09 10:41:53 -04:00
if ( tcase - > frag_skbs ) {
2023-11-10 10:36:00 -05:00
unsigned int total_size = 0 , total_true_size = 0 ;
2023-10-09 10:41:53 -04:00
struct sk_buff * frag_skb , * prev = NULL ;
for ( i = 0 ; i < tcase - > nr_frag_skbs ; i + + ) {
unsigned int frag_size ;
2023-11-10 10:36:00 -05:00
page = alloc_page ( GFP_KERNEL ) ;
KUNIT_ASSERT_NOT_NULL ( test , page ) ;
2023-10-09 10:41:53 -04:00
frag_size = tcase - > frag_skbs [ i ] ;
2023-11-10 10:36:00 -05:00
frag_skb = build_skb ( page_address ( page ) ,
2023-10-09 10:41:53 -04:00
frag_size + shinfo_size ) ;
KUNIT_ASSERT_NOT_NULL ( test , frag_skb ) ;
__skb_put ( frag_skb , frag_size ) ;
if ( prev )
prev - > next = frag_skb ;
else
skb_shinfo ( skb ) - > frag_list = frag_skb ;
prev = frag_skb ;
total_size + = frag_size ;
total_true_size + = frag_skb - > truesize ;
}
skb - > len + = total_size ;
skb - > data_len + = total_size ;
skb - > truesize + = total_true_size ;
if ( tcase - > id = = GSO_TEST_GSO_BY_FRAGS )
skb_shinfo ( skb ) - > gso_size = GSO_BY_FRAGS ;
}
2023-10-09 10:41:52 -04:00
features = NETIF_F_SG | NETIF_F_HW_CSUM ;
if ( tcase - > id = = GSO_TEST_GSO_PARTIAL )
features | = NETIF_F_GSO_PARTIAL ;
2023-10-09 10:41:53 -04:00
/* TODO: this should also work with SG,
* rather than hit BUG_ON ( i > = nfrags )
*/
if ( tcase - > id = = GSO_TEST_FRAG_LIST_NON_UNIFORM )
features & = ~ NETIF_F_SG ;
2023-10-09 10:41:52 -04:00
segs = skb_segment ( skb , features ) ;
2023-10-09 10:41:51 -04:00
if ( IS_ERR ( segs ) ) {
KUNIT_FAIL ( test , " segs error %lld " , PTR_ERR ( segs ) ) ;
goto free_gso_skb ;
} else if ( ! segs ) {
KUNIT_FAIL ( test , " no segments " ) ;
goto free_gso_skb ;
}
2023-10-12 14:02:37 +02:00
last = segs - > prev ;
for ( cur = segs , i = 0 ; cur ; cur = next , i + + ) {
next = cur - > next ;
2023-10-09 10:41:52 -04:00
KUNIT_ASSERT_EQ ( test , cur - > len , sizeof ( hdr ) + tcase - > segs [ i ] ) ;
2023-10-09 10:41:51 -04:00
/* segs have skb->data pointing to the mac header */
KUNIT_ASSERT_PTR_EQ ( test , skb_mac_header ( cur ) , cur - > data ) ;
KUNIT_ASSERT_PTR_EQ ( test , skb_network_header ( cur ) , cur - > data + sizeof ( hdr ) ) ;
/* header was copied to all segs */
KUNIT_ASSERT_EQ ( test , memcmp ( skb_mac_header ( cur ) , hdr , sizeof ( hdr ) ) , 0 ) ;
2023-10-09 10:41:52 -04:00
/* last seg can be found through segs->prev pointer */
2023-10-12 14:02:37 +02:00
if ( ! next )
KUNIT_ASSERT_PTR_EQ ( test , cur , last ) ;
consume_skb ( cur ) ;
2023-10-09 10:41:51 -04:00
}
2023-10-09 10:41:52 -04:00
KUNIT_ASSERT_EQ ( test , i , tcase - > nr_segs ) ;
2023-10-09 10:41:51 -04:00
free_gso_skb :
consume_skb ( skb ) ;
}
static struct kunit_case gso_test_cases [ ] = {
2023-10-09 10:41:52 -04:00
KUNIT_CASE_PARAM ( gso_test_func , gso_test_gen_params ) ,
2023-10-09 10:41:51 -04:00
{ }
} ;
static struct kunit_suite gso_test_suite = {
. name = " net_core_gso " ,
. test_cases = gso_test_cases ,
} ;
kunit_test_suite ( gso_test_suite ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " KUnit tests for segmentation offload " ) ;