kirkwood-cpufreq.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * kirkwood_freq.c: cpufreq driver for the Marvell kirkwood
  4. *
  5. * Copyright (C) 2013 Andrew Lunn <andrew@lunn.ch>
  6. */
  7. #include <linux/kernel.h>
  8. #include <linux/module.h>
  9. #include <linux/clk.h>
  10. #include <linux/cpufreq.h>
  11. #include <linux/of.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/io.h>
  14. #include <asm/proc-fns.h>
  15. #define CPU_SW_INT_BLK BIT(28)
  16. static struct priv
  17. {
  18. struct clk *cpu_clk;
  19. struct clk *ddr_clk;
  20. struct clk *powersave_clk;
  21. struct device *dev;
  22. void __iomem *base;
  23. } priv;
  24. #define STATE_CPU_FREQ 0x01
  25. #define STATE_DDR_FREQ 0x02
  26. /*
  27. * Kirkwood can swap the clock to the CPU between two clocks:
  28. *
  29. * - cpu clk
  30. * - ddr clk
  31. *
  32. * The frequencies are set at runtime before registering this table.
  33. */
  34. static struct cpufreq_frequency_table kirkwood_freq_table[] = {
  35. {0, STATE_CPU_FREQ, 0}, /* CPU uses cpuclk */
  36. {0, STATE_DDR_FREQ, 0}, /* CPU uses ddrclk */
  37. {0, 0, CPUFREQ_TABLE_END},
  38. };
  39. static unsigned int kirkwood_cpufreq_get_cpu_frequency(unsigned int cpu)
  40. {
  41. return clk_get_rate(priv.powersave_clk) / 1000;
  42. }
  43. static int kirkwood_cpufreq_target(struct cpufreq_policy *policy,
  44. unsigned int index)
  45. {
  46. unsigned int state = kirkwood_freq_table[index].driver_data;
  47. unsigned long reg;
  48. local_irq_disable();
  49. /* Disable interrupts to the CPU */
  50. reg = readl_relaxed(priv.base);
  51. reg |= CPU_SW_INT_BLK;
  52. writel_relaxed(reg, priv.base);
  53. switch (state) {
  54. case STATE_CPU_FREQ:
  55. clk_set_parent(priv.powersave_clk, priv.cpu_clk);
  56. break;
  57. case STATE_DDR_FREQ:
  58. clk_set_parent(priv.powersave_clk, priv.ddr_clk);
  59. break;
  60. }
  61. /* Wait-for-Interrupt, while the hardware changes frequency */
  62. cpu_do_idle();
  63. /* Enable interrupts to the CPU */
  64. reg = readl_relaxed(priv.base);
  65. reg &= ~CPU_SW_INT_BLK;
  66. writel_relaxed(reg, priv.base);
  67. local_irq_enable();
  68. return 0;
  69. }
  70. /* Module init and exit code */
  71. static int kirkwood_cpufreq_cpu_init(struct cpufreq_policy *policy)
  72. {
  73. cpufreq_generic_init(policy, kirkwood_freq_table, 5000);
  74. return 0;
  75. }
  76. static struct cpufreq_driver kirkwood_cpufreq_driver = {
  77. .flags = CPUFREQ_NEED_INITIAL_FREQ_CHECK,
  78. .get = kirkwood_cpufreq_get_cpu_frequency,
  79. .verify = cpufreq_generic_frequency_table_verify,
  80. .target_index = kirkwood_cpufreq_target,
  81. .init = kirkwood_cpufreq_cpu_init,
  82. .name = "kirkwood-cpufreq",
  83. };
  84. static int kirkwood_cpufreq_probe(struct platform_device *pdev)
  85. {
  86. struct device_node *np;
  87. int err;
  88. priv.dev = &pdev->dev;
  89. priv.base = devm_platform_ioremap_resource(pdev, 0);
  90. if (IS_ERR(priv.base))
  91. return PTR_ERR(priv.base);
  92. np = of_cpu_device_node_get(0);
  93. if (!np) {
  94. dev_err(&pdev->dev, "failed to get cpu device node\n");
  95. return -ENODEV;
  96. }
  97. priv.cpu_clk = of_clk_get_by_name(np, "cpu_clk");
  98. if (IS_ERR(priv.cpu_clk)) {
  99. dev_err(priv.dev, "Unable to get cpuclk\n");
  100. err = PTR_ERR(priv.cpu_clk);
  101. goto out_node;
  102. }
  103. err = clk_prepare_enable(priv.cpu_clk);
  104. if (err) {
  105. dev_err(priv.dev, "Unable to prepare cpuclk\n");
  106. goto out_node;
  107. }
  108. kirkwood_freq_table[0].frequency = clk_get_rate(priv.cpu_clk) / 1000;
  109. priv.ddr_clk = of_clk_get_by_name(np, "ddrclk");
  110. if (IS_ERR(priv.ddr_clk)) {
  111. dev_err(priv.dev, "Unable to get ddrclk\n");
  112. err = PTR_ERR(priv.ddr_clk);
  113. goto out_cpu;
  114. }
  115. err = clk_prepare_enable(priv.ddr_clk);
  116. if (err) {
  117. dev_err(priv.dev, "Unable to prepare ddrclk\n");
  118. goto out_cpu;
  119. }
  120. kirkwood_freq_table[1].frequency = clk_get_rate(priv.ddr_clk) / 1000;
  121. priv.powersave_clk = of_clk_get_by_name(np, "powersave");
  122. if (IS_ERR(priv.powersave_clk)) {
  123. dev_err(priv.dev, "Unable to get powersave\n");
  124. err = PTR_ERR(priv.powersave_clk);
  125. goto out_ddr;
  126. }
  127. err = clk_prepare_enable(priv.powersave_clk);
  128. if (err) {
  129. dev_err(priv.dev, "Unable to prepare powersave clk\n");
  130. goto out_ddr;
  131. }
  132. err = cpufreq_register_driver(&kirkwood_cpufreq_driver);
  133. if (err) {
  134. dev_err(priv.dev, "Failed to register cpufreq driver\n");
  135. goto out_powersave;
  136. }
  137. of_node_put(np);
  138. return 0;
  139. out_powersave:
  140. clk_disable_unprepare(priv.powersave_clk);
  141. out_ddr:
  142. clk_disable_unprepare(priv.ddr_clk);
  143. out_cpu:
  144. clk_disable_unprepare(priv.cpu_clk);
  145. out_node:
  146. of_node_put(np);
  147. return err;
  148. }
  149. static void kirkwood_cpufreq_remove(struct platform_device *pdev)
  150. {
  151. cpufreq_unregister_driver(&kirkwood_cpufreq_driver);
  152. clk_disable_unprepare(priv.powersave_clk);
  153. clk_disable_unprepare(priv.ddr_clk);
  154. clk_disable_unprepare(priv.cpu_clk);
  155. }
  156. static struct platform_driver kirkwood_cpufreq_platform_driver = {
  157. .probe = kirkwood_cpufreq_probe,
  158. .remove = kirkwood_cpufreq_remove,
  159. .driver = {
  160. .name = "kirkwood-cpufreq",
  161. },
  162. };
  163. module_platform_driver(kirkwood_cpufreq_platform_driver);
  164. MODULE_LICENSE("GPL v2");
  165. MODULE_AUTHOR("Andrew Lunn <andrew@lunn.ch");
  166. MODULE_DESCRIPTION("cpufreq driver for Marvell's kirkwood CPU");
  167. MODULE_ALIAS("platform:kirkwood-cpufreq");