summaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorNitin Kumbhar <nkumbhar@nvidia.com>2018-07-04 13:29:36 -0400
committermobile promotions <svcmobile_promotions@nvidia.com>2018-07-12 17:04:43 -0400
commit97c6a10928cc463825479911c3a6518ade34ebc0 (patch)
treee3a935b6dcf3099070bf30cc83718079458106ae /drivers
parent0b7fbc1ff457580cf8ae3f7a8a55f5ab36bfd8e7 (diff)
gpu: nvgpu: add pm_rumtime fixes
Runtime PM is enabled only for iGPU and not for dGPU. For dGPU, the .probe() of driver pm_runtime_disable()s, if rail-gating is not enabled. With nvgpu kernel module load/unload, .probe() is called multiple times for same struct device *. This results in an overflow of disable_depth (3 bit refcount) and enables runtime PM on 8th iteration and calls RTPM routines even if it's disabled. To effectively manage pm_runtime_disable(), move it from common nvgpu_remove() to iGPU/dGPU specific routines. Also, add restore pm_runtime state of device on driver .remove(). Bug 1987855 Change-Id: I781278da546ef9c9ef7d7da7dbea0757df32716f Signed-off-by: Nitin Kumbhar <nkumbhar@nvidia.com> Reviewed-on: https://git-master.nvidia.com/r/1770804 Reviewed-by: mobile promotions <svcmobile_promotions@nvidia.com> Tested-by: mobile promotions <svcmobile_promotions@nvidia.com>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/gpu/nvgpu/os/linux/module.c13
-rw-r--r--drivers/gpu/nvgpu/os/linux/pci.c39
2 files changed, 44 insertions, 8 deletions
diff --git a/drivers/gpu/nvgpu/os/linux/module.c b/drivers/gpu/nvgpu/os/linux/module.c
index a59ac1f1..e3667947 100644
--- a/drivers/gpu/nvgpu/os/linux/module.c
+++ b/drivers/gpu/nvgpu/os/linux/module.c
@@ -1060,6 +1060,13 @@ static int gk20a_pm_init(struct device *dev)
1060 return err; 1060 return err;
1061} 1061}
1062 1062
1063static int gk20a_pm_deinit(struct device *dev)
1064{
1065 pm_runtime_dont_use_autosuspend(dev);
1066 pm_runtime_disable(dev);
1067 return 0;
1068}
1069
1063/* 1070/*
1064 * Start the process for unloading the driver. Set NVGPU_DRIVER_IS_DYING. 1071 * Start the process for unloading the driver. Set NVGPU_DRIVER_IS_DYING.
1065 */ 1072 */
@@ -1309,9 +1316,6 @@ int nvgpu_remove(struct device *dev, struct class *class)
1309 platform->secure_buffer.destroy(g, 1316 platform->secure_buffer.destroy(g,
1310 &platform->secure_buffer); 1317 &platform->secure_buffer);
1311 1318
1312 if (pm_runtime_enabled(dev))
1313 pm_runtime_disable(dev);
1314
1315 if (platform->remove) 1319 if (platform->remove)
1316 platform->remove(dev); 1320 platform->remove(dev);
1317 1321
@@ -1332,8 +1336,11 @@ static int __exit gk20a_remove(struct platform_device *pdev)
1332 err = nvgpu_remove(dev, &nvgpu_class); 1336 err = nvgpu_remove(dev, &nvgpu_class);
1333 1337
1334 set_gk20a(pdev, NULL); 1338 set_gk20a(pdev, NULL);
1339
1335 gk20a_put(g); 1340 gk20a_put(g);
1336 1341
1342 gk20a_pm_deinit(dev);
1343
1337 return err; 1344 return err;
1338} 1345}
1339 1346
diff --git a/drivers/gpu/nvgpu/os/linux/pci.c b/drivers/gpu/nvgpu/os/linux/pci.c
index dba8a5df..3493b105 100644
--- a/drivers/gpu/nvgpu/os/linux/pci.c
+++ b/drivers/gpu/nvgpu/os/linux/pci.c
@@ -590,11 +590,21 @@ static int nvgpu_pci_pm_runtime_suspend(struct device *dev)
590 return 0; 590 return 0;
591} 591}
592 592
593static int nvgpu_pci_pm_resume(struct device *dev)
594{
595 return gk20a_pm_finalize_poweron(dev);
596}
597
598static int nvgpu_pci_pm_suspend(struct device *dev)
599{
600 return 0;
601}
602
593static const struct dev_pm_ops nvgpu_pci_pm_ops = { 603static const struct dev_pm_ops nvgpu_pci_pm_ops = {
594 .runtime_resume = nvgpu_pci_pm_runtime_resume, 604 .runtime_resume = nvgpu_pci_pm_runtime_resume,
595 .runtime_suspend = nvgpu_pci_pm_runtime_suspend, 605 .runtime_suspend = nvgpu_pci_pm_runtime_suspend,
596 .resume = nvgpu_pci_pm_runtime_resume, 606 .resume = nvgpu_pci_pm_resume,
597 .suspend = nvgpu_pci_pm_runtime_suspend, 607 .suspend = nvgpu_pci_pm_suspend,
598}; 608};
599#endif 609#endif
600 610
@@ -611,10 +621,15 @@ static int nvgpu_pci_pm_init(struct device *dev)
611 g->railgate_delay); 621 g->railgate_delay);
612 622
613 /* 623 /*
614 * Runtime PM for PCI devices is disabled by default, 624 * set gpu dev's use_autosuspend flag to allow
615 * so we need to enable it first 625 * runtime power management of GPU
616 */ 626 */
617 pm_runtime_use_autosuspend(dev); 627 pm_runtime_use_autosuspend(dev);
628
629 /*
630 * runtime PM for PCI devices is forbidden
631 * by default, so unblock RTPM of GPU
632 */
618 pm_runtime_put_noidle(dev); 633 pm_runtime_put_noidle(dev);
619 pm_runtime_allow(dev); 634 pm_runtime_allow(dev);
620 } 635 }
@@ -622,6 +637,19 @@ static int nvgpu_pci_pm_init(struct device *dev)
622 return 0; 637 return 0;
623} 638}
624 639
640static int nvgpu_pci_pm_deinit(struct device *dev)
641{
642#ifdef CONFIG_PM
643 struct gk20a *g = get_gk20a(dev);
644
645 if (!nvgpu_is_enabled(g, NVGPU_CAN_RAILGATE))
646 pm_runtime_enable(dev);
647 else
648 pm_runtime_forbid(dev);
649#endif
650 return 0;
651}
652
625static int nvgpu_pci_probe(struct pci_dev *pdev, 653static int nvgpu_pci_probe(struct pci_dev *pdev,
626 const struct pci_device_id *pent) 654 const struct pci_device_id *pent)
627{ 655{
@@ -826,11 +854,12 @@ static void nvgpu_pci_remove(struct pci_dev *pdev)
826 enable_irq(g->irq_stall); 854 enable_irq(g->irq_stall);
827 } 855 }
828#endif 856#endif
857 nvgpu_pci_pm_deinit(&pdev->dev);
829 858
830 /* free allocated platform data space */ 859 /* free allocated platform data space */
860 gk20a_get_platform(&pdev->dev)->g = NULL;
831 nvgpu_kfree(g, gk20a_get_platform(&pdev->dev)); 861 nvgpu_kfree(g, gk20a_get_platform(&pdev->dev));
832 862
833 gk20a_get_platform(&pdev->dev)->g = NULL;
834 gk20a_put(g); 863 gk20a_put(g);
835} 864}
836 865