cpuidle-mvebu-v7.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. /*
  2. * Marvell Armada 370, 38x and XP SoC cpuidle driver
  3. *
  4. * Copyright (C) 2014 Marvell
  5. *
  6. * Nadav Haklai <nadavh@marvell.com>
  7. * Gregory CLEMENT <gregory.clement@free-electrons.com>
  8. *
  9. * This file is licensed under the terms of the GNU General Public
  10. * License version 2. This program is licensed "as is" without any
  11. * warranty of any kind, whether express or implied.
  12. *
  13. * Maintainer: Gregory CLEMENT <gregory.clement@free-electrons.com>
  14. */
  15. #include <linux/cpu_pm.h>
  16. #include <linux/cpuidle.h>
  17. #include <linux/module.h>
  18. #include <linux/of.h>
  19. #include <linux/suspend.h>
  20. #include <linux/platform_device.h>
  21. #include <asm/cpuidle.h>
  22. #define MVEBU_V7_FLAG_DEEP_IDLE 0x10000
  23. static int (*mvebu_v7_cpu_suspend)(int);
  24. static int mvebu_v7_enter_idle(struct cpuidle_device *dev,
  25. struct cpuidle_driver *drv,
  26. int index)
  27. {
  28. int ret;
  29. bool deepidle = false;
  30. cpu_pm_enter();
  31. if (drv->states[index].flags & MVEBU_V7_FLAG_DEEP_IDLE)
  32. deepidle = true;
  33. ret = mvebu_v7_cpu_suspend(deepidle);
  34. cpu_pm_exit();
  35. if (ret)
  36. return ret;
  37. return index;
  38. }
  39. static struct cpuidle_driver armadaxp_idle_driver = {
  40. .name = "armada_xp_idle",
  41. .states[0] = ARM_CPUIDLE_WFI_STATE,
  42. .states[1] = {
  43. .enter = mvebu_v7_enter_idle,
  44. .exit_latency = 100,
  45. .power_usage = 50,
  46. .target_residency = 1000,
  47. .name = "MV CPU IDLE",
  48. .desc = "CPU power down",
  49. },
  50. .states[2] = {
  51. .enter = mvebu_v7_enter_idle,
  52. .exit_latency = 1000,
  53. .power_usage = 5,
  54. .target_residency = 10000,
  55. .flags = MVEBU_V7_FLAG_DEEP_IDLE,
  56. .name = "MV CPU DEEP IDLE",
  57. .desc = "CPU and L2 Fabric power down",
  58. },
  59. .state_count = 3,
  60. };
  61. static struct cpuidle_driver armada370_idle_driver = {
  62. .name = "armada_370_idle",
  63. .states[0] = ARM_CPUIDLE_WFI_STATE,
  64. .states[1] = {
  65. .enter = mvebu_v7_enter_idle,
  66. .exit_latency = 100,
  67. .power_usage = 5,
  68. .target_residency = 1000,
  69. .flags = MVEBU_V7_FLAG_DEEP_IDLE,
  70. .name = "Deep Idle",
  71. .desc = "CPU and L2 Fabric power down",
  72. },
  73. .state_count = 2,
  74. };
  75. static struct cpuidle_driver armada38x_idle_driver = {
  76. .name = "armada_38x_idle",
  77. .states[0] = ARM_CPUIDLE_WFI_STATE,
  78. .states[1] = {
  79. .enter = mvebu_v7_enter_idle,
  80. .exit_latency = 10,
  81. .power_usage = 5,
  82. .target_residency = 100,
  83. .name = "Idle",
  84. .desc = "CPU and SCU power down",
  85. },
  86. .state_count = 2,
  87. };
  88. static int mvebu_v7_cpuidle_probe(struct platform_device *pdev)
  89. {
  90. mvebu_v7_cpu_suspend = pdev->dev.platform_data;
  91. if (!strcmp(pdev->dev.driver->name, "cpuidle-armada-xp"))
  92. return cpuidle_register(&armadaxp_idle_driver, NULL);
  93. else if (!strcmp(pdev->dev.driver->name, "cpuidle-armada-370"))
  94. return cpuidle_register(&armada370_idle_driver, NULL);
  95. else if (!strcmp(pdev->dev.driver->name, "cpuidle-armada-38x"))
  96. return cpuidle_register(&armada38x_idle_driver, NULL);
  97. else
  98. return -EINVAL;
  99. }
  100. static struct platform_driver armadaxp_cpuidle_plat_driver = {
  101. .driver = {
  102. .name = "cpuidle-armada-xp",
  103. },
  104. .probe = mvebu_v7_cpuidle_probe,
  105. };
  106. module_platform_driver(armadaxp_cpuidle_plat_driver);
  107. static struct platform_driver armada370_cpuidle_plat_driver = {
  108. .driver = {
  109. .name = "cpuidle-armada-370",
  110. },
  111. .probe = mvebu_v7_cpuidle_probe,
  112. };
  113. module_platform_driver(armada370_cpuidle_plat_driver);
  114. static struct platform_driver armada38x_cpuidle_plat_driver = {
  115. .driver = {
  116. .name = "cpuidle-armada-38x",
  117. },
  118. .probe = mvebu_v7_cpuidle_probe,
  119. };
  120. module_platform_driver(armada38x_cpuidle_plat_driver);
  121. MODULE_AUTHOR("Gregory CLEMENT <gregory.clement@free-electrons.com>");
  122. MODULE_DESCRIPTION("Marvell EBU v7 cpuidle driver");
  123. MODULE_LICENSE("GPL");