mirror of
https://github.com/adulau/aha.git
synced 2025-01-04 23:23:18 +00:00
spidernet: fix racy double-free of skb
It appears that under certain circumstances, a race will result in a double-free of an skb. This patch null's out the skb pointer upon the skb free, avoiding the inadvertent deref of bogus data. The next patch fixes the actual race. Signed-off-by: Linas Vepstas <linas@austin.ibm.com> Cc: Jens Osterkamp <Jens.Osterkamp@de.ibm.com> Cc: Kou Ishizaki <kou.ishizaki@toshiba.co.jp> Signed-off-by: Jeff Garzik <jeff@garzik.org>
This commit is contained in:
parent
4cb6f9e57d
commit
d9c199ee78
1 changed files with 15 additions and 8 deletions
|
@ -396,10 +396,11 @@ spider_net_free_rx_chain_contents(struct spider_net_card *card)
|
||||||
descr = card->rx_chain.head;
|
descr = card->rx_chain.head;
|
||||||
do {
|
do {
|
||||||
if (descr->skb) {
|
if (descr->skb) {
|
||||||
dev_kfree_skb(descr->skb);
|
|
||||||
pci_unmap_single(card->pdev, descr->hwdescr->buf_addr,
|
pci_unmap_single(card->pdev, descr->hwdescr->buf_addr,
|
||||||
SPIDER_NET_MAX_FRAME,
|
SPIDER_NET_MAX_FRAME,
|
||||||
PCI_DMA_BIDIRECTIONAL);
|
PCI_DMA_BIDIRECTIONAL);
|
||||||
|
dev_kfree_skb(descr->skb);
|
||||||
|
descr->skb = NULL;
|
||||||
}
|
}
|
||||||
descr = descr->next;
|
descr = descr->next;
|
||||||
} while (descr != card->rx_chain.head);
|
} while (descr != card->rx_chain.head);
|
||||||
|
@ -453,6 +454,7 @@ spider_net_prepare_rx_descr(struct spider_net_card *card,
|
||||||
SPIDER_NET_MAX_FRAME, PCI_DMA_FROMDEVICE);
|
SPIDER_NET_MAX_FRAME, PCI_DMA_FROMDEVICE);
|
||||||
if (pci_dma_mapping_error(buf)) {
|
if (pci_dma_mapping_error(buf)) {
|
||||||
dev_kfree_skb_any(descr->skb);
|
dev_kfree_skb_any(descr->skb);
|
||||||
|
descr->skb = NULL;
|
||||||
if (netif_msg_rx_err(card) && net_ratelimit())
|
if (netif_msg_rx_err(card) && net_ratelimit())
|
||||||
pr_err("Could not iommu-map rx buffer\n");
|
pr_err("Could not iommu-map rx buffer\n");
|
||||||
card->spider_stats.rx_iommu_map_error++;
|
card->spider_stats.rx_iommu_map_error++;
|
||||||
|
@ -682,6 +684,7 @@ static int
|
||||||
spider_net_prepare_tx_descr(struct spider_net_card *card,
|
spider_net_prepare_tx_descr(struct spider_net_card *card,
|
||||||
struct sk_buff *skb)
|
struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
|
struct spider_net_descr_chain *chain = &card->tx_chain;
|
||||||
struct spider_net_descr *descr;
|
struct spider_net_descr *descr;
|
||||||
struct spider_net_hw_descr *hwdescr;
|
struct spider_net_hw_descr *hwdescr;
|
||||||
dma_addr_t buf;
|
dma_addr_t buf;
|
||||||
|
@ -696,10 +699,15 @@ spider_net_prepare_tx_descr(struct spider_net_card *card,
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_irqsave(&card->tx_chain.lock, flags);
|
spin_lock_irqsave(&chain->lock, flags);
|
||||||
descr = card->tx_chain.head;
|
descr = card->tx_chain.head;
|
||||||
|
if (descr->next == chain->tail->prev) {
|
||||||
|
spin_unlock_irqrestore(&chain->lock, flags);
|
||||||
|
pci_unmap_single(card->pdev, buf, skb->len, PCI_DMA_TODEVICE);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
hwdescr = descr->hwdescr;
|
hwdescr = descr->hwdescr;
|
||||||
card->tx_chain.head = descr->next;
|
chain->head = descr->next;
|
||||||
|
|
||||||
descr->skb = skb;
|
descr->skb = skb;
|
||||||
hwdescr->buf_addr = buf;
|
hwdescr->buf_addr = buf;
|
||||||
|
@ -709,7 +717,7 @@ spider_net_prepare_tx_descr(struct spider_net_card *card,
|
||||||
|
|
||||||
hwdescr->dmac_cmd_status =
|
hwdescr->dmac_cmd_status =
|
||||||
SPIDER_NET_DESCR_CARDOWNED | SPIDER_NET_DMAC_NOCS;
|
SPIDER_NET_DESCR_CARDOWNED | SPIDER_NET_DMAC_NOCS;
|
||||||
spin_unlock_irqrestore(&card->tx_chain.lock, flags);
|
spin_unlock_irqrestore(&chain->lock, flags);
|
||||||
|
|
||||||
if (skb->protocol == htons(ETH_P_IP))
|
if (skb->protocol == htons(ETH_P_IP))
|
||||||
switch (skb->nh.iph->protocol) {
|
switch (skb->nh.iph->protocol) {
|
||||||
|
@ -838,6 +846,7 @@ spider_net_release_tx_chain(struct spider_net_card *card, int brutal)
|
||||||
chain->tail = descr->next;
|
chain->tail = descr->next;
|
||||||
hwdescr->dmac_cmd_status |= SPIDER_NET_DESCR_NOT_IN_USE;
|
hwdescr->dmac_cmd_status |= SPIDER_NET_DESCR_NOT_IN_USE;
|
||||||
skb = descr->skb;
|
skb = descr->skb;
|
||||||
|
descr->skb = NULL;
|
||||||
buf_addr = hwdescr->buf_addr;
|
buf_addr = hwdescr->buf_addr;
|
||||||
spin_unlock_irqrestore(&chain->lock, flags);
|
spin_unlock_irqrestore(&chain->lock, flags);
|
||||||
|
|
||||||
|
@ -903,13 +912,10 @@ spider_net_xmit(struct sk_buff *skb, struct net_device *netdev)
|
||||||
{
|
{
|
||||||
int cnt;
|
int cnt;
|
||||||
struct spider_net_card *card = netdev_priv(netdev);
|
struct spider_net_card *card = netdev_priv(netdev);
|
||||||
struct spider_net_descr_chain *chain = &card->tx_chain;
|
|
||||||
|
|
||||||
spider_net_release_tx_chain(card, 0);
|
spider_net_release_tx_chain(card, 0);
|
||||||
|
|
||||||
if ((chain->head->next == chain->tail->prev) ||
|
if (spider_net_prepare_tx_descr(card, skb) != 0) {
|
||||||
(spider_net_prepare_tx_descr(card, skb) != 0)) {
|
|
||||||
|
|
||||||
card->netdev_stats.tx_dropped++;
|
card->netdev_stats.tx_dropped++;
|
||||||
netif_stop_queue(netdev);
|
netif_stop_queue(netdev);
|
||||||
return NETDEV_TX_BUSY;
|
return NETDEV_TX_BUSY;
|
||||||
|
@ -1127,6 +1133,7 @@ spider_net_decode_one_descr(struct spider_net_card *card)
|
||||||
|
|
||||||
bad_desc:
|
bad_desc:
|
||||||
dev_kfree_skb_irq(descr->skb);
|
dev_kfree_skb_irq(descr->skb);
|
||||||
|
descr->skb = NULL;
|
||||||
hwdescr->dmac_cmd_status = SPIDER_NET_DESCR_NOT_IN_USE;
|
hwdescr->dmac_cmd_status = SPIDER_NET_DESCR_NOT_IN_USE;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue