e1000: fix race condition between e1000_down() and e1000_watchdog
authorVincenzo Maffione <v.maffione@gmail.com>
Sat, 16 Sep 2017 16:00:00 +0000 (18:00 +0200)
committerJeff Kirsher <jeffrey.t.kirsher@intel.com>
Thu, 26 Oct 2017 14:42:57 +0000 (07:42 -0700)
commit44c445c3d1b4eacff23141fa7977c3b2ec3a45c9
tree27f2d4565203b13f07fb9bd69615c179fed88008
parent78e0ea6791d7baafb8a0ca82b1bd0c7b3453c919
e1000: fix race condition between e1000_down() and e1000_watchdog

This patch fixes a race condition that can result into the interface being
up and carrier on, but with transmits disabled in the hardware.
The bug may show up by repeatedly IFF_DOWN+IFF_UP the interface, which
allows e1000_watchdog() interleave with e1000_down().

    CPU x                           CPU y
    --------------------------------------------------------------------
    e1000_down():
        netif_carrier_off()
                                    e1000_watchdog():
                                        if (carrier == off) {
                                            netif_carrier_on();
                                            enable_hw_transmit();
                                        }
        disable_hw_transmit();
                                    e1000_watchdog():
                                        /* carrier on, do nothing */

Signed-off-by: Vincenzo Maffione <v.maffione@gmail.com>
Tested-by: Aaron Brown <aaron.f.brown@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
drivers/net/ethernet/intel/e1000/e1000_main.c