diff options
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl-3945.c | 117 | ||||
-rw-r--r-- | drivers/net/wireless/iwlwifi/iwl3945-base.c | 188 |
2 files changed, 148 insertions, 157 deletions
diff --git a/drivers/net/wireless/iwlwifi/iwl-3945.c b/drivers/net/wireless/iwlwifi/iwl-3945.c index 49b781104327..dc3d695bf096 100644 --- a/drivers/net/wireless/iwlwifi/iwl-3945.c +++ b/drivers/net/wireless/iwlwifi/iwl-3945.c | |||
@@ -227,14 +227,126 @@ __le32 iwl3945_get_antenna_flags(const struct iwl3945_priv *priv) | |||
227 | return 0; /* "diversity" is default if error */ | 227 | return 0; /* "diversity" is default if error */ |
228 | } | 228 | } |
229 | 229 | ||
230 | #ifdef CONFIG_IWL3945_DEBUG | ||
231 | #define TX_STATUS_ENTRY(x) case TX_STATUS_FAIL_ ## x: return #x | ||
232 | |||
233 | static const char *iwl3945_get_tx_fail_reason(u32 status) | ||
234 | { | ||
235 | switch (status & TX_STATUS_MSK) { | ||
236 | case TX_STATUS_SUCCESS: | ||
237 | return "SUCCESS"; | ||
238 | TX_STATUS_ENTRY(SHORT_LIMIT); | ||
239 | TX_STATUS_ENTRY(LONG_LIMIT); | ||
240 | TX_STATUS_ENTRY(FIFO_UNDERRUN); | ||
241 | TX_STATUS_ENTRY(MGMNT_ABORT); | ||
242 | TX_STATUS_ENTRY(NEXT_FRAG); | ||
243 | TX_STATUS_ENTRY(LIFE_EXPIRE); | ||
244 | TX_STATUS_ENTRY(DEST_PS); | ||
245 | TX_STATUS_ENTRY(ABORTED); | ||
246 | TX_STATUS_ENTRY(BT_RETRY); | ||
247 | TX_STATUS_ENTRY(STA_INVALID); | ||
248 | TX_STATUS_ENTRY(FRAG_DROPPED); | ||
249 | TX_STATUS_ENTRY(TID_DISABLE); | ||
250 | TX_STATUS_ENTRY(FRAME_FLUSHED); | ||
251 | TX_STATUS_ENTRY(INSUFFICIENT_CF_POLL); | ||
252 | TX_STATUS_ENTRY(TX_LOCKED); | ||
253 | TX_STATUS_ENTRY(NO_BEACON_ON_RADAR); | ||
254 | } | ||
255 | |||
256 | return "UNKNOWN"; | ||
257 | } | ||
258 | #else | ||
259 | static inline const char *iwl3945_get_tx_fail_reason(u32 status) | ||
260 | { | ||
261 | return ""; | ||
262 | } | ||
263 | #endif | ||
264 | |||
265 | |||
266 | /** | ||
267 | * iwl3945_tx_queue_reclaim - Reclaim Tx queue entries already Tx'd | ||
268 | * | ||
269 | * When FW advances 'R' index, all entries between old and new 'R' index | ||
270 | * need to be reclaimed. As result, some free space forms. If there is | ||
271 | * enough free space (> low mark), wake the stack that feeds us. | ||
272 | */ | ||
273 | static void iwl3945_tx_queue_reclaim(struct iwl3945_priv *priv, | ||
274 | int txq_id, int index) | ||
275 | { | ||
276 | struct iwl3945_tx_queue *txq = &priv->txq[txq_id]; | ||
277 | struct iwl3945_queue *q = &txq->q; | ||
278 | struct iwl3945_tx_info *tx_info; | ||
279 | |||
280 | BUG_ON(txq_id == IWL_CMD_QUEUE_NUM); | ||
281 | |||
282 | for (index = iwl_queue_inc_wrap(index, q->n_bd); q->read_ptr != index; | ||
283 | q->read_ptr = iwl_queue_inc_wrap(q->read_ptr, q->n_bd)) { | ||
284 | |||
285 | tx_info = &txq->txb[txq->q.read_ptr]; | ||
286 | ieee80211_tx_status(priv->hw, tx_info->skb[0], | ||
287 | &tx_info->status); | ||
288 | tx_info->skb[0] = NULL; | ||
289 | iwl3945_hw_txq_free_tfd(priv, txq); | ||
290 | } | ||
291 | |||
292 | if (iwl3945_queue_space(q) > q->low_mark && (txq_id >= 0) && | ||
293 | (txq_id != IWL_CMD_QUEUE_NUM) && | ||
294 | priv->mac80211_registered) | ||
295 | ieee80211_wake_queue(priv->hw, txq_id); | ||
296 | } | ||
297 | |||
298 | /** | ||
299 | * iwl3945_rx_reply_tx - Handle Tx response | ||
300 | */ | ||
301 | static void iwl3945_rx_reply_tx(struct iwl3945_priv *priv, | ||
302 | struct iwl3945_rx_mem_buffer *rxb) | ||
303 | { | ||
304 | struct iwl3945_rx_packet *pkt = (void *)rxb->skb->data; | ||
305 | u16 sequence = le16_to_cpu(pkt->hdr.sequence); | ||
306 | int txq_id = SEQ_TO_QUEUE(sequence); | ||
307 | int index = SEQ_TO_INDEX(sequence); | ||
308 | struct iwl3945_tx_queue *txq = &priv->txq[txq_id]; | ||
309 | struct ieee80211_tx_status *tx_status; | ||
310 | struct iwl3945_tx_resp *tx_resp = (void *)&pkt->u.raw[0]; | ||
311 | u32 status = le32_to_cpu(tx_resp->status); | ||
312 | int rate_idx; | ||
313 | |||
314 | if ((index >= txq->q.n_bd) || (iwl3945_x2_queue_used(&txq->q, index) == 0)) { | ||
315 | IWL_ERROR("Read index for DMA queue txq_id (%d) index %d " | ||
316 | "is out of range [0-%d] %d %d\n", txq_id, | ||
317 | index, txq->q.n_bd, txq->q.write_ptr, | ||
318 | txq->q.read_ptr); | ||
319 | return; | ||
320 | } | ||
321 | |||
322 | tx_status = &(txq->txb[txq->q.read_ptr].status); | ||
323 | |||
324 | tx_status->retry_count = tx_resp->failure_frame; | ||
325 | /* tx_status->rts_retry_count = tx_resp->failure_rts; */ | ||
326 | tx_status->flags = ((status & TX_STATUS_MSK) == TX_STATUS_SUCCESS) ? | ||
327 | IEEE80211_TX_STATUS_ACK : 0; | ||
328 | |||
329 | IWL_DEBUG_TX("Tx queue %d Status %s (0x%08x) plcp rate %d retries %d\n", | ||
330 | txq_id, iwl3945_get_tx_fail_reason(status), status, | ||
331 | tx_resp->rate, tx_resp->failure_frame); | ||
332 | |||
333 | rate_idx = iwl3945_hwrate_to_plcp_idx(tx_resp->rate); | ||
334 | tx_status->control.tx_rate = &priv->ieee_rates[rate_idx]; | ||
335 | IWL_DEBUG_TX_REPLY("Tx queue reclaim %d\n", index); | ||
336 | iwl3945_tx_queue_reclaim(priv, txq_id, index); | ||
337 | |||
338 | if (iwl_check_bits(status, TX_ABORT_REQUIRED_MSK)) | ||
339 | IWL_ERROR("TODO: Implement Tx ABORT REQUIRED!!!\n"); | ||
340 | } | ||
341 | |||
342 | |||
343 | |||
230 | /***************************************************************************** | 344 | /***************************************************************************** |
231 | * | 345 | * |
232 | * Intel PRO/Wireless 3945ABG/BG Network Connection | 346 | * Intel PRO/Wireless 3945ABG/BG Network Connection |
233 | * | 347 | * |
234 | * RX handler implementations | 348 | * RX handler implementations |
235 | * | 349 | * |
236 | * Used by iwl-base.c | ||
237 | * | ||
238 | *****************************************************************************/ | 350 | *****************************************************************************/ |
239 | 351 | ||
240 | void iwl3945_hw_rx_statistics(struct iwl3945_priv *priv, struct iwl3945_rx_mem_buffer *rxb) | 352 | void iwl3945_hw_rx_statistics(struct iwl3945_priv *priv, struct iwl3945_rx_mem_buffer *rxb) |
@@ -2510,6 +2622,7 @@ unsigned int iwl3945_hw_get_beacon_cmd(struct iwl3945_priv *priv, | |||
2510 | 2622 | ||
2511 | void iwl3945_hw_rx_handler_setup(struct iwl3945_priv *priv) | 2623 | void iwl3945_hw_rx_handler_setup(struct iwl3945_priv *priv) |
2512 | { | 2624 | { |
2625 | priv->rx_handlers[REPLY_TX] = iwl3945_rx_reply_tx; | ||
2513 | priv->rx_handlers[REPLY_3945_RX] = iwl3945_rx_reply_rx; | 2626 | priv->rx_handlers[REPLY_3945_RX] = iwl3945_rx_reply_rx; |
2514 | } | 2627 | } |
2515 | 2628 | ||
diff --git a/drivers/net/wireless/iwlwifi/iwl3945-base.c b/drivers/net/wireless/iwlwifi/iwl3945-base.c index 39fd5440a248..ecf749c2dc0a 100644 --- a/drivers/net/wireless/iwlwifi/iwl3945-base.c +++ b/drivers/net/wireless/iwlwifi/iwl3945-base.c | |||
@@ -2063,34 +2063,6 @@ int iwl3945_is_network_packet(struct iwl3945_priv *priv, struct ieee80211_hdr *h | |||
2063 | return 1; | 2063 | return 1; |
2064 | } | 2064 | } |
2065 | 2065 | ||
2066 | #define TX_STATUS_ENTRY(x) case TX_STATUS_FAIL_ ## x: return #x | ||
2067 | |||
2068 | static const char *iwl3945_get_tx_fail_reason(u32 status) | ||
2069 | { | ||
2070 | switch (status & TX_STATUS_MSK) { | ||
2071 | case TX_STATUS_SUCCESS: | ||
2072 | return "SUCCESS"; | ||
2073 | TX_STATUS_ENTRY(SHORT_LIMIT); | ||
2074 | TX_STATUS_ENTRY(LONG_LIMIT); | ||
2075 | TX_STATUS_ENTRY(FIFO_UNDERRUN); | ||
2076 | TX_STATUS_ENTRY(MGMNT_ABORT); | ||
2077 | TX_STATUS_ENTRY(NEXT_FRAG); | ||
2078 | TX_STATUS_ENTRY(LIFE_EXPIRE); | ||
2079 | TX_STATUS_ENTRY(DEST_PS); | ||
2080 | TX_STATUS_ENTRY(ABORTED); | ||
2081 | TX_STATUS_ENTRY(BT_RETRY); | ||
2082 | TX_STATUS_ENTRY(STA_INVALID); | ||
2083 | TX_STATUS_ENTRY(FRAG_DROPPED); | ||
2084 | TX_STATUS_ENTRY(TID_DISABLE); | ||
2085 | TX_STATUS_ENTRY(FRAME_FLUSHED); | ||
2086 | TX_STATUS_ENTRY(INSUFFICIENT_CF_POLL); | ||
2087 | TX_STATUS_ENTRY(TX_LOCKED); | ||
2088 | TX_STATUS_ENTRY(NO_BEACON_ON_RADAR); | ||
2089 | } | ||
2090 | |||
2091 | return "UNKNOWN"; | ||
2092 | } | ||
2093 | |||
2094 | /** | 2066 | /** |
2095 | * iwl3945_scan_cancel - Cancel any currently executing HW scan | 2067 | * iwl3945_scan_cancel - Cancel any currently executing HW scan |
2096 | * | 2068 | * |
@@ -3138,125 +3110,6 @@ static int iwl3945_get_measurement(struct iwl3945_priv *priv, | |||
3138 | } | 3110 | } |
3139 | #endif | 3111 | #endif |
3140 | 3112 | ||
3141 | static void iwl3945_txstatus_to_ieee(struct iwl3945_priv *priv, | ||
3142 | struct iwl3945_tx_info *tx_sta) | ||
3143 | { | ||
3144 | |||
3145 | tx_sta->status.ack_signal = 0; | ||
3146 | tx_sta->status.excessive_retries = 0; | ||
3147 | tx_sta->status.queue_length = 0; | ||
3148 | tx_sta->status.queue_number = 0; | ||
3149 | |||
3150 | if (in_interrupt()) | ||
3151 | ieee80211_tx_status_irqsafe(priv->hw, | ||
3152 | tx_sta->skb[0], &(tx_sta->status)); | ||
3153 | else | ||
3154 | ieee80211_tx_status(priv->hw, | ||
3155 | tx_sta->skb[0], &(tx_sta->status)); | ||
3156 | |||
3157 | tx_sta->skb[0] = NULL; | ||
3158 | } | ||
3159 | |||
3160 | /** | ||
3161 | * iwl3945_tx_queue_reclaim - Reclaim Tx queue entries already Tx'd | ||
3162 | * | ||
3163 | * When FW advances 'R' index, all entries between old and new 'R' index | ||
3164 | * need to be reclaimed. As result, some free space forms. If there is | ||
3165 | * enough free space (> low mark), wake the stack that feeds us. | ||
3166 | */ | ||
3167 | static int iwl3945_tx_queue_reclaim(struct iwl3945_priv *priv, int txq_id, int index) | ||
3168 | { | ||
3169 | struct iwl3945_tx_queue *txq = &priv->txq[txq_id]; | ||
3170 | struct iwl3945_queue *q = &txq->q; | ||
3171 | int nfreed = 0; | ||
3172 | |||
3173 | if ((index >= q->n_bd) || (iwl3945_x2_queue_used(q, index) == 0)) { | ||
3174 | IWL_ERROR("Read index for DMA queue txq id (%d), index %d, " | ||
3175 | "is out of range [0-%d] %d %d.\n", txq_id, | ||
3176 | index, q->n_bd, q->write_ptr, q->read_ptr); | ||
3177 | return 0; | ||
3178 | } | ||
3179 | |||
3180 | for (index = iwl_queue_inc_wrap(index, q->n_bd); | ||
3181 | q->read_ptr != index; | ||
3182 | q->read_ptr = iwl_queue_inc_wrap(q->read_ptr, q->n_bd)) { | ||
3183 | if (txq_id != IWL_CMD_QUEUE_NUM) { | ||
3184 | iwl3945_txstatus_to_ieee(priv, | ||
3185 | &(txq->txb[txq->q.read_ptr])); | ||
3186 | iwl3945_hw_txq_free_tfd(priv, txq); | ||
3187 | } else if (nfreed > 1) { | ||
3188 | IWL_ERROR("HCMD skipped: index (%d) %d %d\n", index, | ||
3189 | q->write_ptr, q->read_ptr); | ||
3190 | queue_work(priv->workqueue, &priv->restart); | ||
3191 | } | ||
3192 | nfreed++; | ||
3193 | } | ||
3194 | |||
3195 | if (iwl3945_queue_space(q) > q->low_mark && (txq_id >= 0) && | ||
3196 | (txq_id != IWL_CMD_QUEUE_NUM) && | ||
3197 | priv->mac80211_registered) | ||
3198 | ieee80211_wake_queue(priv->hw, txq_id); | ||
3199 | |||
3200 | |||
3201 | return nfreed; | ||
3202 | } | ||
3203 | |||
3204 | static int iwl3945_is_tx_success(u32 status) | ||
3205 | { | ||
3206 | return (status & 0xFF) == 0x1; | ||
3207 | } | ||
3208 | |||
3209 | /****************************************************************************** | ||
3210 | * | ||
3211 | * Generic RX handler implementations | ||
3212 | * | ||
3213 | ******************************************************************************/ | ||
3214 | /** | ||
3215 | * iwl3945_rx_reply_tx - Handle Tx response | ||
3216 | */ | ||
3217 | static void iwl3945_rx_reply_tx(struct iwl3945_priv *priv, | ||
3218 | struct iwl3945_rx_mem_buffer *rxb) | ||
3219 | { | ||
3220 | struct iwl3945_rx_packet *pkt = (void *)rxb->skb->data; | ||
3221 | u16 sequence = le16_to_cpu(pkt->hdr.sequence); | ||
3222 | int txq_id = SEQ_TO_QUEUE(sequence); | ||
3223 | int index = SEQ_TO_INDEX(sequence); | ||
3224 | struct iwl3945_tx_queue *txq = &priv->txq[txq_id]; | ||
3225 | struct ieee80211_tx_status *tx_status; | ||
3226 | struct iwl3945_tx_resp *tx_resp = (void *)&pkt->u.raw[0]; | ||
3227 | u32 status = le32_to_cpu(tx_resp->status); | ||
3228 | |||
3229 | if ((index >= txq->q.n_bd) || (iwl3945_x2_queue_used(&txq->q, index) == 0)) { | ||
3230 | IWL_ERROR("Read index for DMA queue txq_id (%d) index %d " | ||
3231 | "is out of range [0-%d] %d %d\n", txq_id, | ||
3232 | index, txq->q.n_bd, txq->q.write_ptr, | ||
3233 | txq->q.read_ptr); | ||
3234 | return; | ||
3235 | } | ||
3236 | |||
3237 | tx_status = &(txq->txb[txq->q.read_ptr].status); | ||
3238 | |||
3239 | tx_status->retry_count = tx_resp->failure_frame; | ||
3240 | tx_status->queue_number = status; | ||
3241 | tx_status->queue_length = tx_resp->bt_kill_count; | ||
3242 | tx_status->queue_length |= tx_resp->failure_rts; | ||
3243 | |||
3244 | tx_status->flags = | ||
3245 | iwl3945_is_tx_success(status) ? IEEE80211_TX_STATUS_ACK : 0; | ||
3246 | |||
3247 | IWL_DEBUG_TX("Tx queue %d Status %s (0x%08x) plcp rate %d retries %d\n", | ||
3248 | txq_id, iwl3945_get_tx_fail_reason(status), status, | ||
3249 | tx_resp->rate, tx_resp->failure_frame); | ||
3250 | |||
3251 | IWL_DEBUG_TX_REPLY("Tx queue reclaim %d\n", index); | ||
3252 | if (index != -1) | ||
3253 | iwl3945_tx_queue_reclaim(priv, txq_id, index); | ||
3254 | |||
3255 | if (iwl_check_bits(status, TX_ABORT_REQUIRED_MSK)) | ||
3256 | IWL_ERROR("TODO: Implement Tx ABORT REQUIRED!!!\n"); | ||
3257 | } | ||
3258 | |||
3259 | |||
3260 | static void iwl3945_rx_reply_alive(struct iwl3945_priv *priv, | 3113 | static void iwl3945_rx_reply_alive(struct iwl3945_priv *priv, |
3261 | struct iwl3945_rx_mem_buffer *rxb) | 3114 | struct iwl3945_rx_mem_buffer *rxb) |
3262 | { | 3115 | { |
@@ -3603,13 +3456,44 @@ static void iwl3945_setup_rx_handlers(struct iwl3945_priv *priv) | |||
3603 | priv->rx_handlers[SCAN_COMPLETE_NOTIFICATION] = | 3456 | priv->rx_handlers[SCAN_COMPLETE_NOTIFICATION] = |
3604 | iwl3945_rx_scan_complete_notif; | 3457 | iwl3945_rx_scan_complete_notif; |
3605 | priv->rx_handlers[CARD_STATE_NOTIFICATION] = iwl3945_rx_card_state_notif; | 3458 | priv->rx_handlers[CARD_STATE_NOTIFICATION] = iwl3945_rx_card_state_notif; |
3606 | priv->rx_handlers[REPLY_TX] = iwl3945_rx_reply_tx; | ||
3607 | 3459 | ||
3608 | /* Set up hardware specific Rx handlers */ | 3460 | /* Set up hardware specific Rx handlers */ |
3609 | iwl3945_hw_rx_handler_setup(priv); | 3461 | iwl3945_hw_rx_handler_setup(priv); |
3610 | } | 3462 | } |
3611 | 3463 | ||
3612 | /** | 3464 | /** |
3465 | * iwl3945_cmd_queue_reclaim - Reclaim CMD queue entries | ||
3466 | * When FW advances 'R' index, all entries between old and new 'R' index | ||
3467 | * need to be reclaimed. | ||
3468 | */ | ||
3469 | static void iwl3945_cmd_queue_reclaim(struct iwl3945_priv *priv, | ||
3470 | int txq_id, int index) | ||
3471 | { | ||
3472 | struct iwl3945_tx_queue *txq = &priv->txq[txq_id]; | ||
3473 | struct iwl3945_queue *q = &txq->q; | ||
3474 | int nfreed = 0; | ||
3475 | |||
3476 | if ((index >= q->n_bd) || (iwl3945_x2_queue_used(q, index) == 0)) { | ||
3477 | IWL_ERROR("Read index for DMA queue txq id (%d), index %d, " | ||
3478 | "is out of range [0-%d] %d %d.\n", txq_id, | ||
3479 | index, q->n_bd, q->write_ptr, q->read_ptr); | ||
3480 | return; | ||
3481 | } | ||
3482 | |||
3483 | for (index = iwl_queue_inc_wrap(index, q->n_bd); q->read_ptr != index; | ||
3484 | q->read_ptr = iwl_queue_inc_wrap(q->read_ptr, q->n_bd)) { | ||
3485 | if (nfreed > 1) { | ||
3486 | IWL_ERROR("HCMD skipped: index (%d) %d %d\n", index, | ||
3487 | q->write_ptr, q->read_ptr); | ||
3488 | queue_work(priv->workqueue, &priv->restart); | ||
3489 | break; | ||
3490 | } | ||
3491 | nfreed++; | ||
3492 | } | ||
3493 | } | ||
3494 | |||
3495 | |||
3496 | /** | ||
3613 | * iwl3945_tx_cmd_complete - Pull unused buffers off the queue and reclaim them | 3497 | * iwl3945_tx_cmd_complete - Pull unused buffers off the queue and reclaim them |
3614 | * @rxb: Rx buffer to reclaim | 3498 | * @rxb: Rx buffer to reclaim |
3615 | * | 3499 | * |
@@ -3628,12 +3512,6 @@ static void iwl3945_tx_cmd_complete(struct iwl3945_priv *priv, | |||
3628 | int cmd_index; | 3512 | int cmd_index; |
3629 | struct iwl3945_cmd *cmd; | 3513 | struct iwl3945_cmd *cmd; |
3630 | 3514 | ||
3631 | /* If a Tx command is being handled and it isn't in the actual | ||
3632 | * command queue then there a command routing bug has been introduced | ||
3633 | * in the queue management code. */ | ||
3634 | if (txq_id != IWL_CMD_QUEUE_NUM) | ||
3635 | IWL_ERROR("Error wrong command queue %d command id 0x%X\n", | ||
3636 | txq_id, pkt->hdr.cmd); | ||
3637 | BUG_ON(txq_id != IWL_CMD_QUEUE_NUM); | 3515 | BUG_ON(txq_id != IWL_CMD_QUEUE_NUM); |
3638 | 3516 | ||
3639 | cmd_index = get_cmd_index(&priv->txq[IWL_CMD_QUEUE_NUM].q, index, huge); | 3517 | cmd_index = get_cmd_index(&priv->txq[IWL_CMD_QUEUE_NUM].q, index, huge); |
@@ -3647,7 +3525,7 @@ static void iwl3945_tx_cmd_complete(struct iwl3945_priv *priv, | |||
3647 | !cmd->meta.u.callback(priv, cmd, rxb->skb)) | 3525 | !cmd->meta.u.callback(priv, cmd, rxb->skb)) |
3648 | rxb->skb = NULL; | 3526 | rxb->skb = NULL; |
3649 | 3527 | ||
3650 | iwl3945_tx_queue_reclaim(priv, txq_id, index); | 3528 | iwl3945_cmd_queue_reclaim(priv, txq_id, index); |
3651 | 3529 | ||
3652 | if (!(cmd->meta.flags & CMD_ASYNC)) { | 3530 | if (!(cmd->meta.flags & CMD_ASYNC)) { |
3653 | clear_bit(STATUS_HCMD_ACTIVE, &priv->status); | 3531 | clear_bit(STATUS_HCMD_ACTIVE, &priv->status); |