ad05fe5bea8d225d10d7ed1476a79f4d32ce1943
[firefly-linux-kernel-4.4.55.git] / drivers / gpu / arm / midgard_for_linux / backend / gpu / mali_kbase_devfreq.c
1 /*
2  *
3  * (C) COPYRIGHT 2014-2016 ARM Limited. All rights reserved.
4  *
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
8  * of such GNU licence.
9  *
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.
13  *
14  */
15
16
17 #define ENABLE_DEBUG_LOG
18 #include "../../platform/rk/custom_log.h"
19
20
21 #include <mali_kbase.h>
22 #include <mali_kbase_config_defaults.h>
23 #include <backend/gpu/mali_kbase_pm_internal.h>
24 #ifdef CONFIG_DEVFREQ_THERMAL
25 #include <backend/gpu/mali_kbase_power_model_simple.h>
26 #endif
27
28 #include <linux/clk.h>
29 #include <linux/devfreq.h>
30 #ifdef CONFIG_DEVFREQ_THERMAL
31 #include <linux/devfreq_cooling.h>
32 #endif
33
34 #include <linux/version.h>
35 #if LINUX_VERSION_CODE >= KERNEL_VERSION(3, 13, 0)
36 #include <linux/pm_opp.h>
37 #else /* Linux >= 3.13 */
38 /* In 3.13 the OPP include header file, types, and functions were all
39  * renamed. Use the old filename for the include, and define the new names to
40  * the old, when an old kernel is detected.
41  */
42 #include <linux/opp.h>
43 #define dev_pm_opp opp
44 #define dev_pm_opp_get_voltage opp_get_voltage
45 #define dev_pm_opp_get_opp_count opp_get_opp_count
46 #define dev_pm_opp_find_freq_ceil opp_find_freq_ceil
47 #endif /* Linux >= 3.13 */
48
49
50 static int
51 kbase_devfreq_target(struct device *dev, unsigned long *target_freq, u32 flags)
52 {
53         struct kbase_device *kbdev = dev_get_drvdata(dev);
54         struct dev_pm_opp *opp;
55         unsigned long freq = 0;
56         unsigned long voltage;
57         int err;
58
59         freq = *target_freq;
60
61         rcu_read_lock();
62         opp = devfreq_recommended_opp(dev, &freq, flags);
63         voltage = dev_pm_opp_get_voltage(opp);
64         rcu_read_unlock();
65         if (IS_ERR_OR_NULL(opp)) {
66                 dev_err(dev, "Failed to get opp (%ld)\n", PTR_ERR(opp));
67                 return PTR_ERR(opp);
68         }
69
70         /*
71          * Only update if there is a change of frequency
72          */
73         if (kbdev->current_freq == freq) {
74                 *target_freq = freq;
75                 return 0;
76         }
77
78 #ifdef CONFIG_REGULATOR
79         if (kbdev->regulator && kbdev->current_voltage != voltage
80                         && kbdev->current_freq < freq) {
81                 err = regulator_set_voltage(kbdev->regulator, voltage, voltage);
82                 if (err) {
83                         dev_err(dev, "Failed to increase voltage (%d)\n", err);
84                         return err;
85                 }
86         }
87 #endif
88
89         err = clk_set_rate(kbdev->clock, freq);
90         if (err) {
91                 dev_err(dev, "Failed to set clock %lu (target %lu)\n",
92                                 freq, *target_freq);
93                 return err;
94         }
95
96 #ifdef CONFIG_REGULATOR
97         if (kbdev->regulator && kbdev->current_voltage != voltage
98                         && kbdev->current_freq > freq) {
99                 err = regulator_set_voltage(kbdev->regulator, voltage, voltage);
100                 if (err) {
101                         dev_err(dev, "Failed to decrease voltage (%d)\n", err);
102                         return err;
103                 }
104         }
105 #endif
106
107         *target_freq = freq;
108         kbdev->current_voltage = voltage;
109         kbdev->current_freq = freq;
110
111         kbase_pm_reset_dvfs_utilisation(kbdev);
112
113         return err;
114 }
115
116 static int
117 kbase_devfreq_cur_freq(struct device *dev, unsigned long *freq)
118 {
119         struct kbase_device *kbdev = dev_get_drvdata(dev);
120
121         *freq = kbdev->current_freq;
122
123         return 0;
124 }
125
126 static int
127 kbase_devfreq_status(struct device *dev, struct devfreq_dev_status *stat)
128 {
129         struct kbase_device *kbdev = dev_get_drvdata(dev);
130
131         stat->current_frequency = kbdev->current_freq;
132
133         kbase_pm_get_dvfs_utilisation(kbdev,
134                         &stat->total_time, &stat->busy_time);
135
136         stat->private_data = NULL;
137
138 #ifdef CONFIG_DEVFREQ_THERMAL
139 #if LINUX_VERSION_CODE < KERNEL_VERSION(4, 3, 0)
140         if (kbdev->devfreq_cooling)
141                 memcpy(&kbdev->devfreq_cooling->last_status, stat,
142                                 sizeof(*stat));
143 #endif
144 #endif
145
146         return 0;
147 }
148
149 static int kbase_devfreq_init_freq_table(struct kbase_device *kbdev,
150                 struct devfreq_dev_profile *dp)
151 {
152         int count;
153         int i = 0;
154         unsigned long freq = 0;
155         struct dev_pm_opp *opp;
156
157         rcu_read_lock();
158         count = dev_pm_opp_get_opp_count(kbdev->dev);
159         if (count < 0) {
160                 rcu_read_unlock();
161                 return count;
162         }
163         rcu_read_unlock();
164
165         dp->freq_table = kmalloc_array(count, sizeof(dp->freq_table[0]),
166                                 GFP_KERNEL);
167         if (!dp->freq_table)
168                 return -ENOMEM;
169
170         rcu_read_lock();
171         for (i = 0; i < count; i++, freq++) {
172                 opp = dev_pm_opp_find_freq_ceil(kbdev->dev, &freq);
173                 if (IS_ERR(opp))
174                         break;
175
176                 dp->freq_table[i] = freq;
177         }
178         rcu_read_unlock();
179
180         if (count != i)
181                 dev_warn(kbdev->dev, "Unable to enumerate all OPPs (%d!=%d\n",
182                                 count, i);
183
184         dp->max_state = i;
185
186         return 0;
187 }
188
189 static void kbase_devfreq_term_freq_table(struct kbase_device *kbdev)
190 {
191         struct devfreq_dev_profile *dp = kbdev->devfreq->profile;
192
193         kfree(dp->freq_table);
194 }
195
196 static void kbase_devfreq_exit(struct device *dev)
197 {
198         struct kbase_device *kbdev = dev_get_drvdata(dev);
199
200         kbase_devfreq_term_freq_table(kbdev);
201 }
202
203 int kbase_devfreq_init(struct kbase_device *kbdev)
204 {
205         struct devfreq_dev_profile *dp;
206         int err;
207
208         if (!kbdev->clock)
209                 return -ENODEV;
210
211         kbdev->current_freq = clk_get_rate(kbdev->clock);
212
213         dp = &kbdev->devfreq_profile;
214
215         dp->initial_freq = kbdev->current_freq;
216         /* .KP : set devfreq_dvfs_interval_in_ms */
217         dp->polling_ms = 20;
218         dp->target = kbase_devfreq_target;
219         dp->get_dev_status = kbase_devfreq_status;
220         dp->get_cur_freq = kbase_devfreq_cur_freq;
221         dp->exit = kbase_devfreq_exit;
222
223         if (kbase_devfreq_init_freq_table(kbdev, dp))
224                 return -EFAULT;
225
226         kbdev->devfreq = devfreq_add_device(kbdev->dev, dp,
227                                 "simple_ondemand", NULL);
228         if (IS_ERR(kbdev->devfreq)) {
229                 kbase_devfreq_term_freq_table(kbdev);
230                 return PTR_ERR(kbdev->devfreq);
231         }
232
233         err = devfreq_register_opp_notifier(kbdev->dev, kbdev->devfreq);
234         if (err) {
235                 dev_err(kbdev->dev,
236                         "Failed to register OPP notifier (%d)\n", err);
237                 goto opp_notifier_failed;
238         }
239
240 #ifdef CONFIG_DEVFREQ_THERMAL
241         err = kbase_power_model_simple_init(kbdev);
242         if (err && err != -ENODEV && err != -EPROBE_DEFER) {
243                 dev_err(kbdev->dev,
244                         "Failed to initialize simple power model (%d)\n",
245                         err);
246                 goto cooling_failed;
247         }
248         if (err == -EPROBE_DEFER)
249                 goto cooling_failed;
250         if (err != -ENODEV) {
251                 kbdev->devfreq_cooling = of_devfreq_cooling_register_power(
252                                 kbdev->dev->of_node,
253                                 kbdev->devfreq,
254                                 &power_model_simple_ops);
255                 if (IS_ERR_OR_NULL(kbdev->devfreq_cooling)) {
256                         err = PTR_ERR(kbdev->devfreq_cooling);
257                         dev_err(kbdev->dev,
258                                 "Failed to register cooling device (%d)\n",
259                                 err);
260                         goto cooling_failed;
261                 }
262         } else {
263                 err = 0;
264         }
265         I("success initing power_model_simple.");
266 #endif
267
268         return 0;
269
270 #ifdef CONFIG_DEVFREQ_THERMAL
271 cooling_failed:
272         devfreq_unregister_opp_notifier(kbdev->dev, kbdev->devfreq);
273 #endif /* CONFIG_DEVFREQ_THERMAL */
274 opp_notifier_failed:
275         if (devfreq_remove_device(kbdev->devfreq))
276                 dev_err(kbdev->dev, "Failed to terminate devfreq (%d)\n", err);
277         else
278                 kbdev->devfreq = NULL;
279
280         return err;
281 }
282
283 void kbase_devfreq_term(struct kbase_device *kbdev)
284 {
285         int err;
286
287         dev_dbg(kbdev->dev, "Term Mali devfreq\n");
288
289 #ifdef CONFIG_DEVFREQ_THERMAL
290         if (kbdev->devfreq_cooling)
291                 devfreq_cooling_unregister(kbdev->devfreq_cooling);
292 #endif
293
294         devfreq_unregister_opp_notifier(kbdev->dev, kbdev->devfreq);
295
296         err = devfreq_remove_device(kbdev->devfreq);
297         if (err)
298                 dev_err(kbdev->dev, "Failed to terminate devfreq (%d)\n", err);
299         else
300                 kbdev->devfreq = NULL;
301 }