iwlwifi: pcie: refactor RXBs reclaiming code

Change the code to move rxbs directly from the allocator's
list to the queue's free list. This makes the code more
readable, saves the interim array and the double loop over
the free RBs.

Signed-off-by: Sara Sharon <sara.sharon@intel.com>
Signed-off-by: Emmanuel Grumbach <emmanuel.grumbach@intel.com>
This commit is contained in:
Sara Sharon 2016-02-15 19:30:49 +02:00 committed by Emmanuel Grumbach
parent 4479a899f7
commit d56daea43c
1 changed files with 33 additions and 42 deletions

View File

@ -539,40 +539,46 @@ static void iwl_pcie_rx_allocator(struct iwl_trans *trans)
} }
/* /*
* iwl_pcie_rx_allocator_get - Returns the pre-allocated pages * iwl_pcie_rx_allocator_get - returns the pre-allocated pages
.* .*
.* Called by queue when the queue posted allocation request and .* Called by queue when the queue posted allocation request and
* has freed 8 RBDs in order to restock itself. * has freed 8 RBDs in order to restock itself.
* This function directly moves the allocated RBs to the queue's ownership
* and updates the relevant counters.
*/ */
static int iwl_pcie_rx_allocator_get(struct iwl_trans *trans, static void iwl_pcie_rx_allocator_get(struct iwl_trans *trans,
struct iwl_rx_mem_buffer struct iwl_rxq *rxq)
*out[RX_CLAIM_REQ_ALLOC])
{ {
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans); struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
struct iwl_rb_allocator *rba = &trans_pcie->rba; struct iwl_rb_allocator *rba = &trans_pcie->rba;
int i; int i;
lockdep_assert_held(&rxq->lock);
/* /*
* atomic_dec_if_positive returns req_ready - 1 for any scenario. * atomic_dec_if_positive returns req_ready - 1 for any scenario.
* If req_ready is 0 atomic_dec_if_positive will return -1 and this * If req_ready is 0 atomic_dec_if_positive will return -1 and this
* function will return -ENOMEM, as there are no ready requests. * function will return early, as there are no ready requests.
* atomic_dec_if_positive will perofrm the *actual* decrement only if * atomic_dec_if_positive will perofrm the *actual* decrement only if
* req_ready > 0, i.e. - there are ready requests and the function * req_ready > 0, i.e. - there are ready requests and the function
* hands one request to the caller. * hands one request to the caller.
*/ */
if (atomic_dec_if_positive(&rba->req_ready) < 0) if (atomic_dec_if_positive(&rba->req_ready) < 0)
return -ENOMEM; return;
spin_lock(&rba->lock); spin_lock(&rba->lock);
for (i = 0; i < RX_CLAIM_REQ_ALLOC; i++) { for (i = 0; i < RX_CLAIM_REQ_ALLOC; i++) {
/* Get next free Rx buffer, remove it from free list */ /* Get next free Rx buffer, remove it from free list */
out[i] = list_first_entry(&rba->rbd_allocated, struct iwl_rx_mem_buffer *rxb =
struct iwl_rx_mem_buffer, list); list_first_entry(&rba->rbd_allocated,
list_del(&out[i]->list); struct iwl_rx_mem_buffer, list);
list_move(&rxb->list, &rxq->rx_free);
} }
spin_unlock(&rba->lock); spin_unlock(&rba->lock);
return 0; rxq->used_count -= RX_CLAIM_REQ_ALLOC;
rxq->free_count += RX_CLAIM_REQ_ALLOC;
} }
static void iwl_pcie_rx_allocator_work(struct work_struct *data) static void iwl_pcie_rx_allocator_work(struct work_struct *data)
@ -1149,7 +1155,7 @@ static void iwl_pcie_rx_handle(struct iwl_trans *trans, int queue)
{ {
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans); struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
struct iwl_rxq *rxq = &trans_pcie->rxq[queue]; struct iwl_rxq *rxq = &trans_pcie->rxq[queue];
u32 r, i, j, count = 0; u32 r, i, count = 0;
bool emergency = false; bool emergency = false;
restart: restart:
@ -1193,39 +1199,24 @@ restart:
i = (i + 1) & (rxq->queue_size - 1); i = (i + 1) & (rxq->queue_size - 1);
/* If we have RX_CLAIM_REQ_ALLOC released rx buffers - /*
* try to claim the pre-allocated buffers from the allocator */ * If we have RX_CLAIM_REQ_ALLOC released rx buffers -
if (rxq->used_count >= RX_CLAIM_REQ_ALLOC) { * try to claim the pre-allocated buffers from the allocator.
* If not ready - will try to reclaim next time.
* There is no need to reschedule work - allocator exits only
* on success
*/
if (rxq->used_count >= RX_CLAIM_REQ_ALLOC)
iwl_pcie_rx_allocator_get(trans, rxq);
if (rxq->used_count % RX_CLAIM_REQ_ALLOC == 0 && !emergency) {
struct iwl_rb_allocator *rba = &trans_pcie->rba; struct iwl_rb_allocator *rba = &trans_pcie->rba;
struct iwl_rx_mem_buffer *out[RX_CLAIM_REQ_ALLOC];
if (rxq->used_count % RX_CLAIM_REQ_ALLOC == 0 && /* Add the remaining empty RBDs for allocator use */
!emergency) { spin_lock(&rba->lock);
/* Add the remaining 6 empty RBDs list_splice_tail_init(&rxq->rx_used, &rba->rbd_empty);
* for allocator use spin_unlock(&rba->lock);
*/ } else if (emergency) {
spin_lock(&rba->lock);
list_splice_tail_init(&rxq->rx_used,
&rba->rbd_empty);
spin_unlock(&rba->lock);
}
/* If not ready - continue, will try to reclaim later.
* No need to reschedule work - allocator exits only on
* success */
if (!iwl_pcie_rx_allocator_get(trans, out)) {
/* If success - then RX_CLAIM_REQ_ALLOC
* buffers were retrieved and should be added
* to free list */
rxq->used_count -= RX_CLAIM_REQ_ALLOC;
for (j = 0; j < RX_CLAIM_REQ_ALLOC; j++) {
list_add_tail(&out[j]->list,
&rxq->rx_free);
rxq->free_count++;
}
}
}
if (emergency) {
count++; count++;
if (count == 8) { if (count == 8) {
count = 0; count = 0;