aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'drivers')
-rw-r--r--drivers/net/wireless/iwlwifi/iwl-tx.c53
1 files changed, 40 insertions, 13 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-tx.c b/drivers/net/wireless/iwlwifi/iwl-tx.c
index 37ebcff409e2..87c955e88427 100644
--- a/drivers/net/wireless/iwlwifi/iwl-tx.c
+++ b/drivers/net/wireless/iwlwifi/iwl-tx.c
@@ -72,10 +72,6 @@ static int iwl_hw_txq_free_tfd(struct iwl_priv *priv, struct iwl_tx_queue *txq)
72 int counter = 0; 72 int counter = 0;
73 int index, is_odd; 73 int index, is_odd;
74 74
75 /* Host command buffers stay mapped in memory, nothing to clean */
76 if (txq->q.id == IWL_CMD_QUEUE_NUM)
77 return 0;
78
79 /* Sanity check on number of chunks */ 75 /* Sanity check on number of chunks */
80 counter = IWL_GET_BITS(*bd, num_tbs); 76 counter = IWL_GET_BITS(*bd, num_tbs);
81 if (counter > MAX_NUM_OF_TBS) { 77 if (counter > MAX_NUM_OF_TBS) {
@@ -210,7 +206,7 @@ static void iwl_tx_queue_free(struct iwl_priv *priv, int txq_id)
210 struct iwl_tx_queue *txq = &priv->txq[txq_id]; 206 struct iwl_tx_queue *txq = &priv->txq[txq_id];
211 struct iwl_queue *q = &txq->q; 207 struct iwl_queue *q = &txq->q;
212 struct pci_dev *dev = priv->pci_dev; 208 struct pci_dev *dev = priv->pci_dev;
213 int i, slots_num, len; 209 int i, len;
214 210
215 if (q->n_bd == 0) 211 if (q->n_bd == 0)
216 return; 212 return;
@@ -221,16 +217,10 @@ static void iwl_tx_queue_free(struct iwl_priv *priv, int txq_id)
221 iwl_hw_txq_free_tfd(priv, txq); 217 iwl_hw_txq_free_tfd(priv, txq);
222 218
223 len = sizeof(struct iwl_cmd) * q->n_window; 219 len = sizeof(struct iwl_cmd) * q->n_window;
224 if (q->id == IWL_CMD_QUEUE_NUM)
225 len += IWL_MAX_SCAN_SIZE;
226 220
227 /* De-alloc array of command/tx buffers */ 221 /* De-alloc array of command/tx buffers */
228 slots_num = (txq_id == IWL_CMD_QUEUE_NUM) ? 222 for (i = 0; i < TFD_TX_CMD_SLOTS; i++)
229 TFD_CMD_SLOTS : TFD_TX_CMD_SLOTS;
230 for (i = 0; i < slots_num; i++)
231 kfree(txq->cmd[i]); 223 kfree(txq->cmd[i]);
232 if (txq_id == IWL_CMD_QUEUE_NUM)
233 kfree(txq->cmd[slots_num]);
234 224
235 /* De-alloc circular buffer of TFDs */ 225 /* De-alloc circular buffer of TFDs */
236 if (txq->q.n_bd) 226 if (txq->q.n_bd)
@@ -245,6 +235,40 @@ static void iwl_tx_queue_free(struct iwl_priv *priv, int txq_id)
245 memset(txq, 0, sizeof(*txq)); 235 memset(txq, 0, sizeof(*txq));
246} 236}
247 237
238
239/**
240 * iwl_cmd_queue_free - Deallocate DMA queue.
241 * @txq: Transmit queue to deallocate.
242 *
243 * Empty queue by removing and destroying all BD's.
244 * Free all buffers.
245 * 0-fill, but do not free "txq" descriptor structure.
246 */
247static void iwl_cmd_queue_free(struct iwl_priv *priv)
248{
249 struct iwl_tx_queue *txq = &priv->txq[IWL_CMD_QUEUE_NUM];
250 struct iwl_queue *q = &txq->q;
251 struct pci_dev *dev = priv->pci_dev;
252 int i, len;
253
254 if (q->n_bd == 0)
255 return;
256
257 len = sizeof(struct iwl_cmd) * q->n_window;
258 len += IWL_MAX_SCAN_SIZE;
259
260 /* De-alloc array of command/tx buffers */
261 for (i = 0; i <= TFD_CMD_SLOTS; i++)
262 kfree(txq->cmd[i]);
263
264 /* De-alloc circular buffer of TFDs */
265 if (txq->q.n_bd)
266 pci_free_consistent(dev, sizeof(struct iwl_tfd_frame) *
267 txq->q.n_bd, txq->bd, txq->q.dma_addr);
268
269 /* 0-fill queue descriptor structure */
270 memset(txq, 0, sizeof(*txq));
271}
248/*************** DMA-QUEUE-GENERAL-FUNCTIONS ***** 272/*************** DMA-QUEUE-GENERAL-FUNCTIONS *****
249 * DMA services 273 * DMA services
250 * 274 *
@@ -468,7 +492,10 @@ void iwl_hw_txq_ctx_free(struct iwl_priv *priv)
468 492
469 /* Tx queues */ 493 /* Tx queues */
470 for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++) 494 for (txq_id = 0; txq_id < priv->hw_params.max_txq_num; txq_id++)
471 iwl_tx_queue_free(priv, txq_id); 495 if (txq_id == IWL_CMD_QUEUE_NUM)
496 iwl_cmd_queue_free(priv);
497 else
498 iwl_tx_queue_free(priv, txq_id);
472 499
473 /* Keep-warm buffer */ 500 /* Keep-warm buffer */
474 iwl_kw_free(priv); 501 iwl_kw_free(priv);