public inbox for netdev@vger.kernel.org
 help / color / mirror / Atom feed
* [PATCH net-next 0/3] gro: inline tcp6_gro_{receive,complete}
@ 2026-01-16 15:29 Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 1/3] net: always inline __skb_incr_checksum_unnecessary() Eric Dumazet
                   ` (2 more replies)
  0 siblings, 3 replies; 7+ messages in thread
From: Eric Dumazet @ 2026-01-16 15:29 UTC (permalink / raw)
  To: David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: Simon Horman, netdev, Willem de Bruijn, Kuniyuki Iwashima,
	eric.dumazet, Eric Dumazet

On some platforms, GRO stack is too deep and causes cpu stalls.

Decreasing call depths by one shows a 1.5 % gain on Zen 2 cpus.
(32 RX queues, 100Gbit NIC, RFS enabled, tcp_rr with 128 threads and 10,000 flows)

We can go further by inlining ipv6_gro_{receive,complete}
and take care of IPv4 if there is interest.

Cumulative size increase for this series (of 3):

$ scripts/bloat-o-meter -t vmlinux.0 vmlinux.3
add/remove: 2/2 grow/shrink: 5/1 up/down: 1572/-471 (1101)
Function                                     old     new   delta
ipv6_gro_receive                            1069    1846    +777
ipv6_gro_complete                            433     733    +300
tcp6_check_fraglist_gro                        -     272    +272
tcp6_gro_complete                            227     306     +79
tcp4_gro_complete                            325     397     +72
ipv6_offload_init                            218     274     +56
__pfx_tcp6_check_fraglist_gro                  -      16     +16
__pfx___skb_incr_checksum_unnecessary         32       -     -32
__skb_incr_checksum_unnecessary              186       -    -186
tcp6_gro_receive                             959     706    -253
Total: Before=22592724, After=22593825, chg +0.00%


Eric Dumazet (3):
  net: always inline __skb_incr_checksum_unnecessary()
  gro: inline tcp6_gro_receive()
  gro: inline tcp6_gro_complete()

 include/linux/skbuff.h   |  2 +-
 include/net/tcp.h        |  2 --
 net/ipv6/Makefile        |  2 +-
 net/ipv6/ip6_offload.c   | 43 ++++++++++++++++++++--------------------
 net/ipv6/tcpv6_offload.c | 12 +++++------
 5 files changed, 29 insertions(+), 32 deletions(-)

-- 
2.52.0.457.g6b5491de43-goog


^ permalink raw reply	[flat|nested] 7+ messages in thread

* [PATCH net-next 1/3] net: always inline __skb_incr_checksum_unnecessary()
  2026-01-16 15:29 [PATCH net-next 0/3] gro: inline tcp6_gro_{receive,complete} Eric Dumazet
@ 2026-01-16 15:29 ` Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 3/3] gro: inline tcp6_gro_complete() Eric Dumazet
  2 siblings, 0 replies; 7+ messages in thread
From: Eric Dumazet @ 2026-01-16 15:29 UTC (permalink / raw)
  To: David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: Simon Horman, netdev, Willem de Bruijn, Kuniyuki Iwashima,
	eric.dumazet, Eric Dumazet

clang does not inline this helper in GRO fast path.

We can save space and cpu cycles.

$ scripts/bloat-o-meter -t vmlinux.0 vmlinux.1
add/remove: 0/2 grow/shrink: 2/0 up/down: 156/-218 (-62)
Function                                     old     new   delta
tcp6_gro_complete                            227     311     +84
tcp4_gro_complete                            325     397     +72
__pfx___skb_incr_checksum_unnecessary         32       -     -32
__skb_incr_checksum_unnecessary              186       -    -186
Total: Before=22592724, After=22592662, chg -0.00%

Signed-off-by: Eric Dumazet <edumazet@google.com>
---
 include/linux/skbuff.h | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/include/linux/skbuff.h b/include/linux/skbuff.h
index 86737076101d4a8452e90fe78adcdcfdefb79169..e6bfe5d0c5252b2e7540e1fef9317aab83feced2 100644
--- a/include/linux/skbuff.h
+++ b/include/linux/skbuff.h
@@ -4763,7 +4763,7 @@ static inline void __skb_decr_checksum_unnecessary(struct sk_buff *skb)
 	}
 }
 
