Commit d1eec614 authored by Michael Chan's avatar Michael Chan Committed by David S. Miller

bnxt_en: Restructure cp_ring_arr in struct bnxt_cp_ring_info

The cp_ring_arr is currently a fixed array of 2 pointers for the
TX and RX completion rings.  These pointers are allocated during
ring initialization.  Currntly, we support up to 2 completion rings
for each MSIX.  In order to support more completion rings, we change
this fixed array to a pointer and allocate the required entries
during ring initialization.  This patch keeps the current scheme of
allocating only 2 entries when needed.  Later patches will expand
and allocate more entries when required.
Reviewed-by: default avatarAndy Gospodarek <andrew.gospodarek@broadcom.com>
Signed-off-by: default avatarMichael Chan <michael.chan@broadcom.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 7f0a168b
...@@ -2834,14 +2834,11 @@ static int __bnxt_poll_cqs(struct bnxt *bp, struct bnxt_napi *bnapi, int budget) ...@@ -2834,14 +2834,11 @@ static int __bnxt_poll_cqs(struct bnxt *bp, struct bnxt_napi *bnapi, int budget)
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring; struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
int i, work_done = 0; int i, work_done = 0;
for (i = 0; i < 2; i++) { for (i = 0; i < cpr->cp_ring_count; i++) {
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[i]; struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[i];
if (cpr2) { work_done += __bnxt_poll_work(bp, cpr2, budget - work_done);
work_done += __bnxt_poll_work(bp, cpr2, cpr->has_more_work |= cpr2->has_more_work;
budget - work_done);
cpr->has_more_work |= cpr2->has_more_work;
}
} }
return work_done; return work_done;
} }
...@@ -2852,11 +2849,11 @@ static void __bnxt_poll_cqs_done(struct bnxt *bp, struct bnxt_napi *bnapi, ...@@ -2852,11 +2849,11 @@ static void __bnxt_poll_cqs_done(struct bnxt *bp, struct bnxt_napi *bnapi,
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring; struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
int i; int i;
for (i = 0; i < 2; i++) { for (i = 0; i < cpr->cp_ring_count; i++) {
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[i]; struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[i];
struct bnxt_db_info *db; struct bnxt_db_info *db;
if (cpr2 && cpr2->had_work_done) { if (cpr2->had_work_done) {
db = &cpr2->cp_db; db = &cpr2->cp_db;
bnxt_writeq(bp, db->db_key64 | dbr_type | bnxt_writeq(bp, db->db_key64 | dbr_type |
RING_CMP(cpr2->cp_raw_cons), db->doorbell); RING_CMP(cpr2->cp_raw_cons), db->doorbell);
...@@ -2915,7 +2912,7 @@ static int bnxt_poll_p5(struct napi_struct *napi, int budget) ...@@ -2915,7 +2912,7 @@ static int bnxt_poll_p5(struct napi_struct *napi, int budget)
if (budget && work_done >= budget && idx == BNXT_RX_HDL) if (budget && work_done >= budget && idx == BNXT_RX_HDL)
break; break;
cpr2 = cpr->cp_ring_arr[idx]; cpr2 = &cpr->cp_ring_arr[idx];
work_done += __bnxt_poll_work(bp, cpr2, work_done += __bnxt_poll_work(bp, cpr2,
budget - work_done); budget - work_done);
cpr->has_more_work |= cpr2->has_more_work; cpr->has_more_work |= cpr2->has_more_work;
...@@ -2930,8 +2927,8 @@ static int bnxt_poll_p5(struct napi_struct *napi, int budget) ...@@ -2930,8 +2927,8 @@ static int bnxt_poll_p5(struct napi_struct *napi, int budget)
BNXT_DB_NQ_P5(&cpr->cp_db, raw_cons); BNXT_DB_NQ_P5(&cpr->cp_db, raw_cons);
} }
poll_done: poll_done:
cpr_rx = cpr->cp_ring_arr[BNXT_RX_HDL]; cpr_rx = &cpr->cp_ring_arr[BNXT_RX_HDL];
if (cpr_rx && (bp->flags & BNXT_FLAG_DIM)) { if (cpr_rx->bnapi && (bp->flags & BNXT_FLAG_DIM)) {
struct dim_sample dim_sample = {}; struct dim_sample dim_sample = {};
dim_update_sample(cpr->event_ctr, dim_update_sample(cpr->event_ctr,
...@@ -3541,36 +3538,33 @@ static void bnxt_free_cp_rings(struct bnxt *bp) ...@@ -3541,36 +3538,33 @@ static void bnxt_free_cp_rings(struct bnxt *bp)
bnxt_free_ring(bp, &ring->ring_mem); bnxt_free_ring(bp, &ring->ring_mem);
for (j = 0; j < 2; j++) { if (!cpr->cp_ring_arr)
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[j]; continue;
if (cpr2) { for (j = 0; j < cpr->cp_ring_count; j++) {
ring = &cpr2->cp_ring_struct; struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[j];
bnxt_free_ring(bp, &ring->ring_mem);
bnxt_free_cp_arrays(cpr2); ring = &cpr2->cp_ring_struct;
kfree(cpr2); bnxt_free_ring(bp, &ring->ring_mem);
cpr->cp_ring_arr[j] = NULL; bnxt_free_cp_arrays(cpr2);
}
} }
kfree(cpr->cp_ring_arr);
cpr->cp_ring_arr = NULL;
cpr->cp_ring_count = 0;
} }
} }
static struct bnxt_cp_ring_info *bnxt_alloc_cp_sub_ring(struct bnxt *bp) static int bnxt_alloc_cp_sub_ring(struct bnxt *bp,
struct bnxt_cp_ring_info *cpr)
{ {
struct bnxt_ring_mem_info *rmem; struct bnxt_ring_mem_info *rmem;
struct bnxt_ring_struct *ring; struct bnxt_ring_struct *ring;
struct bnxt_cp_ring_info *cpr;
int rc; int rc;
cpr = kzalloc(sizeof(*cpr), GFP_KERNEL);
if (!cpr)
return NULL;
rc = bnxt_alloc_cp_arrays(cpr, bp->cp_nr_pages); rc = bnxt_alloc_cp_arrays(cpr, bp->cp_nr_pages);
if (rc) { if (rc) {
bnxt_free_cp_arrays(cpr); bnxt_free_cp_arrays(cpr);
kfree(cpr); return -ENOMEM;
return NULL;
} }
ring = &cpr->cp_ring_struct; ring = &cpr->cp_ring_struct;
rmem = &ring->ring_mem; rmem = &ring->ring_mem;
...@@ -3583,10 +3577,8 @@ static struct bnxt_cp_ring_info *bnxt_alloc_cp_sub_ring(struct bnxt *bp) ...@@ -3583,10 +3577,8 @@ static struct bnxt_cp_ring_info *bnxt_alloc_cp_sub_ring(struct bnxt *bp)
if (rc) { if (rc) {
bnxt_free_ring(bp, rmem); bnxt_free_ring(bp, rmem);
bnxt_free_cp_arrays(cpr); bnxt_free_cp_arrays(cpr);
kfree(cpr);
cpr = NULL;
} }
return cpr; return rc;
} }
static int bnxt_alloc_cp_rings(struct bnxt *bp) static int bnxt_alloc_cp_rings(struct bnxt *bp)
...@@ -3598,7 +3590,7 @@ static int bnxt_alloc_cp_rings(struct bnxt *bp) ...@@ -3598,7 +3590,7 @@ static int bnxt_alloc_cp_rings(struct bnxt *bp)
ulp_base_vec = bnxt_get_ulp_msix_base(bp); ulp_base_vec = bnxt_get_ulp_msix_base(bp);
for (i = 0; i < bp->cp_nr_rings; i++) { for (i = 0; i < bp->cp_nr_rings; i++) {
struct bnxt_napi *bnapi = bp->bnapi[i]; struct bnxt_napi *bnapi = bp->bnapi[i];
struct bnxt_cp_ring_info *cpr; struct bnxt_cp_ring_info *cpr, *cpr2;
struct bnxt_ring_struct *ring; struct bnxt_ring_struct *ring;
if (!bnapi) if (!bnapi)
...@@ -3620,23 +3612,27 @@ static int bnxt_alloc_cp_rings(struct bnxt *bp) ...@@ -3620,23 +3612,27 @@ static int bnxt_alloc_cp_rings(struct bnxt *bp)
if (!(bp->flags & BNXT_FLAG_CHIP_P5)) if (!(bp->flags & BNXT_FLAG_CHIP_P5))
continue; continue;
if (i < bp->rx_nr_rings) { cpr->cp_ring_count = 2;
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr = kcalloc(cpr->cp_ring_count, sizeof(*cpr),
bnxt_alloc_cp_sub_ring(bp); GFP_KERNEL);
if (!cpr->cp_ring_arr) {
cpr->cp_ring_count = 0;
return -ENOMEM;
}
cpr->cp_ring_arr[BNXT_RX_HDL] = cpr2; if (i < bp->rx_nr_rings) {
if (!cpr2) cpr2 = &cpr->cp_ring_arr[BNXT_RX_HDL];
return -ENOMEM; rc = bnxt_alloc_cp_sub_ring(bp, cpr2);
if (rc)
return rc;
cpr2->bnapi = bnapi; cpr2->bnapi = bnapi;
} }
if ((sh && i < bp->tx_nr_rings) || if ((sh && i < bp->tx_nr_rings) ||
(!sh && i >= bp->rx_nr_rings)) { (!sh && i >= bp->rx_nr_rings)) {
struct bnxt_cp_ring_info *cpr2 = cpr2 = &cpr->cp_ring_arr[BNXT_TX_HDL];
bnxt_alloc_cp_sub_ring(bp); rc = bnxt_alloc_cp_sub_ring(bp, cpr2);
if (rc)
cpr->cp_ring_arr[BNXT_TX_HDL] = cpr2; return rc;
if (!cpr2)
return -ENOMEM;
cpr2->bnapi = bnapi; cpr2->bnapi = bnapi;
} }
} }
...@@ -3822,11 +3818,10 @@ static void bnxt_init_cp_rings(struct bnxt *bp) ...@@ -3822,11 +3818,10 @@ static void bnxt_init_cp_rings(struct bnxt *bp)
ring->fw_ring_id = INVALID_HW_RING_ID; ring->fw_ring_id = INVALID_HW_RING_ID;
cpr->rx_ring_coal.coal_ticks = bp->rx_coal.coal_ticks; cpr->rx_ring_coal.coal_ticks = bp->rx_coal.coal_ticks;
cpr->rx_ring_coal.coal_bufs = bp->rx_coal.coal_bufs; cpr->rx_ring_coal.coal_bufs = bp->rx_coal.coal_bufs;
for (j = 0; j < 2; j++) { if (!cpr->cp_ring_arr)
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[j]; continue;
for (j = 0; j < cpr->cp_ring_count; j++) {
if (!cpr2) struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[j];
continue;
ring = &cpr2->cp_ring_struct; ring = &cpr2->cp_ring_struct;
ring->fw_ring_id = INVALID_HW_RING_ID; ring->fw_ring_id = INVALID_HW_RING_ID;
...@@ -5251,7 +5246,7 @@ static u16 bnxt_cp_ring_for_rx(struct bnxt *bp, struct bnxt_rx_ring_info *rxr) ...@@ -5251,7 +5246,7 @@ static u16 bnxt_cp_ring_for_rx(struct bnxt *bp, struct bnxt_rx_ring_info *rxr)
struct bnxt_napi *bnapi = rxr->bnapi; struct bnxt_napi *bnapi = rxr->bnapi;
struct bnxt_cp_ring_info *cpr; struct bnxt_cp_ring_info *cpr;
cpr = bnapi->cp_ring.cp_ring_arr[BNXT_RX_HDL]; cpr = &bnapi->cp_ring.cp_ring_arr[BNXT_RX_HDL];
return cpr->cp_ring_struct.fw_ring_id; return cpr->cp_ring_struct.fw_ring_id;
} else { } else {
return bnxt_cp_ring_from_grp(bp, &rxr->rx_ring_struct); return bnxt_cp_ring_from_grp(bp, &rxr->rx_ring_struct);
...@@ -5264,7 +5259,7 @@ static u16 bnxt_cp_ring_for_tx(struct bnxt *bp, struct bnxt_tx_ring_info *txr) ...@@ -5264,7 +5259,7 @@ static u16 bnxt_cp_ring_for_tx(struct bnxt *bp, struct bnxt_tx_ring_info *txr)
struct bnxt_napi *bnapi = txr->bnapi; struct bnxt_napi *bnapi = txr->bnapi;
struct bnxt_cp_ring_info *cpr; struct bnxt_cp_ring_info *cpr;
cpr = bnapi->cp_ring.cp_ring_arr[BNXT_TX_HDL]; cpr = &bnapi->cp_ring.cp_ring_arr[BNXT_TX_HDL];
return cpr->cp_ring_struct.fw_ring_id; return cpr->cp_ring_struct.fw_ring_id;
} else { } else {
return bnxt_cp_ring_from_grp(bp, &txr->tx_ring_struct); return bnxt_cp_ring_from_grp(bp, &txr->tx_ring_struct);
...@@ -6032,7 +6027,7 @@ static int bnxt_hwrm_ring_alloc(struct bnxt *bp) ...@@ -6032,7 +6027,7 @@ static int bnxt_hwrm_ring_alloc(struct bnxt *bp)
u32 type2 = HWRM_RING_ALLOC_CMPL; u32 type2 = HWRM_RING_ALLOC_CMPL;
cpr = &bnapi->cp_ring; cpr = &bnapi->cp_ring;
cpr2 = cpr->cp_ring_arr[BNXT_TX_HDL]; cpr2 = &cpr->cp_ring_arr[BNXT_TX_HDL];
ring = &cpr2->cp_ring_struct; ring = &cpr2->cp_ring_struct;
ring->handle = BNXT_TX_HDL; ring->handle = BNXT_TX_HDL;
map_idx = bnapi->index; map_idx = bnapi->index;
...@@ -6071,7 +6066,7 @@ static int bnxt_hwrm_ring_alloc(struct bnxt *bp) ...@@ -6071,7 +6066,7 @@ static int bnxt_hwrm_ring_alloc(struct bnxt *bp)
u32 type2 = HWRM_RING_ALLOC_CMPL; u32 type2 = HWRM_RING_ALLOC_CMPL;
struct bnxt_cp_ring_info *cpr2; struct bnxt_cp_ring_info *cpr2;
cpr2 = cpr->cp_ring_arr[BNXT_RX_HDL]; cpr2 = &cpr->cp_ring_arr[BNXT_RX_HDL];
ring = &cpr2->cp_ring_struct; ring = &cpr2->cp_ring_struct;
ring->handle = BNXT_RX_HDL; ring->handle = BNXT_RX_HDL;
rc = hwrm_ring_alloc_send_msg(bp, ring, type2, map_idx); rc = hwrm_ring_alloc_send_msg(bp, ring, type2, map_idx);
...@@ -6218,18 +6213,16 @@ static void bnxt_hwrm_ring_free(struct bnxt *bp, bool close_path) ...@@ -6218,18 +6213,16 @@ static void bnxt_hwrm_ring_free(struct bnxt *bp, bool close_path)
struct bnxt_ring_struct *ring; struct bnxt_ring_struct *ring;
int j; int j;
for (j = 0; j < 2; j++) { for (j = 0; j < cpr->cp_ring_count && cpr->cp_ring_arr; j++) {
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[j]; struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[j];
if (cpr2) { ring = &cpr2->cp_ring_struct;
ring = &cpr2->cp_ring_struct; if (ring->fw_ring_id == INVALID_HW_RING_ID)
if (ring->fw_ring_id == INVALID_HW_RING_ID) continue;
continue; hwrm_ring_free_send_msg(bp, ring,
hwrm_ring_free_send_msg(bp, ring, RING_FREE_REQ_RING_TYPE_L2_CMPL,
RING_FREE_REQ_RING_TYPE_L2_CMPL, INVALID_HW_RING_ID);
INVALID_HW_RING_ID); ring->fw_ring_id = INVALID_HW_RING_ID;
ring->fw_ring_id = INVALID_HW_RING_ID;
}
} }
ring = &cpr->cp_ring_struct; ring = &cpr->cp_ring_struct;
if (ring->fw_ring_id != INVALID_HW_RING_ID) { if (ring->fw_ring_id != INVALID_HW_RING_ID) {
...@@ -12005,12 +11998,11 @@ static void bnxt_chk_missed_irq(struct bnxt *bp) ...@@ -12005,12 +11998,11 @@ static void bnxt_chk_missed_irq(struct bnxt *bp)
continue; continue;
cpr = &bnapi->cp_ring; cpr = &bnapi->cp_ring;
for (j = 0; j < 2; j++) { for (j = 0; j < cpr->cp_ring_count; j++) {
struct bnxt_cp_ring_info *cpr2 = cpr->cp_ring_arr[j]; struct bnxt_cp_ring_info *cpr2 = &cpr->cp_ring_arr[j];
u32 val[2]; u32 val[2];
if (!cpr2 || cpr2->has_more_work || if (cpr2->has_more_work || !bnxt_has_work(bp, cpr2))
!bnxt_has_work(bp, cpr2))
continue; continue;
if (cpr2->cp_raw_cons != cpr2->last_cp_raw_cons) { if (cpr2->cp_raw_cons != cpr2->last_cp_raw_cons) {
......
...@@ -1019,7 +1019,8 @@ struct bnxt_cp_ring_info { ...@@ -1019,7 +1019,8 @@ struct bnxt_cp_ring_info {
struct bnxt_ring_struct cp_ring_struct; struct bnxt_ring_struct cp_ring_struct;
struct bnxt_cp_ring_info *cp_ring_arr[2]; int cp_ring_count;
struct bnxt_cp_ring_info *cp_ring_arr;
#define BNXT_RX_HDL 0 #define BNXT_RX_HDL 0
#define BNXT_TX_HDL 1 #define BNXT_TX_HDL 1
}; };
......
...@@ -3941,7 +3941,7 @@ static int bnxt_run_loopback(struct bnxt *bp) ...@@ -3941,7 +3941,7 @@ static int bnxt_run_loopback(struct bnxt *bp)
cpr = &rxr->bnapi->cp_ring; cpr = &rxr->bnapi->cp_ring;
if (bp->flags & BNXT_FLAG_CHIP_P5) if (bp->flags & BNXT_FLAG_CHIP_P5)
cpr = cpr->cp_ring_arr[BNXT_RX_HDL]; cpr = &cpr->cp_ring_arr[BNXT_RX_HDL];
pkt_size = min(bp->dev->mtu + ETH_HLEN, bp->rx_copy_thresh); pkt_size = min(bp->dev->mtu + ETH_HLEN, bp->rx_copy_thresh);
skb = netdev_alloc_skb(bp->dev, pkt_size); skb = netdev_alloc_skb(bp->dev, pkt_size);
if (!skb) if (!skb)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment