Commit d4546c25 authored by David Miller's avatar David Miller Committed by David S. Miller

net: Convert GRO SKB handling to list_head.

Manage pending per-NAPI GRO packets via list_head.

Return an SKB pointer from the GRO receive handlers.  When GRO receive
handlers return non-NULL, it means that this SKB needs to be completed
at this time and removed from the NAPI queue.

Several operations are greatly simplified by this transformation,
especially timing out the oldest SKB in the list when gro_count
exceeds MAX_GRO_SKBS, and napi_gro_flush() which walks the queue
in reverse order.
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 9ff3b40e
...@@ -418,11 +418,12 @@ static int geneve_hlen(struct genevehdr *gh) ...@@ -418,11 +418,12 @@ static int geneve_hlen(struct genevehdr *gh)
return sizeof(*gh) + gh->opt_len * 4; return sizeof(*gh) + gh->opt_len * 4;
} }
static struct sk_buff **geneve_gro_receive(struct sock *sk, static struct sk_buff *geneve_gro_receive(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct sk_buff *p, **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p;
struct genevehdr *gh, *gh2; struct genevehdr *gh, *gh2;
unsigned int hlen, gh_len, off_gnv; unsigned int hlen, gh_len, off_gnv;
const struct packet_offload *ptype; const struct packet_offload *ptype;
...@@ -449,7 +450,7 @@ static struct sk_buff **geneve_gro_receive(struct sock *sk, ...@@ -449,7 +450,7 @@ static struct sk_buff **geneve_gro_receive(struct sock *sk,
goto out; goto out;
} }
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
continue; continue;
......
...@@ -568,11 +568,12 @@ static struct vxlanhdr *vxlan_gro_remcsum(struct sk_buff *skb, ...@@ -568,11 +568,12 @@ static struct vxlanhdr *vxlan_gro_remcsum(struct sk_buff *skb,
return vh; return vh;
} }
static struct sk_buff **vxlan_gro_receive(struct sock *sk, static struct sk_buff *vxlan_gro_receive(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct sk_buff *p, **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p;
struct vxlanhdr *vh, *vh2; struct vxlanhdr *vh, *vh2;
unsigned int hlen, off_vx; unsigned int hlen, off_vx;
int flush = 1; int flush = 1;
...@@ -607,7 +608,7 @@ static struct sk_buff **vxlan_gro_receive(struct sock *sk, ...@@ -607,7 +608,7 @@ static struct sk_buff **vxlan_gro_receive(struct sock *sk,
skb_gro_pull(skb, sizeof(struct vxlanhdr)); /* pull vxlan header */ skb_gro_pull(skb, sizeof(struct vxlanhdr)); /* pull vxlan header */
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
continue; continue;
......
...@@ -59,8 +59,7 @@ struct net_device *devm_alloc_etherdev_mqs(struct device *dev, int sizeof_priv, ...@@ -59,8 +59,7 @@ struct net_device *devm_alloc_etherdev_mqs(struct device *dev, int sizeof_priv,
unsigned int rxqs); unsigned int rxqs);
#define devm_alloc_etherdev(dev, sizeof_priv) devm_alloc_etherdev_mqs(dev, sizeof_priv, 1, 1) #define devm_alloc_etherdev(dev, sizeof_priv) devm_alloc_etherdev_mqs(dev, sizeof_priv, 1, 1)
struct sk_buff **eth_gro_receive(struct sk_buff **head, struct sk_buff *eth_gro_receive(struct list_head *head, struct sk_buff *skb);
struct sk_buff *skb);
int eth_gro_complete(struct sk_buff *skb, int nhoff); int eth_gro_complete(struct sk_buff *skb, int nhoff);
/* Reserved Ethernet Addresses per IEEE 802.1Q */ /* Reserved Ethernet Addresses per IEEE 802.1Q */
......
...@@ -322,7 +322,7 @@ struct napi_struct { ...@@ -322,7 +322,7 @@ struct napi_struct {
int poll_owner; int poll_owner;
#endif #endif
struct net_device *dev; struct net_device *dev;
struct sk_buff *gro_list; struct list_head gro_list;
struct sk_buff *skb; struct sk_buff *skb;
struct hrtimer timer; struct hrtimer timer;
struct list_head dev_list; struct list_head dev_list;
...@@ -2255,10 +2255,10 @@ static inline int gro_recursion_inc_test(struct sk_buff *skb) ...@@ -2255,10 +2255,10 @@ static inline int gro_recursion_inc_test(struct sk_buff *skb)
return ++NAPI_GRO_CB(skb)->recursion_counter == GRO_RECURSION_LIMIT; return ++NAPI_GRO_CB(skb)->recursion_counter == GRO_RECURSION_LIMIT;
} }
typedef struct sk_buff **(*gro_receive_t)(struct sk_buff **, struct sk_buff *); typedef struct sk_buff *(*gro_receive_t)(struct list_head *, struct sk_buff *);
static inline struct sk_buff **call_gro_receive(gro_receive_t cb, static inline struct sk_buff *call_gro_receive(gro_receive_t cb,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
if (unlikely(gro_recursion_inc_test(skb))) { if (unlikely(gro_recursion_inc_test(skb))) {
NAPI_GRO_CB(skb)->flush |= 1; NAPI_GRO_CB(skb)->flush |= 1;
...@@ -2268,12 +2268,12 @@ static inline struct sk_buff **call_gro_receive(gro_receive_t cb, ...@@ -2268,12 +2268,12 @@ static inline struct sk_buff **call_gro_receive(gro_receive_t cb,
return cb(head, skb); return cb(head, skb);
} }
typedef struct sk_buff **(*gro_receive_sk_t)(struct sock *, struct sk_buff **, typedef struct sk_buff *(*gro_receive_sk_t)(struct sock *, struct list_head *,
struct sk_buff *); struct sk_buff *);
static inline struct sk_buff **call_gro_receive_sk(gro_receive_sk_t cb, static inline struct sk_buff *call_gro_receive_sk(gro_receive_sk_t cb,
struct sock *sk, struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
if (unlikely(gro_recursion_inc_test(skb))) { if (unlikely(gro_recursion_inc_test(skb))) {
NAPI_GRO_CB(skb)->flush |= 1; NAPI_GRO_CB(skb)->flush |= 1;
...@@ -2299,8 +2299,8 @@ struct packet_type { ...@@ -2299,8 +2299,8 @@ struct packet_type {
struct offload_callbacks { struct offload_callbacks {
struct sk_buff *(*gso_segment)(struct sk_buff *skb, struct sk_buff *(*gso_segment)(struct sk_buff *skb,
netdev_features_t features); netdev_features_t features);
struct sk_buff **(*gro_receive)(struct sk_buff **head, struct sk_buff *(*gro_receive)(struct list_head *head,
struct sk_buff *skb); struct sk_buff *skb);
int (*gro_complete)(struct sk_buff *skb, int nhoff); int (*gro_complete)(struct sk_buff *skb, int nhoff);
}; };
...@@ -2568,7 +2568,7 @@ struct net_device *dev_get_by_index_rcu(struct net *net, int ifindex); ...@@ -2568,7 +2568,7 @@ struct net_device *dev_get_by_index_rcu(struct net *net, int ifindex);
struct net_device *dev_get_by_napi_id(unsigned int napi_id); struct net_device *dev_get_by_napi_id(unsigned int napi_id);
int netdev_get_name(struct net *net, char *name, int ifindex); int netdev_get_name(struct net *net, char *name, int ifindex);
int dev_restart(struct net_device *dev); int dev_restart(struct net_device *dev);
int skb_gro_receive(struct sk_buff **head, struct sk_buff *skb); int skb_gro_receive(struct sk_buff *p, struct sk_buff *skb);
static inline unsigned int skb_gro_offset(const struct sk_buff *skb) static inline unsigned int skb_gro_offset(const struct sk_buff *skb)
{ {
...@@ -2784,13 +2784,13 @@ static inline void skb_gro_remcsum_cleanup(struct sk_buff *skb, ...@@ -2784,13 +2784,13 @@ static inline void skb_gro_remcsum_cleanup(struct sk_buff *skb,
} }
#ifdef CONFIG_XFRM_OFFLOAD #ifdef CONFIG_XFRM_OFFLOAD
static inline void skb_gro_flush_final(struct sk_buff *skb, struct sk_buff **pp, int flush) static inline void skb_gro_flush_final(struct sk_buff *skb, struct sk_buff *pp, int flush)
{ {
if (PTR_ERR(pp) != -EINPROGRESS) if (PTR_ERR(pp) != -EINPROGRESS)
NAPI_GRO_CB(skb)->flush |= flush; NAPI_GRO_CB(skb)->flush |= flush;
} }
#else #else
static inline void skb_gro_flush_final(struct sk_buff *skb, struct sk_buff **pp, int flush) static inline void skb_gro_flush_final(struct sk_buff *skb, struct sk_buff *pp, int flush)
{ {
NAPI_GRO_CB(skb)->flush |= flush; NAPI_GRO_CB(skb)->flush |= flush;
} }
......
...@@ -677,7 +677,8 @@ struct sk_buff { ...@@ -677,7 +677,8 @@ struct sk_buff {
int ip_defrag_offset; int ip_defrag_offset;
}; };
}; };
struct rb_node rbnode; /* used in netem & tcp stack */ struct rb_node rbnode; /* used in netem & tcp stack */
struct list_head list;
}; };
struct sock *sk; struct sock *sk;
......
...@@ -74,8 +74,8 @@ struct udp_sock { ...@@ -74,8 +74,8 @@ struct udp_sock {
void (*encap_destroy)(struct sock *sk); void (*encap_destroy)(struct sock *sk);
/* GRO functions for UDP socket */ /* GRO functions for UDP socket */
struct sk_buff ** (*gro_receive)(struct sock *sk, struct sk_buff * (*gro_receive)(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb); struct sk_buff *skb);
int (*gro_complete)(struct sock *sk, int (*gro_complete)(struct sock *sk,
struct sk_buff *skb, struct sk_buff *skb,
......
...@@ -43,7 +43,7 @@ int inet_ctl_sock_create(struct sock **sk, unsigned short family, ...@@ -43,7 +43,7 @@ int inet_ctl_sock_create(struct sock **sk, unsigned short family,
int inet_recv_error(struct sock *sk, struct msghdr *msg, int len, int inet_recv_error(struct sock *sk, struct msghdr *msg, int len,
int *addr_len); int *addr_len);
struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb); struct sk_buff *inet_gro_receive(struct list_head *head, struct sk_buff *skb);
int inet_gro_complete(struct sk_buff *skb, int nhoff); int inet_gro_complete(struct sk_buff *skb, int nhoff);
struct sk_buff *inet_gso_segment(struct sk_buff *skb, struct sk_buff *inet_gso_segment(struct sk_buff *skb,
netdev_features_t features); netdev_features_t features);
......
...@@ -1788,7 +1788,7 @@ void tcp_v4_destroy_sock(struct sock *sk); ...@@ -1788,7 +1788,7 @@ void tcp_v4_destroy_sock(struct sock *sk);
struct sk_buff *tcp_gso_segment(struct sk_buff *skb, struct sk_buff *tcp_gso_segment(struct sk_buff *skb,
netdev_features_t features); netdev_features_t features);
struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb); struct sk_buff *tcp_gro_receive(struct list_head *head, struct sk_buff *skb);
int tcp_gro_complete(struct sk_buff *skb); int tcp_gro_complete(struct sk_buff *skb);
void __tcp_v4_send_check(struct sk_buff *skb, __be32 saddr, __be32 daddr); void __tcp_v4_send_check(struct sk_buff *skb, __be32 saddr, __be32 daddr);
......
...@@ -170,8 +170,8 @@ static inline void udp_csum_pull_header(struct sk_buff *skb) ...@@ -170,8 +170,8 @@ static inline void udp_csum_pull_header(struct sk_buff *skb)
typedef struct sock *(*udp_lookup_t)(struct sk_buff *skb, __be16 sport, typedef struct sock *(*udp_lookup_t)(struct sk_buff *skb, __be16 sport,
__be16 dport); __be16 dport);
struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, struct sk_buff *udp_gro_receive(struct list_head *head, struct sk_buff *skb,
struct udphdr *uh, udp_lookup_t lookup); struct udphdr *uh, udp_lookup_t lookup);
int udp_gro_complete(struct sk_buff *skb, int nhoff, udp_lookup_t lookup); int udp_gro_complete(struct sk_buff *skb, int nhoff, udp_lookup_t lookup);
struct sk_buff *__udp_gso_segment(struct sk_buff *gso_skb, struct sk_buff *__udp_gso_segment(struct sk_buff *gso_skb,
......
...@@ -65,9 +65,9 @@ static inline int udp_sock_create(struct net *net, ...@@ -65,9 +65,9 @@ static inline int udp_sock_create(struct net *net,
typedef int (*udp_tunnel_encap_rcv_t)(struct sock *sk, struct sk_buff *skb); typedef int (*udp_tunnel_encap_rcv_t)(struct sock *sk, struct sk_buff *skb);
typedef void (*udp_tunnel_encap_destroy_t)(struct sock *sk); typedef void (*udp_tunnel_encap_destroy_t)(struct sock *sk);
typedef struct sk_buff **(*udp_tunnel_gro_receive_t)(struct sock *sk, typedef struct sk_buff *(*udp_tunnel_gro_receive_t)(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb); struct sk_buff *skb);
typedef int (*udp_tunnel_gro_complete_t)(struct sock *sk, struct sk_buff *skb, typedef int (*udp_tunnel_gro_complete_t)(struct sock *sk, struct sk_buff *skb,
int nhoff); int nhoff);
......
...@@ -647,13 +647,14 @@ static int vlan_ioctl_handler(struct net *net, void __user *arg) ...@@ -647,13 +647,14 @@ static int vlan_ioctl_handler(struct net *net, void __user *arg)
return err; return err;
} }
static struct sk_buff **vlan_gro_receive(struct sk_buff **head, static struct sk_buff *vlan_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct sk_buff *p, **pp = NULL;
struct vlan_hdr *vhdr;
unsigned int hlen, off_vlan;
const struct packet_offload *ptype; const struct packet_offload *ptype;
unsigned int hlen, off_vlan;
struct sk_buff *pp = NULL;
struct vlan_hdr *vhdr;
struct sk_buff *p;
__be16 type; __be16 type;
int flush = 1; int flush = 1;
...@@ -675,7 +676,7 @@ static struct sk_buff **vlan_gro_receive(struct sk_buff **head, ...@@ -675,7 +676,7 @@ static struct sk_buff **vlan_gro_receive(struct sk_buff **head,
flush = 0; flush = 0;
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
struct vlan_hdr *vhdr2; struct vlan_hdr *vhdr2;
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
......
...@@ -4881,36 +4881,25 @@ static int napi_gro_complete(struct sk_buff *skb) ...@@ -4881,36 +4881,25 @@ static int napi_gro_complete(struct sk_buff *skb)
*/ */
void napi_gro_flush(struct napi_struct *napi, bool flush_old) void napi_gro_flush(struct napi_struct *napi, bool flush_old)
{ {
struct sk_buff *skb, *prev = NULL; struct sk_buff *skb, *p;
/* scan list and build reverse chain */
for (skb = napi->gro_list; skb != NULL; skb = skb->next) {
skb->prev = prev;
prev = skb;
}
for (skb = prev; skb; skb = prev) {
skb->next = NULL;
list_for_each_entry_safe_reverse(skb, p, &napi->gro_list, list) {
if (flush_old && NAPI_GRO_CB(skb)->age == jiffies) if (flush_old && NAPI_GRO_CB(skb)->age == jiffies)
return; return;
list_del_init(&skb->list);
prev = skb->prev;
napi_gro_complete(skb); napi_gro_complete(skb);
napi->gro_count--; napi->gro_count--;
} }
napi->gro_list = NULL;
} }
EXPORT_SYMBOL(napi_gro_flush); EXPORT_SYMBOL(napi_gro_flush);
static void gro_list_prepare(struct napi_struct *napi, struct sk_buff *skb) static void gro_list_prepare(struct napi_struct *napi, struct sk_buff *skb)
{ {
struct sk_buff *p;
unsigned int maclen = skb->dev->hard_header_len; unsigned int maclen = skb->dev->hard_header_len;
u32 hash = skb_get_hash_raw(skb); u32 hash = skb_get_hash_raw(skb);
struct sk_buff *p;
for (p = napi->gro_list; p; p = p->next) { list_for_each_entry(p, &napi->gro_list, list) {
unsigned long diffs; unsigned long diffs;
NAPI_GRO_CB(p)->flush = 0; NAPI_GRO_CB(p)->flush = 0;
...@@ -4977,12 +4966,12 @@ static void gro_pull_from_frag0(struct sk_buff *skb, int grow) ...@@ -4977,12 +4966,12 @@ static void gro_pull_from_frag0(struct sk_buff *skb, int grow)
static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff *skb) static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff *skb)
{ {
struct sk_buff **pp = NULL; struct list_head *head = &offload_base;
struct packet_offload *ptype; struct packet_offload *ptype;
__be16 type = skb->protocol; __be16 type = skb->protocol;
struct list_head *head = &offload_base; struct sk_buff *pp = NULL;
int same_flow;
enum gro_result ret; enum gro_result ret;
int same_flow;
int grow; int grow;
if (netif_elide_gro(skb->dev)) if (netif_elide_gro(skb->dev))
...@@ -5039,11 +5028,8 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff ...@@ -5039,11 +5028,8 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff
ret = NAPI_GRO_CB(skb)->free ? GRO_MERGED_FREE : GRO_MERGED; ret = NAPI_GRO_CB(skb)->free ? GRO_MERGED_FREE : GRO_MERGED;
if (pp) { if (pp) {
struct sk_buff *nskb = *pp; list_del_init(&pp->list);
napi_gro_complete(pp);
*pp = nskb->next;
nskb->next = NULL;
napi_gro_complete(nskb);
napi->gro_count--; napi->gro_count--;
} }
...@@ -5054,15 +5040,10 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff ...@@ -5054,15 +5040,10 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff
goto normal; goto normal;
if (unlikely(napi->gro_count >= MAX_GRO_SKBS)) { if (unlikely(napi->gro_count >= MAX_GRO_SKBS)) {
struct sk_buff *nskb = napi->gro_list; struct sk_buff *nskb;
/* locate the end of the list to select the 'oldest' flow */ nskb = list_last_entry(&napi->gro_list, struct sk_buff, list);
while (nskb->next) { list_del(&nskb->list);
pp = &nskb->next;
nskb = *pp;
}
*pp = NULL;
nskb->next = NULL;
napi_gro_complete(nskb); napi_gro_complete(nskb);
} else { } else {
napi->gro_count++; napi->gro_count++;
...@@ -5071,8 +5052,7 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff ...@@ -5071,8 +5052,7 @@ static enum gro_result dev_gro_receive(struct napi_struct *napi, struct sk_buff
NAPI_GRO_CB(skb)->age = jiffies; NAPI_GRO_CB(skb)->age = jiffies;
NAPI_GRO_CB(skb)->last = skb; NAPI_GRO_CB(skb)->last = skb;
skb_shinfo(skb)->gso_size = skb_gro_len(skb); skb_shinfo(skb)->gso_size = skb_gro_len(skb);
skb->next = napi->gro_list; list_add(&skb->list, &napi->gro_list);
napi->gro_list = skb;
ret = GRO_HELD; ret = GRO_HELD;
pull: pull:
...@@ -5478,7 +5458,7 @@ bool napi_complete_done(struct napi_struct *n, int work_done) ...@@ -5478,7 +5458,7 @@ bool napi_complete_done(struct napi_struct *n, int work_done)
NAPIF_STATE_IN_BUSY_POLL))) NAPIF_STATE_IN_BUSY_POLL)))
return false; return false;
if (n->gro_list) { if (!list_empty(&n->gro_list)) {
unsigned long timeout = 0; unsigned long timeout = 0;
if (work_done) if (work_done)
...@@ -5687,7 +5667,7 @@ static enum hrtimer_restart napi_watchdog(struct hrtimer *timer) ...@@ -5687,7 +5667,7 @@ static enum hrtimer_restart napi_watchdog(struct hrtimer *timer)
/* Note : we use a relaxed variant of napi_schedule_prep() not setting /* Note : we use a relaxed variant of napi_schedule_prep() not setting
* NAPI_STATE_MISSED, since we do not react to a device IRQ. * NAPI_STATE_MISSED, since we do not react to a device IRQ.
*/ */
if (napi->gro_list && !napi_disable_pending(napi) && if (!list_empty(&napi->gro_list) && !napi_disable_pending(napi) &&
!test_and_set_bit(NAPI_STATE_SCHED, &napi->state)) !test_and_set_bit(NAPI_STATE_SCHED, &napi->state))
__napi_schedule_irqoff(napi); __napi_schedule_irqoff(napi);
...@@ -5701,7 +5681,7 @@ void netif_napi_add(struct net_device *dev, struct napi_struct *napi, ...@@ -5701,7 +5681,7 @@ void netif_napi_add(struct net_device *dev, struct napi_struct *napi,
hrtimer_init(&napi->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL_PINNED); hrtimer_init(&napi->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL_PINNED);
napi->timer.function = napi_watchdog; napi->timer.function = napi_watchdog;
napi->gro_count = 0; napi->gro_count = 0;
napi->gro_list = NULL; INIT_LIST_HEAD(&napi->gro_list);
napi->skb = NULL; napi->skb = NULL;
napi->poll = poll; napi->poll = poll;
if (weight > NAPI_POLL_WEIGHT) if (weight > NAPI_POLL_WEIGHT)
...@@ -5734,6 +5714,14 @@ void napi_disable(struct napi_struct *n) ...@@ -5734,6 +5714,14 @@ void napi_disable(struct napi_struct *n)
} }
EXPORT_SYMBOL(napi_disable); EXPORT_SYMBOL(napi_disable);
static void gro_list_free(struct list_head *head)
{
struct sk_buff *skb, *p;
list_for_each_entry_safe(skb, p, head, list)
kfree_skb(skb);
}
/* Must be called in process context */ /* Must be called in process context */
void netif_napi_del(struct napi_struct *napi) void netif_napi_del(struct napi_struct *napi)
{ {
...@@ -5743,8 +5731,8 @@ void netif_napi_del(struct napi_struct *napi) ...@@ -5743,8 +5731,8 @@ void netif_napi_del(struct napi_struct *napi)
list_del_init(&napi->dev_list); list_del_init(&napi->dev_list);
napi_free_frags(napi); napi_free_frags(napi);
kfree_skb_list(napi->gro_list); gro_list_free(&napi->gro_list);
napi->gro_list = NULL; INIT_LIST_HEAD(&napi->gro_list);
napi->gro_count = 0; napi->gro_count = 0;
} }
EXPORT_SYMBOL(netif_napi_del); EXPORT_SYMBOL(netif_napi_del);
...@@ -5787,7 +5775,7 @@ static int napi_poll(struct napi_struct *n, struct list_head *repoll) ...@@ -5787,7 +5775,7 @@ static int napi_poll(struct napi_struct *n, struct list_head *repoll)
goto out_unlock; goto out_unlock;
} }
if (n->gro_list) { if (!list_empty(&n->gro_list)) {
/* flush too old packets /* flush too old packets
* If HZ < 1000, flush all packets. * If HZ < 1000, flush all packets.
*/ */
......
...@@ -3815,14 +3815,14 @@ struct sk_buff *skb_segment(struct sk_buff *head_skb, ...@@ -3815,14 +3815,14 @@ struct sk_buff *skb_segment(struct sk_buff *head_skb,
} }
EXPORT_SYMBOL_GPL(skb_segment); EXPORT_SYMBOL_GPL(skb_segment);
int skb_gro_receive(struct sk_buff **head, struct sk_buff *skb) int skb_gro_receive(struct sk_buff *p, struct sk_buff *skb)
{ {
struct skb_shared_info *pinfo, *skbinfo = skb_shinfo(skb); struct skb_shared_info *pinfo, *skbinfo = skb_shinfo(skb);
unsigned int offset = skb_gro_offset(skb); unsigned int offset = skb_gro_offset(skb);
unsigned int headlen = skb_headlen(skb); unsigned int headlen = skb_headlen(skb);
unsigned int len = skb_gro_len(skb); unsigned int len = skb_gro_len(skb);
struct sk_buff *lp, *p = *head;
unsigned int delta_truesize; unsigned int delta_truesize;
struct sk_buff *lp;
if (unlikely(p->len + len >= 65536)) if (unlikely(p->len + len >= 65536))
return -E2BIG; return -E2BIG;
......
...@@ -427,13 +427,13 @@ ssize_t sysfs_format_mac(char *buf, const unsigned char *addr, int len) ...@@ -427,13 +427,13 @@ ssize_t sysfs_format_mac(char *buf, const unsigned char *addr, int len)
} }
EXPORT_SYMBOL(sysfs_format_mac); EXPORT_SYMBOL(sysfs_format_mac);
struct sk_buff **eth_gro_receive(struct sk_buff **head, struct sk_buff *eth_gro_receive(struct list_head *head, struct sk_buff *skb)
struct sk_buff *skb)
{ {
struct sk_buff *p, **pp = NULL;
struct ethhdr *eh, *eh2;
unsigned int hlen, off_eth;
const struct packet_offload *ptype; const struct packet_offload *ptype;
unsigned int hlen, off_eth;
struct sk_buff *pp = NULL;
struct ethhdr *eh, *eh2;
struct sk_buff *p;
__be16 type; __be16 type;
int flush = 1; int flush = 1;
...@@ -448,7 +448,7 @@ struct sk_buff **eth_gro_receive(struct sk_buff **head, ...@@ -448,7 +448,7 @@ struct sk_buff **eth_gro_receive(struct sk_buff **head,
flush = 0; flush = 0;
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
continue; continue;
......
...@@ -1384,12 +1384,12 @@ struct sk_buff *inet_gso_segment(struct sk_buff *skb, ...@@ -1384,12 +1384,12 @@ struct sk_buff *inet_gso_segment(struct sk_buff *skb,
} }
EXPORT_SYMBOL(inet_gso_segment); EXPORT_SYMBOL(inet_gso_segment);
struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb) struct sk_buff *inet_gro_receive(struct list_head *head, struct sk_buff *skb)
{ {
const struct net_offload *ops; const struct net_offload *ops;
struct sk_buff **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p;
const struct iphdr *iph; const struct iphdr *iph;
struct sk_buff *p;
unsigned int hlen; unsigned int hlen;
unsigned int off; unsigned int off;
unsigned int id; unsigned int id;
...@@ -1425,7 +1425,7 @@ struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -1425,7 +1425,7 @@ struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb)
flush = (u16)((ntohl(*(__be32 *)iph) ^ skb_gro_len(skb)) | (id & ~IP_DF)); flush = (u16)((ntohl(*(__be32 *)iph) ^ skb_gro_len(skb)) | (id & ~IP_DF));
id >>= 16; id >>= 16;
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
struct iphdr *iph2; struct iphdr *iph2;
u16 flush_id; u16 flush_id;
...@@ -1505,8 +1505,8 @@ struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -1505,8 +1505,8 @@ struct sk_buff **inet_gro_receive(struct sk_buff **head, struct sk_buff *skb)
} }
EXPORT_SYMBOL(inet_gro_receive); EXPORT_SYMBOL(inet_gro_receive);
static struct sk_buff **ipip_gro_receive(struct sk_buff **head, static struct sk_buff *ipip_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
if (NAPI_GRO_CB(skb)->encap_mark) { if (NAPI_GRO_CB(skb)->encap_mark) {
NAPI_GRO_CB(skb)->flush = 1; NAPI_GRO_CB(skb)->flush = 1;
......
...@@ -28,8 +28,8 @@ ...@@ -28,8 +28,8 @@
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <net/udp.h> #include <net/udp.h>
static struct sk_buff **esp4_gro_receive(struct sk_buff **head, static struct sk_buff *esp4_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
int offset = skb_gro_offset(skb); int offset = skb_gro_offset(skb);
struct xfrm_offload *xo; struct xfrm_offload *xo;
......
...@@ -224,14 +224,14 @@ static int gue_udp_recv(struct sock *sk, struct sk_buff *skb) ...@@ -224,14 +224,14 @@ static int gue_udp_recv(struct sock *sk, struct sk_buff *skb)
return 0; return 0;
} }
static struct sk_buff **fou_gro_receive(struct sock *sk, static struct sk_buff *fou_gro_receive(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
const struct net_offload *ops;
struct sk_buff **pp = NULL;
u8 proto = fou_from_sock(sk)->protocol; u8 proto = fou_from_sock(sk)->protocol;
const struct net_offload **offloads; const struct net_offload **offloads;
const struct net_offload *ops;
struct sk_buff *pp = NULL;
/* We can clear the encap_mark for FOU as we are essentially doing /* We can clear the encap_mark for FOU as we are essentially doing
* one of two possible things. We are either adding an L4 tunnel * one of two possible things. We are either adding an L4 tunnel
...@@ -305,13 +305,13 @@ static struct guehdr *gue_gro_remcsum(struct sk_buff *skb, unsigned int off, ...@@ -305,13 +305,13 @@ static struct guehdr *gue_gro_remcsum(struct sk_buff *skb, unsigned int off,
return guehdr; return guehdr;
} }
static struct sk_buff **gue_gro_receive(struct sock *sk, static struct sk_buff *gue_gro_receive(struct sock *sk,
struct sk_buff **head, struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
const struct net_offload **offloads; const struct net_offload **offloads;
const struct net_offload *ops; const struct net_offload *ops;
struct sk_buff **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p; struct sk_buff *p;
struct guehdr *guehdr; struct guehdr *guehdr;
size_t len, optlen, hdrlen, off; size_t len, optlen, hdrlen, off;
...@@ -397,7 +397,7 @@ static struct sk_buff **gue_gro_receive(struct sock *sk, ...@@ -397,7 +397,7 @@ static struct sk_buff **gue_gro_receive(struct sock *sk,
skb_gro_pull(skb, hdrlen); skb_gro_pull(skb, hdrlen);
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
const struct guehdr *guehdr2; const struct guehdr *guehdr2;
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
......
...@@ -108,10 +108,10 @@ static struct sk_buff *gre_gso_segment(struct sk_buff *skb, ...@@ -108,10 +108,10 @@ static struct sk_buff *gre_gso_segment(struct sk_buff *skb,
return segs; return segs;
} }
static struct sk_buff **gre_gro_receive(struct sk_buff **head, static struct sk_buff *gre_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct sk_buff **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p; struct sk_buff *p;
const struct gre_base_hdr *greh; const struct gre_base_hdr *greh;
unsigned int hlen, grehlen; unsigned int hlen, grehlen;
...@@ -182,7 +182,7 @@ static struct sk_buff **gre_gro_receive(struct sk_buff **head, ...@@ -182,7 +182,7 @@ static struct sk_buff **gre_gro_receive(struct sk_buff **head,
null_compute_pseudo); null_compute_pseudo);
} }
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
const struct gre_base_hdr *greh2; const struct gre_base_hdr *greh2;
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
......
...@@ -180,9 +180,9 @@ struct sk_buff *tcp_gso_segment(struct sk_buff *skb, ...@@ -180,9 +180,9 @@ struct sk_buff *tcp_gso_segment(struct sk_buff *skb,
return segs; return segs;
} }
struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb) struct sk_buff *tcp_gro_receive(struct list_head *head, struct sk_buff *skb)
{ {
struct sk_buff **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p; struct sk_buff *p;
struct tcphdr *th; struct tcphdr *th;
struct tcphdr *th2; struct tcphdr *th2;
...@@ -220,7 +220,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -220,7 +220,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb)
len = skb_gro_len(skb); len = skb_gro_len(skb);
flags = tcp_flag_word(th); flags = tcp_flag_word(th);
for (; (p = *head); head = &p->next) { list_for_each_entry(p, head, list) {
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
continue; continue;
...@@ -233,7 +233,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -233,7 +233,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb)
goto found; goto found;
} }
p = NULL;
goto out_check_final; goto out_check_final;
found: found:
...@@ -263,7 +263,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -263,7 +263,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb)
flush |= (len - 1) >= mss; flush |= (len - 1) >= mss;
flush |= (ntohl(th2->seq) + skb_gro_len(p)) ^ ntohl(th->seq); flush |= (ntohl(th2->seq) + skb_gro_len(p)) ^ ntohl(th->seq);
if (flush || skb_gro_receive(head, skb)) { if (flush || skb_gro_receive(p, skb)) {
mss = 1; mss = 1;
goto out_check_final; goto out_check_final;
} }
...@@ -277,7 +277,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb) ...@@ -277,7 +277,7 @@ struct sk_buff **tcp_gro_receive(struct sk_buff **head, struct sk_buff *skb)
TCP_FLAG_FIN)); TCP_FLAG_FIN));
if (p && (!NAPI_GRO_CB(skb)->same_flow || flush)) if (p && (!NAPI_GRO_CB(skb)->same_flow || flush))
pp = head; pp = p;
out: out:
NAPI_GRO_CB(skb)->flush |= (flush != 0); NAPI_GRO_CB(skb)->flush |= (flush != 0);
...@@ -302,7 +302,7 @@ int tcp_gro_complete(struct sk_buff *skb) ...@@ -302,7 +302,7 @@ int tcp_gro_complete(struct sk_buff *skb)
} }
EXPORT_SYMBOL(tcp_gro_complete); EXPORT_SYMBOL(tcp_gro_complete);
static struct sk_buff **tcp4_gro_receive(struct sk_buff **head, struct sk_buff *skb) static struct sk_buff *tcp4_gro_receive(struct list_head *head, struct sk_buff *skb)
{ {
/* Don't bother verifying checksum if we're going to flush anyway. */ /* Don't bother verifying checksum if we're going to flush anyway. */
if (!NAPI_GRO_CB(skb)->flush && if (!NAPI_GRO_CB(skb)->flush &&
......
...@@ -343,10 +343,11 @@ static struct sk_buff *udp4_ufo_fragment(struct sk_buff *skb, ...@@ -343,10 +343,11 @@ static struct sk_buff *udp4_ufo_fragment(struct sk_buff *skb,
return segs; return segs;
} }
struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, struct sk_buff *udp_gro_receive(struct list_head *head, struct sk_buff *skb,
struct udphdr *uh, udp_lookup_t lookup) struct udphdr *uh, udp_lookup_t lookup)
{ {
struct sk_buff *p, **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p;
struct udphdr *uh2; struct udphdr *uh2;
unsigned int off = skb_gro_offset(skb); unsigned int off = skb_gro_offset(skb);
int flush = 1; int flush = 1;
...@@ -371,7 +372,7 @@ struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, ...@@ -371,7 +372,7 @@ struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb,
unflush: unflush:
flush = 0; flush = 0;
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
if (!NAPI_GRO_CB(p)->same_flow) if (!NAPI_GRO_CB(p)->same_flow)
continue; continue;
...@@ -399,8 +400,8 @@ struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb, ...@@ -399,8 +400,8 @@ struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb,
} }
EXPORT_SYMBOL(udp_gro_receive); EXPORT_SYMBOL(udp_gro_receive);
static struct sk_buff **udp4_gro_receive(struct sk_buff **head, static struct sk_buff *udp4_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct udphdr *uh = udp_gro_udphdr(skb); struct udphdr *uh = udp_gro_udphdr(skb);
......
...@@ -49,8 +49,8 @@ static __u16 esp6_nexthdr_esp_offset(struct ipv6hdr *ipv6_hdr, int nhlen) ...@@ -49,8 +49,8 @@ static __u16 esp6_nexthdr_esp_offset(struct ipv6hdr *ipv6_hdr, int nhlen)
return 0; return 0;
} }
static struct sk_buff **esp6_gro_receive(struct sk_buff **head, static struct sk_buff *esp6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
int offset = skb_gro_offset(skb); int offset = skb_gro_offset(skb);
struct xfrm_offload *xo; struct xfrm_offload *xo;
......
...@@ -163,11 +163,11 @@ static int ipv6_exthdrs_len(struct ipv6hdr *iph, ...@@ -163,11 +163,11 @@ static int ipv6_exthdrs_len(struct ipv6hdr *iph,
return len; return len;
} }
static struct sk_buff **ipv6_gro_receive(struct sk_buff **head, static struct sk_buff *ipv6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
const struct net_offload *ops; const struct net_offload *ops;
struct sk_buff **pp = NULL; struct sk_buff *pp = NULL;
struct sk_buff *p; struct sk_buff *p;
struct ipv6hdr *iph; struct ipv6hdr *iph;
unsigned int nlen; unsigned int nlen;
...@@ -214,7 +214,7 @@ static struct sk_buff **ipv6_gro_receive(struct sk_buff **head, ...@@ -214,7 +214,7 @@ static struct sk_buff **ipv6_gro_receive(struct sk_buff **head,
flush--; flush--;
nlen = skb_network_header_len(skb); nlen = skb_network_header_len(skb);
for (p = *head; p; p = p->next) { list_for_each_entry(p, head, list) {
const struct ipv6hdr *iph2; const struct ipv6hdr *iph2;
__be32 first_word; /* <Version:4><Traffic_Class:8><Flow_Label:20> */ __be32 first_word; /* <Version:4><Traffic_Class:8><Flow_Label:20> */
...@@ -263,8 +263,8 @@ static struct sk_buff **ipv6_gro_receive(struct sk_buff **head, ...@@ -263,8 +263,8 @@ static struct sk_buff **ipv6_gro_receive(struct sk_buff **head,
return pp; return pp;
} }
static struct sk_buff **sit_ip6ip6_gro_receive(struct sk_buff **head, static struct sk_buff *sit_ip6ip6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
/* Common GRO receive for SIT and IP6IP6 */ /* Common GRO receive for SIT and IP6IP6 */
...@@ -278,8 +278,8 @@ static struct sk_buff **sit_ip6ip6_gro_receive(struct sk_buff **head, ...@@ -278,8 +278,8 @@ static struct sk_buff **sit_ip6ip6_gro_receive(struct sk_buff **head,
return ipv6_gro_receive(head, skb); return ipv6_gro_receive(head, skb);
} }
static struct sk_buff **ip4ip6_gro_receive(struct sk_buff **head, static struct sk_buff *ip4ip6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
/* Common GRO receive for SIT and IP6IP6 */ /* Common GRO receive for SIT and IP6IP6 */
......
...@@ -15,8 +15,8 @@ ...@@ -15,8 +15,8 @@
#include <net/ip6_checksum.h> #include <net/ip6_checksum.h>
#include "ip6_offload.h" #include "ip6_offload.h"
static struct sk_buff **tcp6_gro_receive(struct sk_buff **head, static struct sk_buff *tcp6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
/* Don't bother verifying checksum if we're going to flush anyway. */ /* Don't bother verifying checksum if we're going to flush anyway. */
if (!NAPI_GRO_CB(skb)->flush && if (!NAPI_GRO_CB(skb)->flush &&
......
...@@ -114,8 +114,8 @@ static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb, ...@@ -114,8 +114,8 @@ static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb,
return segs; return segs;
} }
static struct sk_buff **udp6_gro_receive(struct sk_buff **head, static struct sk_buff *udp6_gro_receive(struct list_head *head,
struct sk_buff *skb) struct sk_buff *skb)
{ {
struct udphdr *uh = udp_gro_udphdr(skb); struct udphdr *uh = udp_gro_udphdr(skb);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment