]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
bnxt_en: Fix logic that disables Bus Master during firmware reset.
authorVasundhara Volam <vasundhara-v.volam@broadcom.com>
Sun, 2 Feb 2020 07:41:37 +0000 (02:41 -0500)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Tue, 11 Feb 2020 12:36:38 +0000 (04:36 -0800)
[ Upstream commit d407302895d3f3ca3a333c711744a95e0b1b0150 ]

The current logic that calls pci_disable_device() in __bnxt_close_nic()
during firmware reset is flawed.  If firmware is still alive, we're
disabling the device too early, causing some firmware commands to
not reach the firmware.

Fix it by moving the logic to bnxt_reset_close().  If firmware is
in fatal condition, we call pci_disable_device() before we free
any of the rings to prevent DMA corruption of the freed rings.  If
firmware is still alive, we call pci_disable_device() after the
last firmware message has been sent.

Fixes: 3bc7d4a352ef ("bnxt_en: Add BNXT_STATE_IN_FW_RESET state.")
Signed-off-by: Vasundhara Volam <vasundhara-v.volam@broadcom.com>
Signed-off-by: Michael Chan <michael.chan@broadcom.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/net/ethernet/broadcom/bnxt/bnxt.c

index 5a779d949716e6a570e763022c0711e5e0c5bb88..01b603c5e76ad1b7a25c42f3dbc0143375b1f1b3 100644 (file)
@@ -9312,10 +9312,6 @@ static void __bnxt_close_nic(struct bnxt *bp, bool irq_re_init,
        bnxt_debug_dev_exit(bp);
        bnxt_disable_napi(bp);
        del_timer_sync(&bp->timer);
-       if (test_bit(BNXT_STATE_IN_FW_RESET, &bp->state) &&
-           pci_is_enabled(bp->pdev))
-               pci_disable_device(bp->pdev);
-
        bnxt_free_skbs(bp);
 
        /* Save ring stats before shutdown */
@@ -10093,9 +10089,16 @@ static void bnxt_reset(struct bnxt *bp, bool silent)
 static void bnxt_fw_reset_close(struct bnxt *bp)
 {
        bnxt_ulp_stop(bp);
+       /* When firmware is fatal state, disable PCI device to prevent
+        * any potential bad DMAs before freeing kernel memory.
+        */
+       if (test_bit(BNXT_STATE_FW_FATAL_COND, &bp->state))
+               pci_disable_device(bp->pdev);
        __bnxt_close_nic(bp, true, false);
        bnxt_clear_int_mode(bp);
        bnxt_hwrm_func_drv_unrgtr(bp);
+       if (pci_is_enabled(bp->pdev))
+               pci_disable_device(bp->pdev);
        bnxt_free_ctx_mem(bp);
        kfree(bp->ctx);
        bp->ctx = NULL;