]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
net: mana: Support hibernation and kexec
authorDexuan Cui <decui@microsoft.com>
Sat, 30 Oct 2021 00:54:08 +0000 (17:54 -0700)
committerDavid S. Miller <davem@davemloft.net>
Mon, 1 Nov 2021 13:21:49 +0000 (13:21 +0000)
Implement the suspend/resume/shutdown callbacks for hibernation/kexec.

Add mana_gd_setup() and mana_gd_cleanup() for some common code, and
use them in the mand_gd_* callbacks.

Reuse mana_probe/remove() for the hibernation path.

Signed-off-by: Dexuan Cui <decui@microsoft.com>
Reviewed-by: Haiyang Zhang <haiyangz@microsoft.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/microsoft/mana/gdma_main.c
drivers/net/ethernet/microsoft/mana/mana.h
drivers/net/ethernet/microsoft/mana/mana_en.c

index 599dfd5e60901899a778084741cd45abd81e3e2c..c96ac81212f74c899c53782d6c12fcd8b85b6d78 100644 (file)
@@ -1258,6 +1258,52 @@ static void mana_gd_remove_irqs(struct pci_dev *pdev)
        gc->irq_contexts = NULL;
 }
 
+static int mana_gd_setup(struct pci_dev *pdev)
+{
+       struct gdma_context *gc = pci_get_drvdata(pdev);
+       int err;
+
+       mana_gd_init_registers(pdev);
+       mana_smc_init(&gc->shm_channel, gc->dev, gc->shm_base);
+
+       err = mana_gd_setup_irqs(pdev);
+       if (err)
+               return err;
+
+       err = mana_hwc_create_channel(gc);
+       if (err)
+               goto remove_irq;
+
+       err = mana_gd_verify_vf_version(pdev);
+       if (err)
+               goto destroy_hwc;
+
+       err = mana_gd_query_max_resources(pdev);
+       if (err)
+               goto destroy_hwc;
+
+       err = mana_gd_detect_devices(pdev);
+       if (err)
+               goto destroy_hwc;
+
+       return 0;
+
+destroy_hwc:
+       mana_hwc_destroy_channel(gc);
+remove_irq:
+       mana_gd_remove_irqs(pdev);
+       return err;
+}
+
+static void mana_gd_cleanup(struct pci_dev *pdev)
+{
+       struct gdma_context *gc = pci_get_drvdata(pdev);
+
+       mana_hwc_destroy_channel(gc);
+
+       mana_gd_remove_irqs(pdev);
+}
+
 static int mana_gd_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
 {
        struct gdma_context *gc;
@@ -1287,6 +1333,9 @@ static int mana_gd_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
        if (!gc)
                goto release_region;
 
+       mutex_init(&gc->eq_test_event_mutex);
+       pci_set_drvdata(pdev, gc);
+
        bar0_va = pci_iomap(pdev, bar, 0);
        if (!bar0_va)
                goto free_gc;
@@ -1294,47 +1343,23 @@ static int mana_gd_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
        gc->bar0_va = bar0_va;
        gc->dev = &pdev->dev;
 
-       pci_set_drvdata(pdev, gc);
-
-       mana_gd_init_registers(pdev);
 
-       mana_smc_init(&gc->shm_channel, gc->dev, gc->shm_base);
-
-       err = mana_gd_setup_irqs(pdev);
+       err = mana_gd_setup(pdev);
        if (err)
                goto unmap_bar;
 
-       mutex_init(&gc->eq_test_event_mutex);
-
-       err = mana_hwc_create_channel(gc);
+       err = mana_probe(&gc->mana, false);
        if (err)
-               goto remove_irq;
-
-       err = mana_gd_verify_vf_version(pdev);
-       if (err)
-               goto remove_irq;
-
-       err = mana_gd_query_max_resources(pdev);
-       if (err)
-               goto remove_irq;
-
-       err = mana_gd_detect_devices(pdev);
-       if (err)
-               goto remove_irq;
-
-       err = mana_probe(&gc->mana);
-       if (err)
-               goto clean_up_gdma;
+               goto cleanup_gd;
 
        return 0;
 
-clean_up_gdma:
-       mana_hwc_destroy_channel(gc);
-remove_irq:
-       mana_gd_remove_irqs(pdev);
+cleanup_gd:
+       mana_gd_cleanup(pdev);
 unmap_bar:
        pci_iounmap(pdev, bar0_va);
 free_gc:
+       pci_set_drvdata(pdev, NULL);
        vfree(gc);
 release_region:
        pci_release_regions(pdev);
@@ -1349,11 +1374,9 @@ static void mana_gd_remove(struct pci_dev *pdev)
 {
        struct gdma_context *gc = pci_get_drvdata(pdev);
 
-       mana_remove(&gc->mana);
+       mana_remove(&gc->mana, false);
 
-       mana_hwc_destroy_channel(gc);
-
-       mana_gd_remove_irqs(pdev);
+       mana_gd_cleanup(pdev);
 
        pci_iounmap(pdev, gc->bar0_va);
 
@@ -1364,6 +1387,52 @@ static void mana_gd_remove(struct pci_dev *pdev)
        pci_disable_device(pdev);
 }
 
+/* The 'state' parameter is not used. */
+static int mana_gd_suspend(struct pci_dev *pdev, pm_message_t state)
+{
+       struct gdma_context *gc = pci_get_drvdata(pdev);
+
+       mana_remove(&gc->mana, true);
+
+       mana_gd_cleanup(pdev);
+
+       return 0;
+}
+
+/* In case the NIC hardware stops working, the suspend and resume callbacks will
+ * fail -- if this happens, it's safer to just report an error than try to undo
+ * what has been done.
+ */
+static int mana_gd_resume(struct pci_dev *pdev)
+{
+       struct gdma_context *gc = pci_get_drvdata(pdev);
+       int err;
+
+       err = mana_gd_setup(pdev);
+       if (err)
+               return err;
+
+       err = mana_probe(&gc->mana, true);
+       if (err)
+               return err;
+
+       return 0;
+}
+
+/* Quiesce the device for kexec. This is also called upon reboot/shutdown. */
+static void mana_gd_shutdown(struct pci_dev *pdev)
+{
+       struct gdma_context *gc = pci_get_drvdata(pdev);
+
+       dev_info(&pdev->dev, "Shutdown was calledd\n");
+
+       mana_remove(&gc->mana, true);
+
+       mana_gd_cleanup(pdev);
+
+       pci_disable_device(pdev);
+}
+
 #ifndef PCI_VENDOR_ID_MICROSOFT
 #define PCI_VENDOR_ID_MICROSOFT 0x1414
 #endif
@@ -1378,6 +1447,9 @@ static struct pci_driver mana_driver = {
        .id_table       = mana_id_table,
        .probe          = mana_gd_probe,
        .remove         = mana_gd_remove,
+       .suspend        = mana_gd_suspend,
+       .resume         = mana_gd_resume,
+       .shutdown       = mana_gd_shutdown,
 };
 
 module_pci_driver(mana_driver);
index fc98a5ba5ed0701fbc5ff2118864fc9b016c6c5c..d047ee876f12b2fa7bbba0a69d898ba9a36f9cdc 100644 (file)
@@ -374,8 +374,8 @@ int mana_alloc_queues(struct net_device *ndev);
 int mana_attach(struct net_device *ndev);
 int mana_detach(struct net_device *ndev, bool from_close);
 
-int mana_probe(struct gdma_dev *gd);
-void mana_remove(struct gdma_dev *gd);
+int mana_probe(struct gdma_dev *gd, bool resuming);
+void mana_remove(struct gdma_dev *gd, bool suspending);
 
 extern const struct ethtool_ops mana_ethtool_ops;
 
index 46ef9f47d9c577a508b4cd37207c9ad3fb47548d..72cbf45c42d88ff2a33d129676793c7ee2514509 100644 (file)
@@ -1828,11 +1828,12 @@ free_net:
        return err;
 }
 
