aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichal Kazior <michal.kazior@tieto.com>2014-08-22 08:23:33 -0400
committerKalle Valo <kvalo@qca.qualcomm.com>2014-08-25 04:28:56 -0400
commitec5ba4d3b6b60456b067e8c625e87e67cdde2d12 (patch)
tree5efdd7d536de18235039627f75f009aff06c2738
parent145cc1214a271c72b81a064f4d65c3cf612e941e (diff)
ath10k: make sure to really disable irqs
This fixes two corner cases. One is a race between disabling copy engine interrupts and unhandled pending interrupts on the host. This could end up with a runaway tasklet and consequently memory leak of a few copy engine rx buffers. The other one is an unexpected (and non-maskable via device CSR) MSI fw indication interrupt during teardown. This could trigger the same problem as the first corner case. Signed-off-by: Michal Kazior <michal.kazior@tieto.com> Signed-off-by: Kalle Valo <kvalo@qca.qualcomm.com>
-rw-r--r--drivers/net/wireless/ath/ath10k/pci.c43
1 files changed, 37 insertions, 6 deletions
diff --git a/drivers/net/wireless/ath/ath10k/pci.c b/drivers/net/wireless/ath/ath10k/pci.c
index c764dd713f00..6224952ba523 100644
--- a/drivers/net/wireless/ath/ath10k/pci.c
+++ b/drivers/net/wireless/ath/ath10k/pci.c
@@ -1121,6 +1121,40 @@ static int ath10k_pci_post_rx(struct ath10k *ar)
1121 return 0; 1121 return 0;
1122} 1122}
1123 1123
1124static void ath10k_pci_irq_disable(struct ath10k *ar)
1125{
1126 struct ath10k_pci *ar_pci = ath10k_pci_priv(ar);
1127 int i;
1128
1129 ath10k_ce_disable_interrupts(ar);
1130
1131 /* Regardless how many interrupts were assigned for MSI the first one
1132 * is always used for firmware indications (crashes). There's no way to
1133 * mask the irq in the device so call disable_irq(). Legacy (shared)
1134 * interrupts can be masked on the device though.
1135 */
1136 if (ar_pci->num_msi_intrs > 0)
1137 disable_irq(ar_pci->pdev->irq);
1138 else
1139 ath10k_pci_disable_and_clear_legacy_irq(ar);
1140
1141 for (i = 0; i < max(1, ar_pci->num_msi_intrs); i++)
1142 synchronize_irq(ar_pci->pdev->irq + i);
1143}
1144
1145static void ath10k_pci_irq_enable(struct ath10k *ar)
1146{
1147 struct ath10k_pci *ar_pci = ath10k_pci_priv(ar);
1148
1149 ath10k_ce_enable_interrupts(ar);
1150
1151 /* See comment in ath10k_pci_irq_disable() */
1152 if (ar_pci->num_msi_intrs > 0)
1153 enable_irq(ar_pci->pdev->irq);
1154 else
1155 ath10k_pci_enable_legacy_irq(ar);
1156}
1157
1124static int ath10k_pci_hif_start(struct ath10k *ar) 1158static int ath10k_pci_hif_start(struct ath10k *ar)
1125{ 1159{
1126 struct ath10k_pci *ar_pci = ath10k_pci_priv(ar); 1160 struct ath10k_pci *ar_pci = ath10k_pci_priv(ar);
@@ -1138,7 +1172,7 @@ static int ath10k_pci_hif_start(struct ath10k *ar)
1138 goto err_early_irq; 1172 goto err_early_irq;
1139 } 1173 }
1140 1174
1141 ath10k_ce_enable_interrupts(ar); 1175 ath10k_pci_irq_enable(ar);
1142 1176
1143 /* Post buffers once to start things off. */ 1177 /* Post buffers once to start things off. */
1144 ret = ath10k_pci_post_rx(ar); 1178 ret = ath10k_pci_post_rx(ar);
@@ -1152,7 +1186,7 @@ static int ath10k_pci_hif_start(struct ath10k *ar)
1152 return 0; 1186 return 0;
1153 1187
1154err_stop: 1188err_stop:
1155 ath10k_ce_disable_interrupts(ar); 1189 ath10k_pci_irq_disable(ar);
1156 ath10k_pci_free_irq(ar); 1190 ath10k_pci_free_irq(ar);
1157 ath10k_pci_kill_tasklet(ar); 1191 ath10k_pci_kill_tasklet(ar);
1158err_early_irq: 1192err_early_irq:
@@ -1275,10 +1309,7 @@ static void ath10k_pci_hif_stop(struct ath10k *ar)
1275 if (WARN_ON(!ar_pci->started)) 1309 if (WARN_ON(!ar_pci->started))
1276 return; 1310 return;
1277 1311
1278 ret = ath10k_ce_disable_interrupts(ar); 1312 ath10k_pci_irq_disable(ar);
1279 if (ret)
1280 ath10k_warn("failed to disable CE interrupts: %d\n", ret);
1281
1282 ath10k_pci_free_irq(ar); 1313 ath10k_pci_free_irq(ar);
1283 ath10k_pci_kill_tasklet(ar); 1314 ath10k_pci_kill_tasklet(ar);
1284 1315