diff options
Diffstat (limited to 'net/core/netpoll.c')
-rw-r--r-- | net/core/netpoll.c | 36 |
1 files changed, 27 insertions, 9 deletions
diff --git a/net/core/netpoll.c b/net/core/netpoll.c index 9cb781830380..471da451cd48 100644 --- a/net/core/netpoll.c +++ b/net/core/netpoll.c | |||
@@ -54,6 +54,7 @@ static atomic_t trapped; | |||
54 | sizeof(struct iphdr) + sizeof(struct ethhdr)) | 54 | sizeof(struct iphdr) + sizeof(struct ethhdr)) |
55 | 55 | ||
56 | static void zap_completion_queue(void); | 56 | static void zap_completion_queue(void); |
57 | static void arp_reply(struct sk_buff *skb); | ||
57 | 58 | ||
58 | static void queue_process(void *p) | 59 | static void queue_process(void *p) |
59 | { | 60 | { |
@@ -153,6 +154,22 @@ static void poll_napi(struct netpoll *np) | |||
153 | } | 154 | } |
154 | } | 155 | } |
155 | 156 | ||
157 | static void service_arp_queue(struct netpoll_info *npi) | ||
158 | { | ||
159 | struct sk_buff *skb; | ||
160 | |||
161 | if (unlikely(!npi)) | ||
162 | return; | ||
163 | |||
164 | skb = skb_dequeue(&npi->arp_tx); | ||
165 | |||
166 | while (skb != NULL) { | ||
167 | arp_reply(skb); | ||
168 | skb = skb_dequeue(&npi->arp_tx); | ||
169 | } | ||
170 | return; | ||
171 | } | ||
172 | |||
156 | void netpoll_poll(struct netpoll *np) | 173 | void netpoll_poll(struct netpoll *np) |
157 | { | 174 | { |
158 | if(!np->dev || !netif_running(np->dev) || !np->dev->poll_controller) | 175 | if(!np->dev || !netif_running(np->dev) || !np->dev->poll_controller) |
@@ -163,6 +180,8 @@ void netpoll_poll(struct netpoll *np) | |||
163 | if (np->dev->poll) | 180 | if (np->dev->poll) |
164 | poll_napi(np); | 181 | poll_napi(np); |
165 | 182 | ||
183 | service_arp_queue(np->dev->npinfo); | ||
184 | |||
166 | zap_completion_queue(); | 185 | zap_completion_queue(); |
167 | } | 186 | } |
168 | 187 | ||
@@ -279,14 +298,10 @@ static void netpoll_send_skb(struct netpoll *np, struct sk_buff *skb) | |||
279 | * network drivers do not expect to be called if the queue is | 298 | * network drivers do not expect to be called if the queue is |
280 | * stopped. | 299 | * stopped. |
281 | */ | 300 | */ |
282 | if (netif_queue_stopped(np->dev)) { | 301 | status = NETDEV_TX_BUSY; |
283 | netif_tx_unlock(np->dev); | 302 | if (!netif_queue_stopped(np->dev)) |
284 | netpoll_poll(np); | 303 | status = np->dev->hard_start_xmit(skb, np->dev); |
285 | udelay(50); | ||
286 | continue; | ||
287 | } | ||
288 | 304 | ||
289 | status = np->dev->hard_start_xmit(skb, np->dev); | ||
290 | netif_tx_unlock(np->dev); | 305 | netif_tx_unlock(np->dev); |
291 | 306 | ||
292 | /* success */ | 307 | /* success */ |
@@ -446,7 +461,9 @@ int __netpoll_rx(struct sk_buff *skb) | |||
446 | int proto, len, ulen; | 461 | int proto, len, ulen; |
447 | struct iphdr *iph; | 462 | struct iphdr *iph; |
448 | struct udphdr *uh; | 463 | struct udphdr *uh; |
449 | struct netpoll *np = skb->dev->npinfo->rx_np; | 464 | struct netpoll_info *npi = skb->dev->npinfo; |
465 | struct netpoll *np = npi->rx_np; | ||
466 | |||
450 | 467 | ||
451 | if (!np) | 468 | if (!np) |
452 | goto out; | 469 | goto out; |
@@ -456,7 +473,7 @@ int __netpoll_rx(struct sk_buff *skb) | |||
456 | /* check if netpoll clients need ARP */ | 473 | /* check if netpoll clients need ARP */ |
457 | if (skb->protocol == __constant_htons(ETH_P_ARP) && | 474 | if (skb->protocol == __constant_htons(ETH_P_ARP) && |
458 | atomic_read(&trapped)) { | 475 | atomic_read(&trapped)) { |
459 | arp_reply(skb); | 476 | skb_queue_tail(&npi->arp_tx, skb); |
460 | return 1; | 477 | return 1; |
461 | } | 478 | } |
462 | 479 | ||
@@ -651,6 +668,7 @@ int netpoll_setup(struct netpoll *np) | |||
651 | npinfo->poll_owner = -1; | 668 | npinfo->poll_owner = -1; |
652 | npinfo->tries = MAX_RETRIES; | 669 | npinfo->tries = MAX_RETRIES; |
653 | spin_lock_init(&npinfo->rx_lock); | 670 | spin_lock_init(&npinfo->rx_lock); |
671 | skb_queue_head_init(&npinfo->arp_tx); | ||
654 | } else | 672 | } else |
655 | npinfo = ndev->npinfo; | 673 | npinfo = ndev->npinfo; |
656 | 674 | ||