bnxt_en: Improve the IRQ disable sequence during shutdown.
The IRQ is disabled by writing to the completion ring doorbell. This should be done before the hardware completion ring is freed for correctness. The current code disables IRQs after all the completion rings are freed. Fix it by calling bnxt_disable_int_sync() before freeing the completion rings. Rearrange the code to avoid forward declaration. Signed-off-by: Michael Chan <michael.chan@broadocm.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
e7b9569102
commit
9d8bc09766
@ -2953,6 +2953,45 @@ alloc_mem_err:
|
|||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void bnxt_disable_int(struct bnxt *bp)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!bp->bnapi)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; i < bp->cp_nr_rings; i++) {
|
||||||
|
struct bnxt_napi *bnapi = bp->bnapi[i];
|
||||||
|
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
|
||||||
|
|
||||||
|
BNXT_CP_DB(cpr->cp_doorbell, cpr->cp_raw_cons);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bnxt_disable_int_sync(struct bnxt *bp)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
atomic_inc(&bp->intr_sem);
|
||||||
|
|
||||||
|
bnxt_disable_int(bp);
|
||||||
|
for (i = 0; i < bp->cp_nr_rings; i++)
|
||||||
|
synchronize_irq(bp->irq_tbl[i].vector);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bnxt_enable_int(struct bnxt *bp)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
atomic_set(&bp->intr_sem, 0);
|
||||||
|
for (i = 0; i < bp->cp_nr_rings; i++) {
|
||||||
|
struct bnxt_napi *bnapi = bp->bnapi[i];
|
||||||
|
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
|
||||||
|
|
||||||
|
BNXT_CP_DB_REARM(cpr->cp_doorbell, cpr->cp_raw_cons);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void bnxt_hwrm_cmd_hdr_init(struct bnxt *bp, void *request, u16 req_type,
|
void bnxt_hwrm_cmd_hdr_init(struct bnxt *bp, void *request, u16 req_type,
|
||||||
u16 cmpl_ring, u16 target_id)
|
u16 cmpl_ring, u16 target_id)
|
||||||
{
|
{
|
||||||
@ -3937,6 +3976,12 @@ static void bnxt_hwrm_ring_free(struct bnxt *bp, bool close_path)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* The completion rings are about to be freed. After that the
|
||||||
|
* IRQ doorbell will not work anymore. So we need to disable
|
||||||
|
* IRQ here.
|
||||||
|
*/
|
||||||
|
bnxt_disable_int_sync(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 = &bnapi->cp_ring;
|
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
|
||||||
@ -4658,34 +4703,6 @@ static int bnxt_init_nic(struct bnxt *bp, bool irq_re_init)
|
|||||||
return bnxt_init_chip(bp, irq_re_init);
|
return bnxt_init_chip(bp, irq_re_init);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bnxt_disable_int(struct bnxt *bp)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
if (!bp->bnapi)
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (i = 0; i < bp->cp_nr_rings; i++) {
|
|
||||||
struct bnxt_napi *bnapi = bp->bnapi[i];
|
|
||||||
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
|
|
||||||
|
|
||||||
BNXT_CP_DB(cpr->cp_doorbell, cpr->cp_raw_cons);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void bnxt_enable_int(struct bnxt *bp)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
atomic_set(&bp->intr_sem, 0);
|
|
||||||
for (i = 0; i < bp->cp_nr_rings; i++) {
|
|
||||||
struct bnxt_napi *bnapi = bp->bnapi[i];
|
|
||||||
struct bnxt_cp_ring_info *cpr = &bnapi->cp_ring;
|
|
||||||
|
|
||||||
BNXT_CP_DB_REARM(cpr->cp_doorbell, cpr->cp_raw_cons);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int bnxt_set_real_num_queues(struct bnxt *bp)
|
static int bnxt_set_real_num_queues(struct bnxt *bp)
|
||||||
{
|
{
|
||||||
int rc;
|
int rc;
|
||||||
@ -5640,19 +5657,6 @@ static int bnxt_open(struct net_device *dev)
|
|||||||
return __bnxt_open_nic(bp, true, true);
|
return __bnxt_open_nic(bp, true, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bnxt_disable_int_sync(struct bnxt *bp)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
atomic_inc(&bp->intr_sem);
|
|
||||||
if (!netif_running(bp->dev))
|
|
||||||
return;
|
|
||||||
|
|
||||||
bnxt_disable_int(bp);
|
|
||||||
for (i = 0; i < bp->cp_nr_rings; i++)
|
|
||||||
synchronize_irq(bp->irq_tbl[i].vector);
|
|
||||||
}
|
|
||||||
|
|
||||||
int bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
int bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
||||||
{
|
{
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
@ -5674,13 +5678,12 @@ int bnxt_close_nic(struct bnxt *bp, bool irq_re_init, bool link_re_init)
|
|||||||
while (test_bit(BNXT_STATE_IN_SP_TASK, &bp->state))
|
while (test_bit(BNXT_STATE_IN_SP_TASK, &bp->state))
|
||||||
msleep(20);
|
msleep(20);
|
||||||
|
|
||||||
/* Flush rings before disabling interrupts */
|
/* Flush rings and and disable interrupts */
|
||||||
bnxt_shutdown_nic(bp, irq_re_init);
|
bnxt_shutdown_nic(bp, irq_re_init);
|
||||||
|
|
||||||
/* TODO CHIMP_FW: Link/PHY related cleanup if (link_re_init) */
|
/* TODO CHIMP_FW: Link/PHY related cleanup if (link_re_init) */
|
||||||
|
|
||||||
bnxt_disable_napi(bp);
|
bnxt_disable_napi(bp);
|
||||||
bnxt_disable_int_sync(bp);
|
|
||||||
del_timer_sync(&bp->timer);
|
del_timer_sync(&bp->timer);
|
||||||
bnxt_free_skbs(bp);
|
bnxt_free_skbs(bp);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user