extern int             netdev_max_backlog;
 extern int             dev_rx_weight;
 extern int             dev_tx_weight;
-extern int             gro_normal_batch;
 
 enum {
        NESTED_SYNC_IMM_BIT,
 
 #include <net/ip6_checksum.h>
 #include <linux/skbuff.h>
 #include <net/udp.h>
+#include <net/hotdata.h>
 
 struct napi_gro_cb {
        union {
 {
        list_add_tail(&skb->list, &napi->rx_list);
        napi->rx_count += segs;
-       if (napi->rx_count >= READ_ONCE(gro_normal_batch))
+       if (napi->rx_count >= READ_ONCE(net_hotdata.gro_normal_batch))
                gro_normal_list(napi);
 }
 
 #endif
 }
 
-extern struct list_head offload_base;
-
 #endif /* _NET_IPV6_GRO_H */
 
--- /dev/null
+/* SPDX-License-Identifier: GPL-2.0-or-later */
+#ifndef _NET_HOTDATA_H
+#define _NET_HOTDATA_H
+
+#include <linux/types.h>
+
+/* Read mostly data used in network fast paths. */
+struct net_hotdata {
+       struct list_head        offload_base;
+       int                     gro_normal_batch;
+};
+
+extern struct net_hotdata net_hotdata;
+
+#endif /* _NET_HOTDATA_H */
 
 obj-$(CONFIG_NETDEV_ADDR_LIST_TEST) += dev_addr_lists_test.o
 
 obj-y += net-sysfs.o
+obj-y += hotdata.o
 obj-$(CONFIG_PAGE_POOL) += page_pool.o page_pool_user.o
 obj-$(CONFIG_PROC_FS) += net-procfs.o
 obj-$(CONFIG_NET_PKTGEN) += pktgen.o
 
 #define GRO_MAX_HEAD (MAX_HEADER + 128)
 
 static DEFINE_SPINLOCK(offload_lock);
-struct list_head offload_base __read_mostly = LIST_HEAD_INIT(offload_base);
-/* Maximum number of GRO_NORMAL skbs to batch up for list-RX */
-int gro_normal_batch __read_mostly = 8;
 
 /**
  *     dev_add_offload - register offload handlers
        struct packet_offload *elem;
 
        spin_lock(&offload_lock);
-       list_for_each_entry(elem, &offload_base, list) {
+       list_for_each_entry(elem, &net_hotdata.offload_base, list) {
                if (po->priority < elem->priority)
                        break;
        }
  */
 static void __dev_remove_offload(struct packet_offload *po)
 {
-       struct list_head *head = &offload_base;
+       struct list_head *head = &net_hotdata.offload_base;
        struct packet_offload *po1;
 
        spin_lock(&offload_lock);
 
 static void napi_gro_complete(struct napi_struct *napi, struct sk_buff *skb)
 {
+       struct list_head *head = &net_hotdata.offload_base;
        struct packet_offload *ptype;
        __be16 type = skb->protocol;
-       struct list_head *head = &offload_base;
        int err = -ENOENT;
 
        BUILD_BUG_ON(sizeof(struct napi_gro_cb) > sizeof(skb->cb));
 {
        u32 bucket = skb_get_hash_raw(skb) & (GRO_HASH_BUCKETS - 1);
        struct gro_list *gro_list = &napi->gro_hash[bucket];
-       struct list_head *head = &offload_base;
+       struct list_head *head = &net_hotdata.offload_base;
        struct packet_offload *ptype;
        __be16 type = skb->protocol;
        struct sk_buff *pp = NULL;
 
 struct packet_offload *gro_find_receive_by_type(__be16 type)
 {
-       struct list_head *offload_head = &offload_base;
+       struct list_head *offload_head = &net_hotdata.offload_base;
        struct packet_offload *ptype;
 
        list_for_each_entry_rcu(ptype, offload_head, list) {
 
 struct packet_offload *gro_find_complete_by_type(__be16 type)
 {
-       struct list_head *offload_head = &offload_base;
+       struct list_head *offload_head = &net_hotdata.offload_base;
        struct packet_offload *ptype;
 
        list_for_each_entry_rcu(ptype, offload_head, list) {
 
        struct packet_offload *ptype;
 
        rcu_read_lock();
-       list_for_each_entry_rcu(ptype, &offload_base, list) {
+       list_for_each_entry_rcu(ptype, &net_hotdata.offload_base, list) {
                if (ptype->type == type && ptype->callbacks.gso_segment) {
                        segs = ptype->callbacks.gso_segment(skb, features);
                        break;
        __skb_pull(skb, vlan_depth);
 
        rcu_read_lock();
-       list_for_each_entry_rcu(ptype, &offload_base, list) {
+       list_for_each_entry_rcu(ptype, &net_hotdata.offload_base, list) {
                if (ptype->type == type && ptype->callbacks.gso_segment) {
                        segs = ptype->callbacks.gso_segment(skb, features);
                        break;
 
--- /dev/null
+// SPDX-License-Identifier: GPL-2.0-or-later
+#include <net/hotdata.h>
+#include <linux/cache.h>
+#include <linux/list.h>
+
+struct net_hotdata net_hotdata __cacheline_aligned = {
+       .offload_base = LIST_HEAD_INIT(net_hotdata.offload_base),
+       .gro_normal_batch = 8,
+};
 
 #include <net/net_ratelimit.h>
 #include <net/busy_poll.h>
 #include <net/pkt_sched.h>
+#include <net/hotdata.h>
 
 #include "dev.h"
 
        },
        {
                .procname       = "gro_normal_batch",
-               .data           = &gro_normal_batch,
+               .data           = &net_hotdata.gro_normal_batch,
                .maxlen         = sizeof(unsigned int),
                .mode           = 0644,
                .proc_handler   = proc_dointvec_minmax,