-int mana_probe(struct gdma_dev *gd)
+int mana_probe(struct gdma_dev *gd, bool resuming)
 {
        struct gdma_context *gc = gd->gdma_context;
+       struct mana_context *ac = gd->driver_data;
        struct device *dev = gc->dev;
-       struct mana_context *ac;
+       u16 num_ports = 0;
        int err;
        int i;
 
@@ -1844,44 +1845,70 @@ int mana_probe(struct gdma_dev *gd)
        if (err)
                return err;
 
-       ac = kzalloc(sizeof(*ac), GFP_KERNEL);
-       if (!ac)
-               return -ENOMEM;
+       if (!resuming) {
+               ac = kzalloc(sizeof(*ac), GFP_KERNEL);
+               if (!ac)
+                       return -ENOMEM;
 
-       ac->gdma_dev = gd;
-       ac->num_ports = 1;
-       gd->driver_data = ac;
+               ac->gdma_dev = gd;
+               gd->driver_data = ac;
+       }
 
        err = mana_create_eq(ac);
        if (err)
                goto out;
 
        err = mana_query_device_cfg(ac, MANA_MAJOR_VERSION, MANA_MINOR_VERSION,
-                                   MANA_MICRO_VERSION, &ac->num_ports);
+                                   MANA_MICRO_VERSION, &num_ports);
        if (err)
                goto out;
 