-static inline void __skb_incr_checksum_unnecessary(struct sk_buff *skb)
+static __always_inline void __skb_incr_checksum_unnecessary(struct sk_buff *skb)
 {
 	if (skb->ip_summed == CHECKSUM_UNNECESSARY) {
 		if (skb->csum_level < SKB_MAX_CSUM_LEVEL)
-- 
2.52.0.457.g6b5491de43-goog


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
  2026-01-16 15:29 [PATCH net-next 0/3] gro: inline tcp6_gro_{receive,complete} Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 1/3] net: always inline __skb_incr_checksum_unnecessary() Eric Dumazet
@ 2026-01-16 15:29 ` Eric Dumazet
  2026-01-17  1:52   ` kernel test robot
                     ` (2 more replies)
  2026-01-16 15:29 ` [PATCH net-next 3/3] gro: inline tcp6_gro_complete() Eric Dumazet
  2 siblings, 3 replies; 7+ messages in thread
From: Eric Dumazet @ 2026-01-16 15:29 UTC (permalink / raw)
  To: David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: Simon Horman, netdev, Willem de Bruijn, Kuniyuki Iwashima,
	eric.dumazet, Eric Dumazet

FDO/LTO are unable to inline tcp6_gro_receive() from ipv6_gro_receive()

Make sure tcp6_check_fraglist_gro() is only called only when needed,
so that compiler can leave it out-of-line.

$ scripts/bloat-o-meter -t vmlinux.1 vmlinux.2
add/remove: 2/0 grow/shrink: 3/1 up/down: 1123/-253 (870)
Function                                     old     new   delta
ipv6_gro_receive                            1069    1846    +777
tcp6_check_fraglist_gro                        -     272    +272
ipv6_offload_init                            218     274     +56
__pfx_tcp6_check_fraglist_gro                  -      16     +16
ipv6_gro_complete                            433     435      +2
tcp6_gro_receive                             959     706    -253
Total: Before=22592662, After=22593532, chg +0.00%

Signed-off-by: Eric Dumazet <edumazet@google.com>
---
 include/net/tcp.h        |  1 -
 net/ipv6/Makefile        |  2 +-
 net/ipv6/ip6_offload.c   | 22 +++++++++++++---------
 net/ipv6/tcpv6_offload.c | 10 ++++------
 4 files changed, 18 insertions(+), 17 deletions(-)

diff --git a/include/net/tcp.h b/include/net/tcp.h
index ef0fee58fde82620399df49a35c7ecae8e34068e..2a7744de226ed0a378719fd60fa2a3830bef2e7e 100644
--- a/include/net/tcp.h
+++ b/include/net/tcp.h
@@ -2328,7 +2328,6 @@ struct sk_buff *tcp_gro_receive(struct list_head *head, struct sk_buff *skb,
 INDIRECT_CALLABLE_DECLARE(int tcp4_gro_complete(struct sk_buff *skb, int thoff));
 INDIRECT_CALLABLE_DECLARE(struct sk_buff *tcp4_gro_receive(struct list_head *head, struct sk_buff *skb));
 INDIRECT_CALLABLE_DECLARE(int tcp6_gro_complete(struct sk_buff *skb, int thoff));
-INDIRECT_CALLABLE_DECLARE(struct sk_buff *tcp6_gro_receive(struct list_head *head, struct sk_buff *skb));
 #ifdef CONFIG_INET
 void tcp_gro_complete(struct sk_buff *skb);
 #else
diff --git a/net/ipv6/Makefile b/net/ipv6/Makefile
index d283c59df4c1c421bc043056fe11e5437cc4aece..0492f1a0b4918ada8c56cf649fbec04c7114863a 100644
--- a/net/ipv6/Makefile
+++ b/net/ipv6/Makefile
@@ -45,7 +45,7 @@ obj-$(CONFIG_IPV6_FOU) += fou6.o
 
 obj-y += addrconf_core.o exthdrs_core.o ip6_checksum.o ip6_icmp.o
 obj-$(CONFIG_INET) += output_core.o protocol.o \
-			ip6_offload.o tcpv6_offload.o exthdrs_offload.o
+			ip6_offload.o exthdrs_offload.o
 
 obj-$(subst m,y,$(CONFIG_IPV6)) += inet6_hashtables.o
 
diff --git a/net/ipv6/ip6_offload.c b/net/ipv6/ip6_offload.c
index fce91183797a60fcbf271c73e086aeb0aa9d40c6..4d96154c0dcd019322908ab6ddaa663a2a565e44 100644
--- a/net/ipv6/ip6_offload.c
+++ b/net/ipv6/ip6_offload.c
@@ -19,6 +19,7 @@
 #include <net/gso.h>
 
 #include "ip6_offload.h"
+#include "tcpv6_offload.c"
 
 /* All GRO functions are always builtin, except UDP over ipv6, which lays in
  * ipv6 module, as it depends on UDPv6 lookup function, so we need special care
@@ -30,13 +31,6 @@
 #define INDIRECT_CALL_L4(f, f2, f1, ...) INDIRECT_CALL_1(f, f2, __VA_ARGS__)
 #endif
 
-#define indirect_call_gro_receive_l4(f2, f1, cb, head, skb)	\
-({								\
-	unlikely(gro_recursion_inc_test(skb)) ?			\
-		NAPI_GRO_CB(skb)->flush |= 1, NULL :		\
-		INDIRECT_CALL_L4(cb, f2, f1, head, skb);	\
-})
-
 static int ipv6_gro_pull_exthdrs(struct sk_buff *skb, int off, int proto)
 {
 	const struct net_offload *ops = NULL;
@@ -298,9 +292,19 @@ INDIRECT_CALLABLE_SCOPE struct sk_buff *ipv6_gro_receive(struct list_head *head,
 
 	skb_gro_postpull_rcsum(skb, iph, nlen);
 
-	pp = indirect_call_gro_receive_l4(tcp6_gro_receive, udp6_gro_receive,
-					 ops->callbacks.gro_receive, head, skb);
+	if (unlikely(gro_recursion_inc_test(skb))) {
+		flush = 1;
+		goto out;
+	}
 
+	if (likely(proto == IPPROTO_TCP))
+		pp = tcp6_gro_receive(head, skb);
+#if IS_BUILTIN(CONFIG_IPV6)
+	else if (likely(proto == IPPROTO_UDP))
+		pp = udp6_gro_receive(head, skb);
+#endif
+	else
+		pp = ops->callbacks.gro_receive(head, skb);
 out:
 	skb_gro_flush_final(skb, pp, flush);
 
diff --git a/net/ipv6/tcpv6_offload.c b/net/ipv6/tcpv6_offload.c
index effeba58630b5ac2593b824bd8fc10a473954b6c..7f19ce423058870f285b7f8ae2a4d116d783f9fb 100644
--- a/net/ipv6/tcpv6_offload.c
+++ b/net/ipv6/tcpv6_offload.c
@@ -24,9 +24,6 @@ static void tcp6_check_fraglist_gro(struct list_head *head, struct sk_buff *skb,
 	struct net *net;
 	int iif, sdif;
 
-	if (likely(!(skb->dev->features & NETIF_F_GRO_FRAGLIST)))
-		return;
-
 	p = tcp_gro_lookup(head, th);
 	if (p) {
 		NAPI_GRO_CB(skb)->is_flist = NAPI_GRO_CB(p)->is_flist;
@@ -45,8 +42,8 @@ static void tcp6_check_fraglist_gro(struct list_head *head, struct sk_buff *skb,
 #endif /* IS_ENABLED(CONFIG_IPV6) */
 }
 
-INDIRECT_CALLABLE_SCOPE
-struct sk_buff *tcp6_gro_receive(struct list_head *head, struct sk_buff *skb)
+static __always_inline struct sk_buff *tcp6_gro_receive(struct list_head *head,
+							struct sk_buff *skb)
 {
 	struct tcphdr *th;
 
@@ -60,7 +57,8 @@ struct sk_buff *tcp6_gro_receive(struct list_head *head, struct sk_buff *skb)
 	if (!th)
 		goto flush;
 
-	tcp6_check_fraglist_gro(head, skb, th);
+	if (unlikely(skb->dev->features & NETIF_F_GRO_FRAGLIST))
+		tcp6_check_fraglist_gro(head, skb, th);
 
 	return tcp_gro_receive(head, skb, th);
 
-- 
2.52.0.457.g6b5491de43-goog


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH net-next 3/3] gro: inline tcp6_gro_complete()
  2026-01-16 15:29 [PATCH net-next 0/3] gro: inline tcp6_gro_{receive,complete} Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 1/3] net: always inline __skb_incr_checksum_unnecessary() Eric Dumazet
  2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
