| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | #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; | 
					
						
							|  |  |  | } ____cacheline_aligned_in_smp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct gro_cells { | 
					
						
							|  |  |  | 	unsigned int		gro_cells_mask; | 
					
						
							|  |  |  | 	struct gro_cell		*cells; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void gro_cells_receive(struct gro_cells *gcells, struct sk_buff *skb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct gro_cell *cell = gcells->cells; | 
					
						
							|  |  |  | 	struct net_device *dev = skb->dev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!cell || skb_cloned(skb) || !(dev->features & NETIF_F_GRO)) { | 
					
						
							|  |  |  | 		netif_rx(skb); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (skb_rx_queue_recorded(skb)) | 
					
						
							|  |  |  | 		cell += skb_get_rx_queue(skb) & gcells->gro_cells_mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (skb_queue_len(&cell->napi_skbs) > netdev_max_backlog) { | 
					
						
							|  |  |  | 		atomic_long_inc(&dev->rx_dropped); | 
					
						
							|  |  |  | 		kfree_skb(skb); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 	/* We run in BH context */ | 
					
						
							|  |  |  | 	spin_lock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	__skb_queue_tail(&cell->napi_skbs, skb); | 
					
						
							|  |  |  | 	if (skb_queue_len(&cell->napi_skbs) == 1) | 
					
						
							|  |  |  | 		napi_schedule(&cell->napi); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 	spin_unlock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | /* called unser BH context */ | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 	spin_lock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 	while (work_done < budget) { | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 		skb = __skb_dequeue(&cell->napi_skbs); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 		if (!skb) | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 		spin_unlock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 		napi_gro_receive(napi, skb); | 
					
						
							|  |  |  | 		work_done++; | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 		spin_lock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (work_done < budget) | 
					
						
							|  |  |  | 		napi_complete(napi); | 
					
						
							| 
									
										
										
										
											2012-12-10 12:32:03 +00:00
										 |  |  | 	spin_unlock(&cell->napi_skbs.lock); | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 	return work_done; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int gro_cells_init(struct gro_cells *gcells, struct net_device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gcells->gro_cells_mask = roundup_pow_of_two(netif_get_num_default_rss_queues()) - 1; | 
					
						
							| 
									
										
										
										
											2013-01-26 09:24:19 +00:00
										 |  |  | 	gcells->cells = kcalloc(gcells->gro_cells_mask + 1, | 
					
						
							|  |  |  | 				sizeof(struct gro_cell), | 
					
						
							| 
									
										
										
										
											2012-09-27 19:29:05 +00:00
										 |  |  | 				GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!gcells->cells) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i <= gcells->gro_cells_mask; i++) { | 
					
						
							|  |  |  | 		struct gro_cell *cell = gcells->cells + i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		skb_queue_head_init(&cell->napi_skbs); | 
					
						
							|  |  |  | 		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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct gro_cell *cell = gcells->cells; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!cell) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	for (i = 0; i <= gcells->gro_cells_mask; i++,cell++) { | 
					
						
							|  |  |  | 		netif_napi_del(&cell->napi); | 
					
						
							|  |  |  | 		skb_queue_purge(&cell->napi_skbs); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	kfree(gcells->cells); | 
					
						
							|  |  |  | 	gcells->cells = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |