lima_devfreq.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright 2020 Martin Blumenstingl <martin.blumenstingl@googlemail.com>
  4. *
  5. * Based on panfrost_devfreq.c:
  6. * Copyright 2019 Collabora ltd.
  7. */
  8. #include <linux/clk.h>
  9. #include <linux/devfreq.h>
  10. #include <linux/devfreq_cooling.h>
  11. #include <linux/device.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/pm_opp.h>
  14. #include <linux/property.h>
  15. #include "lima_device.h"
  16. #include "lima_devfreq.h"
  17. static void lima_devfreq_update_utilization(struct lima_devfreq *devfreq)
  18. {
  19. ktime_t now, last;
  20. now = ktime_get();
  21. last = devfreq->time_last_update;
  22. if (devfreq->busy_count > 0)
  23. devfreq->busy_time += ktime_sub(now, last);
  24. else
  25. devfreq->idle_time += ktime_sub(now, last);
  26. devfreq->time_last_update = now;
  27. }
  28. static int lima_devfreq_target(struct device *dev, unsigned long *freq,
  29. u32 flags)
  30. {
  31. struct dev_pm_opp *opp;
  32. opp = devfreq_recommended_opp(dev, freq, flags);
  33. if (IS_ERR(opp))
  34. return PTR_ERR(opp);
  35. dev_pm_opp_put(opp);
  36. return dev_pm_opp_set_rate(dev, *freq);
  37. }
  38. static void lima_devfreq_reset(struct lima_devfreq *devfreq)
  39. {
  40. devfreq->busy_time = 0;
  41. devfreq->idle_time = 0;
  42. devfreq->time_last_update = ktime_get();
  43. }
  44. static int lima_devfreq_get_dev_status(struct device *dev,
  45. struct devfreq_dev_status *status)
  46. {
  47. struct lima_device *ldev = dev_get_drvdata(dev);
  48. struct lima_devfreq *devfreq = &ldev->devfreq;
  49. unsigned long irqflags;
  50. status->current_frequency = clk_get_rate(ldev->clk_gpu);
  51. spin_lock_irqsave(&devfreq->lock, irqflags);
  52. lima_devfreq_update_utilization(devfreq);
  53. status->total_time = ktime_to_ns(ktime_add(devfreq->busy_time,
  54. devfreq->idle_time));
  55. status->busy_time = ktime_to_ns(devfreq->busy_time);
  56. lima_devfreq_reset(devfreq);
  57. spin_unlock_irqrestore(&devfreq->lock, irqflags);
  58. dev_dbg(ldev->dev, "busy %lu total %lu %lu %% freq %lu MHz\n",
  59. status->busy_time, status->total_time,
  60. status->busy_time / (status->total_time / 100),
  61. status->current_frequency / 1000 / 1000);
  62. return 0;
  63. }
  64. static struct devfreq_dev_profile lima_devfreq_profile = {
  65. .timer = DEVFREQ_TIMER_DELAYED,
  66. .polling_ms = 50, /* ~3 frames */
  67. .target = lima_devfreq_target,
  68. .get_dev_status = lima_devfreq_get_dev_status,
  69. };
  70. void lima_devfreq_fini(struct lima_device *ldev)
  71. {
  72. struct lima_devfreq *devfreq = &ldev->devfreq;
  73. if (devfreq->cooling) {
  74. devfreq_cooling_unregister(devfreq->cooling);
  75. devfreq->cooling = NULL;
  76. }
  77. if (devfreq->devfreq) {
  78. devm_devfreq_remove_device(ldev->dev, devfreq->devfreq);
  79. devfreq->devfreq = NULL;
  80. }
  81. }
  82. int lima_devfreq_init(struct lima_device *ldev)
  83. {
  84. struct thermal_cooling_device *cooling;
  85. struct device *dev = ldev->dev;
  86. struct devfreq *devfreq;
  87. struct lima_devfreq *ldevfreq = &ldev->devfreq;
  88. struct dev_pm_opp *opp;
  89. unsigned long cur_freq;
  90. int ret;
  91. const char *regulator_names[] = { "mali", NULL };
  92. if (!device_property_present(dev, "operating-points-v2"))
  93. /* Optional, continue without devfreq */
  94. return 0;
  95. spin_lock_init(&ldevfreq->lock);
  96. /*
  97. * clkname is set separately so it is not affected by the optional
  98. * regulator setting which may return error.
  99. */
  100. ret = devm_pm_opp_set_clkname(dev, "core");
  101. if (ret)
  102. return ret;
  103. ret = devm_pm_opp_set_regulators(dev, regulator_names);
  104. if (ret) {
  105. /* Continue if the optional regulator is missing */
  106. if (ret != -ENODEV)
  107. return ret;
  108. }
  109. ret = devm_pm_opp_of_add_table(dev);
  110. if (ret)
  111. return ret;
  112. lima_devfreq_reset(ldevfreq);
  113. cur_freq = clk_get_rate(ldev->clk_gpu);
  114. opp = devfreq_recommended_opp(dev, &cur_freq, 0);
  115. if (IS_ERR(opp))
  116. return PTR_ERR(opp);
  117. lima_devfreq_profile.initial_freq = cur_freq;
  118. dev_pm_opp_put(opp);
  119. /*
  120. * Setup default thresholds for the simple_ondemand governor.
  121. * The values are chosen based on experiments.
  122. */
  123. ldevfreq->gov_data.upthreshold = 30;
  124. ldevfreq->gov_data.downdifferential = 5;
  125. devfreq = devm_devfreq_add_device(dev, &lima_devfreq_profile,
  126. DEVFREQ_GOV_SIMPLE_ONDEMAND,
  127. &ldevfreq->gov_data);
  128. if (IS_ERR(devfreq)) {
  129. dev_err(dev, "Couldn't initialize GPU devfreq\n");
  130. return PTR_ERR(devfreq);
  131. }
  132. ldevfreq->devfreq = devfreq;
  133. cooling = of_devfreq_cooling_register(dev->of_node, devfreq);
  134. if (IS_ERR(cooling))
  135. dev_info(dev, "Failed to register cooling device\n");
  136. else
  137. ldevfreq->cooling = cooling;
  138. return 0;
  139. }
  140. void lima_devfreq_record_busy(struct lima_devfreq *devfreq)
  141. {
  142. unsigned long irqflags;
  143. if (!devfreq->devfreq)
  144. return;
  145. spin_lock_irqsave(&devfreq->lock, irqflags);
  146. lima_devfreq_update_utilization(devfreq);
  147. devfreq->busy_count++;
  148. spin_unlock_irqrestore(&devfreq->lock, irqflags);
  149. }
  150. void lima_devfreq_record_idle(struct lima_devfreq *devfreq)
  151. {
  152. unsigned long irqflags;
  153. if (!devfreq->devfreq)
  154. return;
  155. spin_lock_irqsave(&devfreq->lock, irqflags);
  156. lima_devfreq_update_utilization(devfreq);
  157. WARN_ON(--devfreq->busy_count < 0);
  158. spin_unlock_irqrestore(&devfreq->lock, irqflags);
  159. }
  160. int lima_devfreq_resume(struct lima_devfreq *devfreq)
  161. {
  162. unsigned long irqflags;
  163. if (!devfreq->devfreq)
  164. return 0;
  165. spin_lock_irqsave(&devfreq->lock, irqflags);
  166. lima_devfreq_reset(devfreq);
  167. spin_unlock_irqrestore(&devfreq->lock, irqflags);
  168. return devfreq_resume_device(devfreq->devfreq);
  169. }
  170. int lima_devfreq_suspend(struct lima_devfreq *devfreq)
  171. {
  172. if (!devfreq->devfreq)
  173. return 0;
  174. return devfreq_suspend_device(devfreq->devfreq);
  175. }