@ 2026-01-16 15:29 ` Eric Dumazet
  2 siblings, 0 replies; 7+ messages in thread
From: Eric Dumazet @ 2026-01-16 15:29 UTC (permalink / raw)
  To: David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: Simon Horman, netdev, Willem de Bruijn, Kuniyuki Iwashima,
	eric.dumazet, Eric Dumazet

Remove one function call from GRO stack for native IPv6 + TCP packets.

$ scripts/bloat-o-meter -t vmlinux.2 vmlinux.3
add/remove: 0/0 grow/shrink: 1/1 up/down: 298/-5 (293)
Function                                     old     new   delta
ipv6_gro_complete                            435     733    +298
tcp6_gro_complete                            311     306      -5
Total: Before=22593532, After=22593825, chg +0.00%

Signed-off-by: Eric Dumazet <edumazet@google.com>
---
 include/net/tcp.h        |  1 -
 net/ipv6/ip6_offload.c   | 21 +++++++++------------
 net/ipv6/tcpv6_offload.c |  2 +-
 3 files changed, 10 insertions(+), 14 deletions(-)

diff --git a/include/net/tcp.h b/include/net/tcp.h
index 2a7744de226ed0a378719fd60fa2a3830bef2e7e..68bc08c40d5435ce196dd492b46b5d81f8e29b3f 100644
--- a/include/net/tcp.h
+++ b/include/net/tcp.h
@@ -2327,7 +2327,6 @@ struct sk_buff *tcp_gro_receive(struct list_head *head, struct sk_buff *skb,
 				struct tcphdr *th);
 INDIRECT_CALLABLE_DECLARE(int tcp4_gro_complete(struct sk_buff *skb, int thoff));
 INDIRECT_CALLABLE_DECLARE(struct sk_buff *tcp4_gro_receive(struct list_head *head, struct sk_buff *skb));
-INDIRECT_CALLABLE_DECLARE(int tcp6_gro_complete(struct sk_buff *skb, int thoff));
 #ifdef CONFIG_INET
 void tcp_gro_complete(struct sk_buff *skb);
 #else
diff --git a/net/ipv6/ip6_offload.c b/net/ipv6/ip6_offload.c
index 4d96154c0dcd019322908ab6ddaa663a2a565e44..32a104ead8760d33e152e0b0a6a6896d70d155b5 100644
--- a/net/ipv6/ip6_offload.c
+++ b/net/ipv6/ip6_offload.c
@@ -21,16 +21,6 @@
 #include "ip6_offload.h"
 #include "tcpv6_offload.c"
 
-/* All GRO functions are always builtin, except UDP over ipv6, which lays in
- * ipv6 module, as it depends on UDPv6 lookup function, so we need special care
- * when ipv6 is built as a module
- */
-#if IS_BUILTIN(CONFIG_IPV6)
-#define INDIRECT_CALL_L4(f, f2, f1, ...) INDIRECT_CALL_2(f, f2, f1, __VA_ARGS__)
-#else
-#define INDIRECT_CALL_L4(f, f2, f1, ...) INDIRECT_CALL_1(f, f2, __VA_ARGS__)
-#endif
-
 static int ipv6_gro_pull_exthdrs(struct sk_buff *skb, int off, int proto)
 {
 	const struct net_offload *ops = NULL;
@@ -383,11 +373,18 @@ INDIRECT_CALLABLE_SCOPE int ipv6_gro_complete(struct sk_buff *skb, int nhoff)
 	}
 
 	nhoff += sizeof(*iph) + ipv6_exthdrs_len(iph, &ops);
+
+	if (likely(ops == &net_hotdata.tcpv6_offload))
+		return tcp6_gro_complete(skb, nhoff);
+#if IS_BUILTIN(CONFIG_IPV6)
+	if (ops == &net_hotdata.udpv6_offload)
+		return udp6_gro_complete(skb, nhoff);
+#endif
+
 	if (WARN_ON(!ops || !ops->callbacks.gro_complete))
 		goto out;
 
-	err = INDIRECT_CALL_L4(ops->callbacks.gro_complete, tcp6_gro_complete,
-			       udp6_gro_complete, skb, nhoff);
+	err = ops->callbacks.gro_complete(skb, nhoff);
 
 out:
 	return err;
diff --git a/net/ipv6/tcpv6_offload.c b/net/ipv6/tcpv6_offload.c
index 7f19ce423058870f285b7f8ae2a4d116d783f9fb..46fa2069d321663ed232e2836db77e3fcb1f4f07 100644
--- a/net/ipv6/tcpv6_offload.c
+++ b/net/ipv6/tcpv6_offload.c
@@ -67,7 +67,7 @@ static __always_inline struct sk_buff *tcp6_gro_receive(struct list_head *head,
 	return NULL;
 }
 
-INDIRECT_CALLABLE_SCOPE int tcp6_gro_complete(struct sk_buff *skb, int thoff)
+static __always_inline int tcp6_gro_complete(struct sk_buff *skb, int thoff)
 {
 	const u16 offset = NAPI_GRO_CB(skb)->network_offsets[skb->encapsulation];
 	const struct ipv6hdr *iph = (struct ipv6hdr *)(skb->data + offset);
-- 
2.52.0.457.g6b5491de43-goog


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* Re: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
  2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
@ 2026-01-17  1:52   ` kernel test robot
  2026-01-17  2:05   ` kernel test robot
  2026-01-17  2:05   ` kernel test robot
  2 siblings, 0 replies; 7+ messages in thread
