3 * (C) COPYRIGHT 2014-2016 ARM Limited. All rights reserved.
5 * This program is free software and is provided to you under the terms of the
6 * GNU General Public License version 2 as published by the Free Software
7 * Foundation, and any use by you of this program is subject to the terms
10 * A copy of the licence is included with the program, and can also be obtained
11 * from Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
12 * Boston, MA 02110-1301, USA.
17 #define ENABLE_DEBUG_LOG
18 #include "../../platform/rk/custom_log.h"
21 #include <mali_kbase.h>
22 #include <mali_kbase_tlstream.h>
23 #include <mali_kbase_config_defaults.h>
24 #include <backend/gpu/mali_kbase_pm_internal.h>
25 #ifdef CONFIG_DEVFREQ_THERMAL
26 #include <backend/gpu/mali_kbase_power_model_simple.h>
29 #include <linux/clk.h>
30 #include <linux/devfreq.h>
31 #ifdef CONFIG_DEVFREQ_THERMAL
32 #include <linux/devfreq_cooling.h>
35 #include <linux/version.h>
36 #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 13, 0)
37 #include <linux/pm_opp.h>
38 #else /* Linux >= 3.13 */
39 /* In 3.13 the OPP include header file, types, and functions were all
40 * renamed. Use the old filename for the include, and define the new names to
41 * the old, when an old kernel is detected.
43 #include <linux/opp.h>
44 #define dev_pm_opp opp
45 #define dev_pm_opp_get_voltage opp_get_voltage
46 #define dev_pm_opp_get_opp_count opp_get_opp_count
47 #define dev_pm_opp_find_freq_ceil opp_find_freq_ceil
48 #endif /* Linux >= 3.13 */
52 kbase_devfreq_target(struct device *dev, unsigned long *target_freq, u32 flags)
54 struct kbase_device *kbdev = dev_get_drvdata(dev);
55 struct dev_pm_opp *opp;
56 unsigned long freq = 0;
57 unsigned long voltage;
63 opp = devfreq_recommended_opp(dev, &freq, flags);
64 voltage = dev_pm_opp_get_voltage(opp);
66 if (IS_ERR_OR_NULL(opp)) {
67 dev_err(dev, "Failed to get opp (%ld)\n", PTR_ERR(opp));
72 * Only update if there is a change of frequency
74 if (kbdev->current_freq == freq) {
79 #ifdef CONFIG_REGULATOR
80 if (kbdev->regulator && kbdev->current_voltage != voltage
81 && kbdev->current_freq < freq) {
82 err = regulator_set_voltage(kbdev->regulator, voltage, voltage);
84 dev_err(dev, "Failed to increase voltage (%d)\n", err);
90 err = clk_set_rate(kbdev->clock, freq);
92 dev_err(dev, "Failed to set clock %lu (target %lu)\n",
97 #ifdef CONFIG_REGULATOR
98 if (kbdev->regulator && kbdev->current_voltage != voltage
99 && kbdev->current_freq > freq) {
100 err = regulator_set_voltage(kbdev->regulator, voltage, voltage);
102 dev_err(dev, "Failed to decrease voltage (%d)\n", err);
109 kbdev->current_voltage = voltage;
110 kbdev->current_freq = freq;
112 kbase_tlstream_aux_devfreq_target((u64)freq);
114 kbase_pm_reset_dvfs_utilisation(kbdev);
120 kbase_devfreq_cur_freq(struct device *dev, unsigned long *freq)
122 struct kbase_device *kbdev = dev_get_drvdata(dev);
124 *freq = kbdev->current_freq;
130 kbase_devfreq_status(struct device *dev, struct devfreq_dev_status *stat)
132 struct kbase_device *kbdev = dev_get_drvdata(dev);
134 stat->current_frequency = kbdev->current_freq;
136 kbase_pm_get_dvfs_utilisation(kbdev,
137 &stat->total_time, &stat->busy_time);
139 stat->private_data = NULL;
141 #ifdef CONFIG_DEVFREQ_THERMAL
142 #if LINUX_VERSION_CODE < KERNEL_VERSION(4, 3, 0)
143 if (kbdev->devfreq_cooling)
144 memcpy(&kbdev->devfreq_cooling->last_status, stat,
152 static int kbase_devfreq_init_freq_table(struct kbase_device *kbdev,
153 struct devfreq_dev_profile *dp)
157 unsigned long freq = 0;
158 struct dev_pm_opp *opp;
161 count = dev_pm_opp_get_opp_count(kbdev->dev);
168 dp->freq_table = kmalloc_array(count, sizeof(dp->freq_table[0]),
174 for (i = 0; i < count; i++, freq++) {
175 opp = dev_pm_opp_find_freq_ceil(kbdev->dev, &freq);
179 dp->freq_table[i] = freq;
184 dev_warn(kbdev->dev, "Unable to enumerate all OPPs (%d!=%d\n",
192 static void kbase_devfreq_term_freq_table(struct kbase_device *kbdev)
194 struct devfreq_dev_profile *dp = kbdev->devfreq->profile;
196 kfree(dp->freq_table);
199 static void kbase_devfreq_exit(struct device *dev)
201 struct kbase_device *kbdev = dev_get_drvdata(dev);
203 kbase_devfreq_term_freq_table(kbdev);
206 int kbase_devfreq_init(struct kbase_device *kbdev)
208 struct devfreq_dev_profile *dp;
214 kbdev->current_freq = clk_get_rate(kbdev->clock);
216 dp = &kbdev->devfreq_profile;
218 dp->initial_freq = kbdev->current_freq;
219 /* .KP : set devfreq_dvfs_interval_in_ms */
221 dp->target = kbase_devfreq_target;
222 dp->get_dev_status = kbase_devfreq_status;
223 dp->get_cur_freq = kbase_devfreq_cur_freq;
224 dp->exit = kbase_devfreq_exit;
226 if (kbase_devfreq_init_freq_table(kbdev, dp))
229 kbdev->devfreq = devfreq_add_device(kbdev->dev, dp,
230 "simple_ondemand", NULL);
231 if (IS_ERR(kbdev->devfreq)) {
232 kbase_devfreq_term_freq_table(kbdev);
233 return PTR_ERR(kbdev->devfreq);
236 err = devfreq_register_opp_notifier(kbdev->dev, kbdev->devfreq);
239 "Failed to register OPP notifier (%d)\n", err);
240 goto opp_notifier_failed;
243 #ifdef CONFIG_DEVFREQ_THERMAL
244 err = kbase_power_model_simple_init(kbdev);
245 if (err && err != -ENODEV && err != -EPROBE_DEFER) {
247 "Failed to initialize simple power model (%d)\n",
251 if (err == -EPROBE_DEFER)
253 if (err != -ENODEV) {
254 kbdev->devfreq_cooling = of_devfreq_cooling_register_power(
257 &power_model_simple_ops);
258 if (IS_ERR_OR_NULL(kbdev->devfreq_cooling)) {
259 err = PTR_ERR(kbdev->devfreq_cooling);
261 "Failed to register cooling device (%d)\n",
268 I("success initing power_model_simple.");
273 #ifdef CONFIG_DEVFREQ_THERMAL
275 devfreq_unregister_opp_notifier(kbdev->dev, kbdev->devfreq);
276 #endif /* CONFIG_DEVFREQ_THERMAL */
278 if (devfreq_remove_device(kbdev->devfreq))
279 dev_err(kbdev->dev, "Failed to terminate devfreq (%d)\n", err);
281 kbdev->devfreq = NULL;
286 void kbase_devfreq_term(struct kbase_device *kbdev)
290 dev_dbg(kbdev->dev, "Term Mali devfreq\n");
292 #ifdef CONFIG_DEVFREQ_THERMAL
293 if (kbdev->devfreq_cooling)
294 devfreq_cooling_unregister(kbdev->devfreq_cooling);
297 devfreq_unregister_opp_notifier(kbdev->dev, kbdev->devfreq);
299 err = devfreq_remove_device(kbdev->devfreq);
301 dev_err(kbdev->dev, "Failed to terminate devfreq (%d)\n", err);
303 kbdev->devfreq = NULL;