#ifndef _NET_GRO_CELLS_H #define _NET_GRO_CELLS_H #include <linux/skbuff.h> #include <linux/slab.h> #include <linux/netdevice.h> struct gro_cell { struct sk_buff_head napi_skbs; struct napi_struct napi; }; struct gro_cells { struct gro_cell __percpu *cells; }; static inline int gro_cells_receive(struct gro_cells *gcells, struct sk_buff *skb) { struct gro_cell *cell; struct net_device *dev = skb->dev; if (!gcells->cells || skb_cloned(skb) || !(dev->features & NETIF_F_GRO)) return netif_rx(skb); cell = this_cpu_ptr(gcells->cells); if (skb_queue_len(&cell->napi_skbs) > netdev_max_backlog) { atomic_long_inc(&dev->rx_dropped); kfree_skb(skb); return NET_RX_DROP; } __skb_queue_tail(&cell->napi_skbs, skb); if (skb_queue_len(&cell->napi_skbs) == 1) napi_schedule(&cell->napi); return NET_RX_SUCCESS; } /* called under BH context */ static inline int gro_cell_poll(struct napi_struct *napi, int budget) { struct gro_cell *cell = container_of(napi, struct gro_cell, napi); struct sk_buff *skb; int work_done = 0; while (work_done < budget) { skb = __skb_dequeue(&cell->napi_skbs); if (!skb) break; napi_gro_receive(napi, skb); work_done++; } if (work_done < budget) napi_complete_done(napi, work_done); return work_done; } static inline int gro_cells_init(struct gro_cells *gcells, struct net_device *dev) { int i; gcells->cells = alloc_percpu(struct gro_cell); if (!gcells->cells) return -ENOMEM; for_each_possible_cpu(i) { struct gro_cell *cell = per_cpu_ptr(gcells->cells, i); __skb_queue_head_init(&cell->napi_skbs); set_bit(NAPI_STATE_NO_BUSY_POLL, &cell->napi.state); netif_napi_add(dev, &cell->napi, gro_cell_poll, 64); napi_enable(&cell->napi); } return 0; } static inline void gro_cells_destroy(struct gro_cells *gcells) { int i; if (!gcells->cells) return; for_each_possible_cpu(i) { struct gro_cell *cell = per_cpu_ptr(gcells->cells, i); netif_napi_del(&cell->napi); __skb_queue_purge(&cell->napi_skbs); } free_percpu(gcells->cells); gcells->cells = NULL; } #endif dace6c224e9fc0e81337b6fee3cfd136824c983'>diff</a></td><td class='form'><form class='right' method='get' action='/cgit.cgi/linux/net-next.git/log/drivers/usb/dwc2/core_intr.c'> <input type='hidden' name='id' value='fdace6c224e9fc0e81337b6fee3cfd136824c983'/><select name='qt'> <option value='grep'>log msg</option> <option value='author'>author</option> <option value='committer'>committer</option> <option value='range'>range</option> </select> <input class='txt' type='search' size='10' name='q' value=''/> <input type='submit' value='search'/> </form> </td></tr></table> <div class='path'>path: <a href='/cgit.cgi/linux/net-next.git/log/?id=fdace6c224e9fc0e81337b6fee3cfd136824c983'>root</a>/<a href='/cgit.cgi/linux/net-next.git/log/drivers?id=fdace6c224e9fc0e81337b6fee3cfd136824c983'>drivers</a>/<a href='/cgit.cgi/linux/net-next.git/log/drivers/usb?id=fdace6c224e9fc0e81337b6fee3cfd136824c983'>usb</a>/<a href='/cgit.cgi/linux/net-next.git/log/drivers/usb/dwc2?id=fdace6c224e9fc0e81337b6fee3cfd136824c983'>dwc2</a>/<a href='/cgit.cgi/linux/net-next.git/log/drivers/usb/dwc2/core_intr.c?id=fdace6c224e9fc0e81337b6fee3cfd136824c983'>core_intr.c</a></div><div class='content'><table class='list nowrap'><tr class='nohover'><th class='left'>Age</th><th class='left'>Commit message (<a href='/cgit.cgi/linux/net-next.git/log/drivers/usb/dwc2/core_intr.c?id=fdace6c224e9fc0e81337b6fee3cfd136824c983&showmsg=1'>Expand</a>)</th><th class='left'>Author</th><th class='left'>Files</th><th class='left'>Lines</th></tr>