From: kernel test robot @ 2026-01-17  1:52 UTC (permalink / raw)
  To: Eric Dumazet, David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: llvm, oe-kbuild-all, Simon Horman, netdev, Willem de Bruijn,
	Kuniyuki Iwashima, eric.dumazet, Eric Dumazet

Hi Eric,

kernel test robot noticed the following build errors:

[auto build test ERROR on net-next/main]

url:    https://github.com/intel-lab-lkp/linux/commits/Eric-Dumazet/net-always-inline-__skb_incr_checksum_unnecessary/20260116-233745
base:   net-next/main
patch link:    https://lore.kernel.org/r/20260116152957.1825626-3-edumazet%40google.com
patch subject: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
config: i386-randconfig-011-20260117 (https://download.01.org/0day-ci/archive/20260117/202601170955.0wVpALLC-lkp@intel.com/config)
compiler: clang version 20.1.8 (https://github.com/llvm/llvm-project 87f0227cb60147a26a1eeb4fb06e3b505e9c7261)
reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20260117/202601170955.0wVpALLC-lkp@intel.com/reproduce)

If you fix the issue in a separate patch/commit (i.e. not just a new version of
the same patch/commit), kindly add following tags
| Reported-by: kernel test robot <lkp@intel.com>
| Closes: https://lore.kernel.org/oe-kbuild-all/202601170955.0wVpALLC-lkp@intel.com/

All errors (new ones prefixed by >>):

>> net/ipv6/ip6_offload.c:304:8: error: call to undeclared function 'udp6_gro_receive'; ISO C99 and later do not support implicit function declarations [-Wimplicit-function-declaration]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                      ^
   net/ipv6/ip6_offload.c:304:8: note: did you mean 'udp_gro_receive'?
   include/net/gro.h:419:17: note: 'udp_gro_receive' declared here
     419 | struct sk_buff *udp_gro_receive(struct list_head *head, struct sk_buff *skb,
         |                 ^
>> net/ipv6/ip6_offload.c:304:6: error: incompatible integer to pointer conversion assigning to 'struct sk_buff *' from 'int' [-Wint-conversion]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                    ^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~
   2 errors generated.


vim +/udp6_gro_receive +304 net/ipv6/ip6_offload.c

   215	
   216	INDIRECT_CALLABLE_SCOPE struct sk_buff *ipv6_gro_receive(struct list_head *head,
   217								 struct sk_buff *skb)
   218	{
   219		const struct net_offload *ops;
   220		struct sk_buff *pp = NULL;
   221		struct sk_buff *p;
   222		struct ipv6hdr *iph;
   223		unsigned int nlen;
   224		unsigned int hlen;
   225		unsigned int off;
   226		u16 flush = 1;
   227		int proto;
   228	
   229		off = skb_gro_offset(skb);
   230		hlen = off + sizeof(*iph);
   231		iph = skb_gro_header(skb, hlen, off);
   232		if (unlikely(!iph))
   233			goto out;
   234	
   235		NAPI_GRO_CB(skb)->network_offsets[NAPI_GRO_CB(skb)->encap_mark] = off;
   236	
   237		flush += ntohs(iph->payload_len) != skb->len - hlen;
   238	
   239		proto = iph->nexthdr;
   240		ops = rcu_dereference(inet6_offloads[proto]);
   241		if (!ops || !ops->callbacks.gro_receive) {
   242			proto = ipv6_gro_pull_exthdrs(skb, hlen, proto);
   243	
   244			ops = rcu_dereference(inet6_offloads[proto]);
   245			if (!ops || !ops->callbacks.gro_receive)
   246				goto out;
   247	
   248			iph = skb_gro_network_header(skb);
   249		} else {
   250			skb_gro_pull(skb, sizeof(*iph));
   251		}
   252	
   253		skb_set_transport_header(skb, skb_gro_offset(skb));
   254	
   255		NAPI_GRO_CB(skb)->proto = proto;
   256	
   257		flush--;
   258		nlen = skb_gro_offset(skb) - off;
   259	
   260		list_for_each_entry(p, head, list) {
   261			const struct ipv6hdr *iph2;
   262			__be32 first_word; /* <Version:4><Traffic_Class:8><Flow_Label:20> */
   263	
   264			if (!NAPI_GRO_CB(p)->same_flow)
   265				continue;
   266	
   267			iph2 = (struct ipv6hdr *)(p->data + off);
   268			first_word = *(__be32 *)iph ^ *(__be32 *)iph2;
   269	
   270			/* All fields must match except length and Traffic Class.
   271			 * XXX skbs on the gro_list have all been parsed and pulled
   272			 * already so we don't need to compare nlen
   273			 * (nlen != (sizeof(*iph2) + ipv6_exthdrs_len(iph2, &ops)))
   274			 * memcmp() alone below is sufficient, right?
   275			 */
   276			 if ((first_word & htonl(0xF00FFFFF)) ||
   277			     !ipv6_addr_equal(&iph->saddr, &iph2->saddr) ||
   278			     !ipv6_addr_equal(&iph->daddr, &iph2->daddr) ||
   279			     iph->nexthdr != iph2->nexthdr) {
   280	not_same_flow:
   281				NAPI_GRO_CB(p)->same_flow = 0;
   282				continue;
   283			}
   284			if (unlikely(nlen > sizeof(struct ipv6hdr))) {
   285				if (memcmp(iph + 1, iph2 + 1,
   286					   nlen - sizeof(struct ipv6hdr)))
   287					goto not_same_flow;
   288			}
   289		}
   290	
   291		NAPI_GRO_CB(skb)->flush |= flush;
   292	
   293		skb_gro_postpull_rcsum(skb, iph, nlen);
   294	
   295		if (unlikely(gro_recursion_inc_test(skb))) {
   296			flush = 1;
   297			goto out;
   298		}
   299	
   300		if (likely(proto == IPPROTO_TCP))
   301			pp = tcp6_gro_receive(head, skb);
   302	#if IS_BUILTIN(CONFIG_IPV6)
   303		else if (likely(proto == IPPROTO_UDP))
 > 304			pp = udp6_gro_receive(head, skb);
   305	#endif
   306		else
   307			pp = ops->callbacks.gro_receive(head, skb);
   308	out:
   309		skb_gro_flush_final(skb, pp, flush);
   310	
   311		return pp;
   312	}
   313	

-- 
0-DAY CI Kernel Test Service
https://github.com/intel/lkp-tests/wiki

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
  2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
  2026-01-17  1:52   ` kernel test robot
@ 2026-01-17  2:05   ` kernel test robot
  2026-01-17  2:05   ` kernel test robot
  2 siblings, 0 replies; 7+ messages in thread
From: kernel test robot @ 2026-01-17  2:05 UTC (permalink / raw)
  To: Eric Dumazet, David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: oe-kbuild-all, Simon Horman, netdev, Willem de Bruijn,
	Kuniyuki Iwashima, eric.dumazet, Eric Dumazet

Hi Eric,

kernel test robot noticed the following build warnings:

[auto build test WARNING on net-next/main]

url:    https://github.com/intel-lab-lkp/linux/commits/Eric-Dumazet/net-always-inline-__skb_incr_checksum_unnecessary/20260116-233745
base:   net-next/main
patch link:    https://lore.kernel.org/r/20260116152957.1825626-3-edumazet%40google.com
patch subject: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
config: i386-randconfig-015-20260117 (https://download.01.org/0day-ci/archive/20260117/202601170903.keV6lwyg-lkp@intel.com/config)
compiler: gcc-12 (Debian 12.4.0-5) 12.4.0
reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20260117/202601170903.keV6lwyg-lkp@intel.com/reproduce)

If you fix the issue in a separate patch/commit (i.e. not just a new version of
the same patch/commit), kindly add following tags
| Reported-by: kernel test robot <lkp@intel.com>
| Closes: https://lore.kernel.org/oe-kbuild-all/202601170903.keV6lwyg-lkp@intel.com/

All warnings (new ones prefixed by >>):

   net/ipv6/ip6_offload.c: In function 'ipv6_gro_receive':
   net/ipv6/ip6_offload.c:304:22: error: implicit declaration of function 'udp6_gro_receive'; did you mean 'udp_gro_receive'? [-Werror=implicit-function-declaration]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                      ^~~~~~~~~~~~~~~~
         |                      udp_gro_receive
>> net/ipv6/ip6_offload.c:304:20: warning: assignment to 'struct sk_buff *' from 'int' makes pointer from integer without a cast [-Wint-conversion]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                    ^
   cc1: some warnings being treated as errors


vim +304 net/ipv6/ip6_offload.c

   215	
   216	INDIRECT_CALLABLE_SCOPE struct sk_buff *ipv6_gro_receive(struct list_head *head,
   217								 struct sk_buff *skb)
   218	{
   219		const struct net_offload *ops;
   220		struct sk_buff *pp = NULL;
   221		struct sk_buff *p;
   222		struct ipv6hdr *iph;
   223		unsigned int nlen;
   224		unsigned int hlen;
   225		unsigned int off;
   226		u16 flush = 1;
   227		int proto;
   228	
   229		off = skb_gro_offset(skb);
   230		hlen = off + sizeof(*iph);
   231		iph = skb_gro_header(skb, hlen, off);
   232		if (unlikely(!iph))
   233			goto out;
   234	
   235		NAPI_GRO_CB(skb)->network_offsets[NAPI_GRO_CB(skb)->encap_mark] = off;
   236	
   237		flush += ntohs(iph->payload_len) != skb->len - hlen;
   238	
   239		proto = iph->nexthdr;
   240		ops = rcu_dereference(inet6_offloads[proto]);
   241		if (!ops || !ops->callbacks.gro_receive) {
   242			proto = ipv6_gro_pull_exthdrs(skb, hlen, proto);
   243	
   244			ops = rcu_dereference(inet6_offloads[proto]);
   245			if (!ops || !ops->callbacks.gro_receive)
   246				goto out;
   247	
   248			iph = skb_gro_network_header(skb);
   249		} else {
   250			skb_gro_pull(skb, sizeof(*iph));
   251		}
   252	
   253		skb_set_transport_header(skb, skb_gro_offset(skb));
   254	
   255		NAPI_GRO_CB(skb)->proto = proto;
   256	
   257		flush--;
   258		nlen = skb_gro_offset(skb) - off;
   259	
   260		list_for_each_entry(p, head, list) {
   261			const struct ipv6hdr *iph2;
   262			__be32 first_word; /* <Version:4><Traffic_Class:8><Flow_Label:20> */
   263	
   264			if (!NAPI_GRO_CB(p)->same_flow)
   265				continue;
   266	
   267			iph2 = (struct ipv6hdr *)(p->data + off);
   268			first_word = *(__be32 *)iph ^ *(__be32 *)iph2;
   269	
   270			/* All fields must match except length and Traffic Class.
   271			 * XXX skbs on the gro_list have all been parsed and pulled
   272			 * already so we don't need to compare nlen
   273			 * (nlen != (sizeof(*iph2) + ipv6_exthdrs_len(iph2, &ops)))
   274			 * memcmp() alone below is sufficient, right?
   275			 */
   276			 if ((first_word & htonl(0xF00FFFFF)) ||
   277			     !ipv6_addr_equal(&iph->saddr, &iph2->saddr) ||
   278			     !ipv6_addr_equal(&iph->daddr, &iph2->daddr) ||
   279			     iph->nexthdr != iph2->nexthdr) {
   280	not_same_flow:
   281				NAPI_GRO_CB(p)->same_flow = 0;
   282				continue;
   283			}
   284			if (unlikely(nlen > sizeof(struct ipv6hdr))) {
   285				if (memcmp(iph + 1, iph2 + 1,
   286					   nlen - sizeof(struct ipv6hdr)))
   287					goto not_same_flow;
   288			}
   289		}
   290	
   291		NAPI_GRO_CB(skb)->flush |= flush;
   292	
   293		skb_gro_postpull_rcsum(skb, iph, nlen);
   294	
   295		if (unlikely(gro_recursion_inc_test(skb))) {
   296			flush = 1;
   297			goto out;
   298		}
   299	
   300		if (likely(proto == IPPROTO_TCP))
   301			pp = tcp6_gro_receive(head, skb);
   302	#if IS_BUILTIN(CONFIG_IPV6)
   303		else if (likely(proto == IPPROTO_UDP))
 > 304			pp = udp6_gro_receive(head, skb);
   305	#endif
   306		else
   307			pp = ops->callbacks.gro_receive(head, skb);
   308	out:
   309		skb_gro_flush_final(skb, pp, flush);
   310	
   311		return pp;
   312	}
   313	

-- 
0-DAY CI Kernel Test Service
https://github.com/intel/lkp-tests/wiki

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
  2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
  2026-01-17  1:52   ` kernel test robot
  2026-01-17  2:05   ` kernel test robot
@ 2026-01-17  2:05   ` kernel test robot
  2 siblings, 0 replies; 7+ messages in thread
From: kernel test robot @ 2026-01-17  2:05 UTC (permalink / raw)
  To: Eric Dumazet, David S . Miller, Jakub Kicinski, Paolo Abeni
  Cc: oe-kbuild-all, Simon Horman, netdev, Willem de Bruijn,
	Kuniyuki Iwashima, eric.dumazet, Eric Dumazet

Hi Eric,

kernel test robot noticed the following build errors:

[auto build test ERROR on net-next/main]

url:    https://github.com/intel-lab-lkp/linux/commits/Eric-Dumazet/net-always-inline-__skb_incr_checksum_unnecessary/20260116-233745
base:   net-next/main
patch link:    https://lore.kernel.org/r/20260116152957.1825626-3-edumazet%40google.com
patch subject: [PATCH net-next 2/3] gro: inline tcp6_gro_receive()
config: powerpc-ge_imp3a_defconfig (https://download.01.org/0day-ci/archive/20260117/202601170958.L4uFo8qq-lkp@intel.com/config)
compiler: powerpc-linux-gcc (GCC) 15.2.0
reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20260117/202601170958.L4uFo8qq-lkp@intel.com/reproduce)

If you fix the issue in a separate patch/commit (i.e. not just a new version of
the same patch/commit), kindly add following tags
| Reported-by: kernel test robot <lkp@intel.com>
| Closes: https://lore.kernel.org/oe-kbuild-all/202601170958.L4uFo8qq-lkp@intel.com/

All errors (new ones prefixed by >>):

   net/ipv6/ip6_offload.c: In function 'ipv6_gro_receive':
>> net/ipv6/ip6_offload.c:304:22: error: implicit declaration of function 'udp6_gro_receive'; did you mean 'udp_gro_receive'? [-Wimplicit-function-declaration]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                      ^~~~~~~~~~~~~~~~
         |                      udp_gro_receive
>> net/ipv6/ip6_offload.c:304:20: error: assignment to 'struct sk_buff *' from 'int' makes pointer from integer without a cast [-Wint-conversion]
     304 |                 pp = udp6_gro_receive(head, skb);
         |                    ^


vim +304 net/ipv6/ip6_offload.c

   215	
   216	INDIRECT_CALLABLE_SCOPE struct sk_buff *ipv6_gro_receive(struct list_head *head,
   217								 struct sk_buff *skb)
   218	{
   219		const struct net_offload *ops;
   220		struct sk_buff *pp = NULL;
   221		struct sk_buff *p;
   222		struct ipv6hdr *iph;
   223		unsigned int nlen;
   224		unsigned int hlen;
   225		unsigned int off;
   226		u16 flush = 1;
   227		int proto;
   228	
   229		off = skb_gro_offset(skb);
   230		hlen = off + sizeof(*iph);
   231		iph = skb_gro_header(skb, hlen, off);
   232		if (unlikely(!iph))
   233			goto out;
   234	
   235		NAPI_GRO_CB(skb)->network_offsets[NAPI_GRO_CB(skb)->encap_mark] = off;
   236	
   237		flush += ntohs(iph->payload_len) != skb->len - hlen;
   238	
   239		proto = iph->nexthdr;
   240		ops = rcu_dereference(inet6_offloads[proto]);
   241		if (!ops || !ops->callbacks.gro_receive) {
   242			proto = ipv6_gro_pull_exthdrs(skb, hlen, proto);
   243	
   244			ops = rcu_dereference(inet6_offloads[proto]);
   245			if (!ops || !ops->callbacks.gro_receive)
   246				goto out;
   247	
   248			iph = skb_gro_network_header(skb);
   249		} else {
   250			skb_gro_pull(skb, sizeof(*iph));
   251		}
   252	
   253		skb_set_transport_header(skb, skb_gro_offset(skb));
   254	
   255		NAPI_GRO_CB(skb)->proto = proto;
   256	
   257		flush--;
   258		nlen = skb_gro_offset(skb) - off;
   259	
   260		list_for_each_entry(p, head, list) {
   261			const struct ipv6hdr *iph2;
   262			__be32 first_word; /* <Version:4><Traffic_Class:8><Flow_Label:20> */
   263	
   264			if (!NAPI_GRO_CB(p)->same_flow)
   265				continue;
   266	
   267			iph2 = (struct ipv6hdr *)(p->data + off);
   268			first_word = *(__be32 *)iph ^ *(__be32 *)iph2;
   269	
   270			/* All fields must match except length and Traffic Class.
   271			 * XXX skbs on the gro_list have all been parsed and pulled
   272			 * already so we don't need to compare nlen
   273			 * (nlen != (sizeof(*iph2) + ipv6_exthdrs_len(iph2, &ops)))
   274			 * memcmp() alone below is sufficient, right?
   275			 */
   276			 if ((first_word & htonl(0xF00FFFFF)) ||
   277			     !ipv6_addr_equal(&iph->saddr, &iph2->saddr) ||
   278			     !ipv6_addr_equal(&iph->daddr, &iph2->daddr) ||
   279			     iph->nexthdr != iph2->nexthdr) {
   280	not_same_flow:
   281				NAPI_GRO_CB(p)->same_flow = 0;
   282				continue;
   283			}
   284			if (unlikely(nlen > sizeof(struct ipv6hdr))) {
   285				if (memcmp(iph + 1, iph2 + 1,
   286					   nlen - sizeof(struct ipv6hdr)))
   287					goto not_same_flow;
   288			}
   289		}
   290	
   291		NAPI_GRO_CB(skb)->flush |= flush;
   292	
   293		skb_gro_postpull_rcsum(skb, iph, nlen);
   294	
   295		if (unlikely(gro_recursion_inc_test(skb))) {
   296			flush = 1;
   297			goto out;
   298		}
   299	
   300		if (likely(proto == IPPROTO_TCP))
   301			pp = tcp6_gro_receive(head, skb);
   302	#if IS_BUILTIN(CONFIG_IPV6)
   303		else if (likely(proto == IPPROTO_UDP))
 > 304			pp = udp6_gro_receive(head, skb);
   305	#endif
   306		else
   307			pp = ops->callbacks.gro_receive(head, skb);
   308	out:
   309		skb_gro_flush_final(skb, pp, flush);
   310	
   311		return pp;
   312	}
   313	

-- 
0-DAY CI Kernel Test Service
https://github.com/intel/lkp-tests/wiki

^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2026-01-17  2:05 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-01-16 15:29 [PATCH net-next 0/3] gro: inline tcp6_gro_{receive,complete} Eric Dumazet
2026-01-16 15:29 ` [PATCH net-next 1/3] net: always inline __skb_incr_checksum_unnecessary() Eric Dumazet
2026-01-16 15:29 ` [PATCH net-next 2/3] gro: inline tcp6_gro_receive() Eric Dumazet
2026-01-17  1:52   ` kernel test robot
2026-01-17  2:05   ` kernel test robot
2026-01-17  2:05   ` kernel test robot
2026-01-16 15:29 ` [PATCH net-next 3/3] gro: inline tcp6_gro_complete() Eric Dumazet

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox