diff options
author | Bob Copeland <me@bobcopeland.com> | 2009-01-10 14:42:54 -0500 |
---|---|---|
committer | John W. Linville <linville@tuxdriver.com> | 2009-02-11 11:27:16 -0500 |
commit | fcf6b1bca8cdfefc986909b57277af4628955bd8 (patch) | |
tree | 7a8000cbdeb8deb45f5139d179b2d243586e1321 /drivers/net | |
parent | 7b7a799d664a46eec6cb7de200c90f40730497a7 (diff) |
ath5k: fix bf->skb==NULL panic in ath5k_tasklet_rx
Under memory pressure, we may not be able to allocate a new skb for
new packets. If the allocation fails, ath5k_tasklet_rx will exit but
will leave a buffer in the list with a NULL skb, eventually triggering
a BUG_ON.
Extract the skb allocation from ath5k_rxbuf_setup() and change the
tasklet to allocate the next skb before accepting a packet.
Changes-licensed-under: 3-Clause-BSD
Signed-off-by: Bob Copeland <me@bobcopeland.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/wireless/ath5k/base.c | 85 |
1 files changed, 53 insertions, 32 deletions
diff --git a/drivers/net/wireless/ath5k/base.c b/drivers/net/wireless/ath5k/base.c index a533ed60bb4d..1d77ee9d6e99 100644 --- a/drivers/net/wireless/ath5k/base.c +++ b/drivers/net/wireless/ath5k/base.c | |||
@@ -1098,6 +1098,42 @@ ath5k_hw_to_driver_rix(struct ath5k_softc *sc, int hw_rix) | |||
1098 | * Buffers setup * | 1098 | * Buffers setup * |
1099 | \***************/ | 1099 | \***************/ |
1100 | 1100 | ||
1101 | static | ||
1102 | struct sk_buff *ath5k_rx_skb_alloc(struct ath5k_softc *sc, dma_addr_t *skb_addr) | ||
1103 | { | ||
1104 | struct sk_buff *skb; | ||
1105 | unsigned int off; | ||
1106 | |||
1107 | /* | ||
1108 | * Allocate buffer with headroom_needed space for the | ||
1109 | * fake physical layer header at the start. | ||
1110 | */ | ||
1111 | skb = dev_alloc_skb(sc->rxbufsize + sc->cachelsz - 1); | ||
1112 | |||
1113 | if (!skb) { | ||
1114 | ATH5K_ERR(sc, "can't alloc skbuff of size %u\n", | ||
1115 | sc->rxbufsize + sc->cachelsz - 1); | ||
1116 | return NULL; | ||
1117 | } | ||
1118 | /* | ||
1119 | * Cache-line-align. This is important (for the | ||
1120 | * 5210 at least) as not doing so causes bogus data | ||
1121 | * in rx'd frames. | ||
1122 | */ | ||
1123 | off = ((unsigned long)skb->data) % sc->cachelsz; | ||
1124 | if (off != 0) | ||
1125 | skb_reserve(skb, sc->cachelsz - off); | ||
1126 | |||
1127 | *skb_addr = pci_map_single(sc->pdev, | ||
1128 | skb->data, sc->rxbufsize, PCI_DMA_FROMDEVICE); | ||
1129 | if (unlikely(pci_dma_mapping_error(sc->pdev, *skb_addr))) { | ||
1130 | ATH5K_ERR(sc, "%s: DMA mapping failed\n", __func__); | ||
1131 | dev_kfree_skb(skb); | ||
1132 | return NULL; | ||
1133 | } | ||
1134 | return skb; | ||
1135 | } | ||
1136 | |||
1101 | static int | 1137 | static int |
1102 | ath5k_rxbuf_setup(struct ath5k_softc *sc, struct ath5k_buf *bf) | 1138 | ath5k_rxbuf_setup(struct ath5k_softc *sc, struct ath5k_buf *bf) |
1103 | { | 1139 | { |
@@ -1105,37 +1141,11 @@ ath5k_rxbuf_setup(struct ath5k_softc *sc, struct ath5k_buf *bf) | |||
1105 | struct sk_buff *skb = bf->skb; | 1141 | struct sk_buff *skb = bf->skb; |
1106 | struct ath5k_desc *ds; | 1142 | struct ath5k_desc *ds; |
1107 | 1143 | ||
1108 | if (likely(skb == NULL)) { | 1144 | if (!skb) { |
1109 | unsigned int off; | 1145 | skb = ath5k_rx_skb_alloc(sc, &bf->skbaddr); |
1110 | 1146 | if (!skb) | |
1111 | /* | ||
1112 | * Allocate buffer with headroom_needed space for the | ||
1113 | * fake physical layer header at the start. | ||
1114 | */ | ||
1115 | skb = dev_alloc_skb(sc->rxbufsize + sc->cachelsz - 1); | ||
1116 | if (unlikely(skb == NULL)) { | ||
1117 | ATH5K_ERR(sc, "can't alloc skbuff of size %u\n", | ||
1118 | sc->rxbufsize + sc->cachelsz - 1); | ||
1119 | return -ENOMEM; | 1147 | return -ENOMEM; |
1120 | } | ||
1121 | /* | ||
1122 | * Cache-line-align. This is important (for the | ||
1123 | * 5210 at least) as not doing so causes bogus data | ||
1124 | * in rx'd frames. | ||
1125 | */ | ||
1126 | off = ((unsigned long)skb->data) % sc->cachelsz; | ||
1127 | if (off != 0) | ||
1128 | skb_reserve(skb, sc->cachelsz - off); | ||
1129 | |||
1130 | bf->skb = skb; | 1148 | bf->skb = skb; |
1131 | bf->skbaddr = pci_map_single(sc->pdev, | ||
1132 | skb->data, sc->rxbufsize, PCI_DMA_FROMDEVICE); | ||
1133 | if (unlikely(pci_dma_mapping_error(sc->pdev, bf->skbaddr))) { | ||
1134 | ATH5K_ERR(sc, "%s: DMA mapping failed\n", __func__); | ||
1135 | dev_kfree_skb(skb); | ||
1136 | bf->skb = NULL; | ||
1137 | return -ENOMEM; | ||
1138 | } | ||
1139 | } | 1149 | } |
1140 | 1150 | ||
1141 | /* | 1151 | /* |
@@ -1664,7 +1674,8 @@ ath5k_tasklet_rx(unsigned long data) | |||
1664 | { | 1674 | { |
1665 | struct ieee80211_rx_status rxs = {}; | 1675 | struct ieee80211_rx_status rxs = {}; |
1666 | struct ath5k_rx_status rs = {}; | 1676 | struct ath5k_rx_status rs = {}; |
1667 | struct sk_buff *skb; | 1677 | struct sk_buff *skb, *next_skb; |
1678 | dma_addr_t next_skb_addr; | ||
1668 | struct ath5k_softc *sc = (void *)data; | 1679 | struct ath5k_softc *sc = (void *)data; |
1669 | struct ath5k_buf *bf, *bf_last; | 1680 | struct ath5k_buf *bf, *bf_last; |
1670 | struct ath5k_desc *ds; | 1681 | struct ath5k_desc *ds; |
@@ -1749,10 +1760,17 @@ ath5k_tasklet_rx(unsigned long data) | |||
1749 | goto next; | 1760 | goto next; |
1750 | } | 1761 | } |
1751 | accept: | 1762 | accept: |
1763 | next_skb = ath5k_rx_skb_alloc(sc, &next_skb_addr); | ||
1764 | |||
1765 | /* | ||
1766 | * If we can't replace bf->skb with a new skb under memory | ||
1767 | * pressure, just skip this packet | ||
1768 | */ | ||
1769 | if (!next_skb) | ||
1770 | goto next; | ||
1771 | |||
1752 | pci_unmap_single(sc->pdev, bf->skbaddr, sc->rxbufsize, | 1772 | pci_unmap_single(sc->pdev, bf->skbaddr, sc->rxbufsize, |
1753 | PCI_DMA_FROMDEVICE); | 1773 | PCI_DMA_FROMDEVICE); |
1754 | bf->skb = NULL; | ||
1755 | |||
1756 | skb_put(skb, rs.rs_datalen); | 1774 | skb_put(skb, rs.rs_datalen); |
1757 | 1775 | ||
1758 | /* The MAC header is padded to have 32-bit boundary if the | 1776 | /* The MAC header is padded to have 32-bit boundary if the |
@@ -1825,6 +1843,9 @@ accept: | |||
1825 | ath5k_check_ibss_tsf(sc, skb, &rxs); | 1843 | ath5k_check_ibss_tsf(sc, skb, &rxs); |
1826 | 1844 | ||
1827 | __ieee80211_rx(sc->hw, skb, &rxs); | 1845 | __ieee80211_rx(sc->hw, skb, &rxs); |
1846 | |||
1847 | bf->skb = next_skb; | ||
1848 | bf->skbaddr = next_skb_addr; | ||
1828 | next: | 1849 | next: |
1829 | list_move_tail(&bf->list, &sc->rxbuf); | 1850 | list_move_tail(&bf->list, &sc->rxbuf); |
1830 | } while (ath5k_rxbuf_setup(sc, bf) == 0); | 1851 | } while (ath5k_rxbuf_setup(sc, bf) == 0); |