+       if (!resuming) {
+               ac->num_ports = num_ports;
+       } else {
+               if (ac->num_ports != num_ports) {
+                       dev_err(dev, "The number of vPorts changed: %d->%d\n",
+                               ac->num_ports, num_ports);
+                       err = -EPROTO;
+                       goto out;
+               }
+       }
+
+       if (ac->num_ports == 0)
+               dev_err(dev, "Failed to detect any vPort\n");
+
        if (ac->num_ports > MAX_PORTS_IN_MANA_DEV)
                ac->num_ports = MAX_PORTS_IN_MANA_DEV;
 
-       for (i = 0; i < ac->num_ports; i++) {
-               err = mana_probe_port(ac, i, &ac->ports[i]);
-               if (err)
-                       break;
+       if (!resuming) {
+               for (i = 0; i < ac->num_ports; i++) {
+                       err = mana_probe_port(ac, i, &ac->ports[i]);
+                       if (err)
+                               break;
+               }
+       } else {
+               for (i = 0; i < ac->num_ports; i++) {
+                       rtnl_lock();
+                       err = mana_attach(ac->ports[i]);
+                       rtnl_unlock();
+                       if (err)
+                               break;
+               }
        }
 out:
        if (err)
-               mana_remove(gd);
+               mana_remove(gd, false);
 
        return err;
 }
 
-void mana_remove(struct gdma_dev *gd)
+void mana_remove(struct gdma_dev *gd, bool suspending)
 {
        struct gdma_context *gc = gd->gdma_context;
        struct mana_context *ac = gd->driver_data;
        struct device *dev = gc->dev;
        struct net_device *ndev;
+       int err;
        int i;
 
        for (i = 0; i < ac->num_ports; i++) {
@@ -1897,7 +1924,16 @@ void mana_remove(struct gdma_dev *gd)
                 */
                rtnl_lock();
 
-               mana_detach(ndev, false);
+               err = mana_detach(ndev, false);
+               if (err)
+                       netdev_err(ndev, "Failed to detach vPort %d: %d\n",
+                                  i, err);
+
+               if (suspending) {
+                       /* No need to unregister the ndev. */
+                       rtnl_unlock();
+                       continue;
+               }
 
                unregister_netdevice(ndev);
 
@@ -1910,6 +1946,10 @@ void mana_remove(struct gdma_dev *gd)
 
 out:
        mana_gd_deregister_device(gd);
+
+       if (suspending)
+               return;
+
        gd->driver_data = NULL;
        gd->gdma_context = NULL;
        kfree(ac);