cpuidle.c 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * CPU idle for DaVinci SoCs
  4. *
  5. * Copyright (C) 2009 Texas Instruments Incorporated. https://www.ti.com/
  6. *
  7. * Derived from Marvell Kirkwood CPU idle code
  8. * (arch/arm/mach-kirkwood/cpuidle.c)
  9. */
  10. #include <linux/kernel.h>
  11. #include <linux/init.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/cpuidle.h>
  14. #include <linux/io.h>
  15. #include <linux/export.h>
  16. #include <asm/cpuidle.h>
  17. #include "cpuidle.h"
  18. #include "ddr2.h"
  19. #define DAVINCI_CPUIDLE_MAX_STATES 2
  20. static void __iomem *ddr2_reg_base;
  21. static bool ddr2_pdown;
  22. static void davinci_save_ddr_power(int enter, bool pdown)
  23. {
  24. u32 val;
  25. val = __raw_readl(ddr2_reg_base + DDR2_SDRCR_OFFSET);
  26. if (enter) {
  27. if (pdown)
  28. val |= DDR2_SRPD_BIT;
  29. else
  30. val &= ~DDR2_SRPD_BIT;
  31. val |= DDR2_LPMODEN_BIT;
  32. } else {
  33. val &= ~(DDR2_SRPD_BIT | DDR2_LPMODEN_BIT);
  34. }
  35. __raw_writel(val, ddr2_reg_base + DDR2_SDRCR_OFFSET);
  36. }
  37. /* Actual code that puts the SoC in different idle states */
  38. static int davinci_enter_idle(struct cpuidle_device *dev,
  39. struct cpuidle_driver *drv, int index)
  40. {
  41. davinci_save_ddr_power(1, ddr2_pdown);
  42. cpu_do_idle();
  43. davinci_save_ddr_power(0, ddr2_pdown);
  44. return index;
  45. }
  46. static struct cpuidle_driver davinci_idle_driver = {
  47. .name = "cpuidle-davinci",
  48. .owner = THIS_MODULE,
  49. .states[0] = ARM_CPUIDLE_WFI_STATE,
  50. .states[1] = {
  51. .enter = davinci_enter_idle,
  52. .exit_latency = 10,
  53. .target_residency = 10000,
  54. .name = "DDR SR",
  55. .desc = "WFI and DDR Self Refresh",
  56. },
  57. .state_count = DAVINCI_CPUIDLE_MAX_STATES,
  58. };
  59. static int __init davinci_cpuidle_probe(struct platform_device *pdev)
  60. {
  61. struct davinci_cpuidle_config *pdata = pdev->dev.platform_data;
  62. if (!pdata) {
  63. dev_err(&pdev->dev, "cannot get platform data\n");
  64. return -ENOENT;
  65. }
  66. ddr2_reg_base = pdata->ddr2_ctlr_base;
  67. ddr2_pdown = pdata->ddr2_pdown;
  68. return cpuidle_register(&davinci_idle_driver, NULL);
  69. }
  70. static struct platform_driver davinci_cpuidle_driver = {
  71. .driver = {
  72. .name = "cpuidle-davinci",
  73. },
  74. };
  75. static int __init davinci_cpuidle_init(void)
  76. {
  77. return platform_driver_probe(&davinci_cpuidle_driver,
  78. davinci_cpuidle_probe);
  79. }
  80. device_initcall(davinci_cpuidle_init);