mirror of https://gitee.com/openkylin/linux.git
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:
parent
4479a899f7
commit
d56daea43c
|
@ -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
|
||||
* 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,
|
||||
struct iwl_rx_mem_buffer
|
||||
*out[RX_CLAIM_REQ_ALLOC])
|
||||
static void iwl_pcie_rx_allocator_get(struct iwl_trans *trans,
|
||||
struct iwl_rxq *rxq)
|
||||
{
|
||||
struct iwl_trans_pcie *trans_pcie = IWL_TRANS_GET_PCIE_TRANS(trans);
|
||||
struct iwl_rb_allocator *rba = &trans_pcie->rba;
|
||||
int i;
|
||||
|
||||
lockdep_assert_held(&rxq->lock);
|
||||
|
||||
/*
|
||||
* 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
|
||||
* 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
|
||||
* req_ready > 0, i.e. - there are ready requests and the function
|
||||
* hands one request to the caller.
|
||||
*/
|
||||
if (atomic_dec_if_positive(&rba->req_ready) < 0)
|
||||
return -ENOMEM;
|
||||
return;
|
||||
|
||||
spin_lock(&rba->lock);
|
||||
for (i = 0; i < RX_CLAIM_REQ_ALLOC; i++) {
|
||||
/* Get next free Rx buffer, remove it from free list */
|
||||
out[i] = list_first_entry(&rba->rbd_allocated,
|
||||
struct iwl_rx_mem_buffer *rxb =
|
||||
list_first_entry(&rba->rbd_allocated,
|
||||
struct iwl_rx_mem_buffer, list);
|
||||
list_del(&out[i]->list);
|
||||
|
||||
list_move(&rxb->list, &rxq->rx_free);
|
||||
}
|
||||
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)
|
||||
|
@ -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_rxq *rxq = &trans_pcie->rxq[queue];
|
||||
u32 r, i, j, count = 0;
|
||||
u32 r, i, count = 0;
|
||||
bool emergency = false;
|
||||
|
||||
restart:
|
||||
|
@ -1193,39 +1199,24 @@ static void iwl_pcie_rx_handle(struct iwl_trans *trans, int queue)
|
|||
|
||||
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 (rxq->used_count >= RX_CLAIM_REQ_ALLOC) {
|
||||
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 &&
|
||||
!emergency) {
|
||||
/* Add the remaining 6 empty RBDs
|
||||
* for allocator use
|
||||
/*
|
||||
* If we have RX_CLAIM_REQ_ALLOC released rx buffers -
|
||||
* 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
|
||||
*/
|
||||
spin_lock(&rba->lock);
|
||||
list_splice_tail_init(&rxq->rx_used,
|
||||
&rba->rbd_empty);
|
||||
spin_unlock(&rba->lock);
|
||||
}
|
||||
if (rxq->used_count >= RX_CLAIM_REQ_ALLOC)
|
||||
iwl_pcie_rx_allocator_get(trans, rxq);
|
||||
|
||||
/* 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) {
|
||||
if (rxq->used_count % RX_CLAIM_REQ_ALLOC == 0 && !emergency) {
|
||||
struct iwl_rb_allocator *rba = &trans_pcie->rba;
|
||||
|
||||
/* Add the remaining empty RBDs for allocator use */
|
||||
spin_lock(&rba->lock);
|
||||
list_splice_tail_init(&rxq->rx_used, &rba->rbd_empty);
|
||||
spin_unlock(&rba->lock);
|
||||
} else if (emergency) {
|
||||
count++;
|
||||
if (count == 8) {
|
||||
count = 0;
|
||||
|
|
Loading…
Reference in New Issue