blob: 5614ddd9edca8c46e65ae6a655121fd239a4333f [file] [log] [blame]
Anji Jonnala6137b8f2013-07-16 11:39:15 +05301/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Praveen Chidambaram3895bde2012-05-14 19:42:40 +05302 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
13
14#include <linux/module.h>
15#include <linux/kernel.h>
16#include <linux/init.h>
17#include <linux/spinlock.h>
18#include <linux/uaccess.h>
19#include <linux/proc_fs.h>
Anji Jonnala6137b8f2013-07-16 11:39:15 +053020#include <linux/seq_file.h>
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053021
22#include "pm.h"
23
24struct msm_pm_time_stats {
25 const char *name;
26 int64_t first_bucket_time;
27 int bucket[CONFIG_MSM_IDLE_STATS_BUCKET_COUNT];
28 int64_t min_time[CONFIG_MSM_IDLE_STATS_BUCKET_COUNT];
29 int64_t max_time[CONFIG_MSM_IDLE_STATS_BUCKET_COUNT];
30 int count;
31 int64_t total_time;
32 bool enabled;
33};
34
35struct msm_pm_cpu_time_stats {
36 struct msm_pm_time_stats stats[MSM_PM_STAT_COUNT];
37};
38
Anil kumar mamidala01fea962013-12-03 17:16:24 +053039static struct msm_pm_time_stats suspend_stats;
40
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053041static DEFINE_SPINLOCK(msm_pm_stats_lock);
42static DEFINE_PER_CPU_SHARED_ALIGNED(
43 struct msm_pm_cpu_time_stats, msm_pm_stats);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053044/*
Anil kumar mamidala01fea962013-12-03 17:16:24 +053045 * Function to update stats
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053046 */
Anil kumar mamidala01fea962013-12-03 17:16:24 +053047static void update_stats(struct msm_pm_time_stats *stats, int64_t t)
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053048{
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053049 int64_t bt;
50 int i;
51
Anil kumar mamidala01fea962013-12-03 17:16:24 +053052 if (!stats)
53 return;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053054
Anil kumar mamidala01fea962013-12-03 17:16:24 +053055 stats->total_time += t;
56 stats->count++;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053057
58 bt = t;
Anil kumar mamidala01fea962013-12-03 17:16:24 +053059 do_div(bt, stats->first_bucket_time);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053060
61 if (bt < 1ULL << (CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT *
Anil kumar mamidala01fea962013-12-03 17:16:24 +053062 (CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1)))
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053063 i = DIV_ROUND_UP(fls((uint32_t)bt),
Anil kumar mamidala01fea962013-12-03 17:16:24 +053064 CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053065 else
66 i = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
67
68 if (i >= CONFIG_MSM_IDLE_STATS_BUCKET_COUNT)
69 i = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
70
Anil kumar mamidala01fea962013-12-03 17:16:24 +053071 stats->bucket[i]++;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053072
Anil kumar mamidala01fea962013-12-03 17:16:24 +053073 if (t < stats->min_time[i] || !stats->max_time[i])
74 stats->min_time[i] = t;
75 if (t > stats->max_time[i])
76 stats->max_time[i] = t;
77}
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053078
Anil kumar mamidala01fea962013-12-03 17:16:24 +053079/*
80 * Add the given time data to the statistics collection.
81 */
82void msm_pm_add_stat(enum msm_pm_time_stats_id id, int64_t t)
83{
84 struct msm_pm_time_stats *stats;
85 unsigned long flags;
86
87 spin_lock_irqsave(&msm_pm_stats_lock, flags);
88 if (id == MSM_PM_STAT_SUSPEND) {
89 stats = &suspend_stats;
90 } else {
91 stats = __get_cpu_var(msm_pm_stats).stats;
92 if (!stats[id].enabled)
93 goto add_bail;
94 stats = &stats[id];
95 }
96 update_stats(stats, t);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053097add_bail:
98 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
99}
100
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530101static void stats_show(struct seq_file *m,
102 struct msm_pm_time_stats *stats,
103 int cpu, int suspend)
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530104{
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530105 int64_t bucket_time;
106 int64_t s;
107 uint32_t ns;
108 int i;
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530109 int bucket_count = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
110 int bucket_shift = CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530111
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530112 if (!stats || !m)
113 return;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530114
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530115 s = stats->total_time;
116 ns = do_div(s, NSEC_PER_SEC);
117 if (suspend)
118 seq_printf(m,
119 "%s:\n"
120 " count: %7d\n"
121 " total_time: %lld.%09u\n",
122 stats->name,
123 stats->count,
124 s, ns);
125 else
126 seq_printf(m,
127 "[cpu %u] %s:\n"
128 " count: %7d\n"
129 " total_time: %lld.%09u\n",
130 cpu, stats->name,
131 stats->count,
132 s, ns);
133
134 bucket_time = stats->first_bucket_time;
135 for (i = 0; i < bucket_count; i++) {
136 s = bucket_time;
137 ns = do_div(s, NSEC_PER_SEC);
138 seq_printf(m,
139 " <%6lld.%09u: %7d (%lld-%lld)\n",
140 s, ns, stats->bucket[i],
141 stats->min_time[i],
142 stats->max_time[i]);
143 bucket_time <<= bucket_shift;
144 }
145
146 seq_printf(m, " >=%6lld.%09u: %7d (%lld-%lld)\n",
147 s, ns, stats->bucket[i],
148 stats->min_time[i],
149 stats->max_time[i]);
150}
151/*
152 * Write out the power management statistics.
153 */
154static int msm_pm_stats_show(struct seq_file *m, void *v)
155{
156 int cpu;
157 int id;
158 unsigned long flags;
159
160 spin_lock_irqsave(&msm_pm_stats_lock, flags);
161
162 for_each_possible_cpu(cpu) {
163 struct msm_pm_time_stats *stats;
164
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530165 stats = per_cpu(msm_pm_stats, cpu).stats;
166
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530167 for (id = 0; id < MSM_PM_STAT_COUNT; id++) {
168 /* Skip the disabled ones */
169 if (!stats[id].enabled)
170 continue;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530171
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530172 if (id == MSM_PM_STAT_SUSPEND)
173 continue;
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530174
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530175 stats_show(m, &stats[id], cpu, false);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530176 }
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530177 }
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530178 stats_show(m, &suspend_stats, cpu, true);
179 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530180 return 0;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530181}
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530182
183#define MSM_PM_STATS_RESET "reset"
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530184/*
185 * Reset the power management statistics values.
186 */
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530187static ssize_t msm_pm_write_proc(struct file *file, const char __user *buffer,
188 size_t count, loff_t *off)
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530189{
190 char buf[sizeof(MSM_PM_STATS_RESET)];
191 int ret;
192 unsigned long flags;
193 unsigned int cpu;
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600194 size_t len = strnlen(MSM_PM_STATS_RESET, sizeof(MSM_PM_STATS_RESET));
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530195
196 if (count < sizeof(MSM_PM_STATS_RESET)) {
197 ret = -EINVAL;
198 goto write_proc_failed;
199 }
200
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600201 if (copy_from_user(buf, buffer, len)) {
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530202 ret = -EFAULT;
203 goto write_proc_failed;
204 }
205
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600206 if (strncmp(buf, MSM_PM_STATS_RESET, len)) {
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530207 ret = -EINVAL;
208 goto write_proc_failed;
209 }
210
211 spin_lock_irqsave(&msm_pm_stats_lock, flags);
212 for_each_possible_cpu(cpu) {
213 struct msm_pm_time_stats *stats;
214 int i;
215
216 stats = per_cpu(msm_pm_stats, cpu).stats;
217 for (i = 0; i < MSM_PM_STAT_COUNT; i++) {
218 memset(stats[i].bucket,
219 0, sizeof(stats[i].bucket));
220 memset(stats[i].min_time,
221 0, sizeof(stats[i].min_time));
222 memset(stats[i].max_time,
223 0, sizeof(stats[i].max_time));
224 stats[i].count = 0;
225 stats[i].total_time = 0;
226 }
227 }
Anil kumar mamidala99e18cb2014-02-06 19:26:21 +0530228 memset(suspend_stats.bucket,
229 0, sizeof(suspend_stats.bucket));
230 memset(suspend_stats.min_time,
231 0, sizeof(suspend_stats.min_time));
232 memset(suspend_stats.max_time,
233 0, sizeof(suspend_stats.max_time));
234 suspend_stats.count = 0;
235 suspend_stats.total_time = 0;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530236
237 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
238 return count;
239
240write_proc_failed:
241 return ret;
242}
243#undef MSM_PM_STATS_RESET
244
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530245static int msm_pm_stats_open(struct inode *inode, struct file *file)
246{
247 return single_open(file, msm_pm_stats_show, NULL);
248}
249
250static const struct file_operations msm_pm_stats_fops = {
251 .open = msm_pm_stats_open,
252 .read = seq_read,
253 .llseek = seq_lseek,
254 .release = single_release,
255 .write = msm_pm_write_proc,
256};
257
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530258void msm_pm_add_stats(enum msm_pm_time_stats_id *enable_stats, int size)
259{
260 unsigned int cpu;
261 struct proc_dir_entry *d_entry;
262 int i = 0;
263
264 for_each_possible_cpu(cpu) {
265 struct msm_pm_time_stats *stats =
266 per_cpu(msm_pm_stats, cpu).stats;
267
268 stats[MSM_PM_STAT_REQUESTED_IDLE].name = "idle-request";
269 stats[MSM_PM_STAT_REQUESTED_IDLE].first_bucket_time =
270 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
271
272 stats[MSM_PM_STAT_IDLE_SPIN].name = "idle-spin";
273 stats[MSM_PM_STAT_IDLE_SPIN].first_bucket_time =
274 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
275
276 stats[MSM_PM_STAT_IDLE_WFI].name = "idle-wfi";
277 stats[MSM_PM_STAT_IDLE_WFI].first_bucket_time =
278 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
279
280 stats[MSM_PM_STAT_RETENTION].name = "retention";
281 stats[MSM_PM_STAT_RETENTION].first_bucket_time =
282 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
283
284 stats[MSM_PM_STAT_IDLE_STANDALONE_POWER_COLLAPSE].name =
285 "idle-standalone-power-collapse";
286 stats[MSM_PM_STAT_IDLE_STANDALONE_POWER_COLLAPSE].
287 first_bucket_time = CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
288
289 stats[MSM_PM_STAT_IDLE_FAILED_STANDALONE_POWER_COLLAPSE].name =
290 "idle-failed-standalone-power-collapse";
291 stats[MSM_PM_STAT_IDLE_FAILED_STANDALONE_POWER_COLLAPSE].
292 first_bucket_time =
293 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
294
295 stats[MSM_PM_STAT_IDLE_POWER_COLLAPSE].name =
296 "idle-power-collapse";
297 stats[MSM_PM_STAT_IDLE_POWER_COLLAPSE].first_bucket_time =
298 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
299
300 stats[MSM_PM_STAT_IDLE_FAILED_POWER_COLLAPSE].name =
301 "idle-failed-power-collapse";
302 stats[MSM_PM_STAT_IDLE_FAILED_POWER_COLLAPSE].
303 first_bucket_time =
304 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
305
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530306 stats[MSM_PM_STAT_NOT_IDLE].name = "not-idle";
307 stats[MSM_PM_STAT_NOT_IDLE].first_bucket_time =
308 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
309
310 for (i = 0; i < size; i++)
311 stats[enable_stats[i]].enabled = true;
312
313 }
Anil kumar mamidala01fea962013-12-03 17:16:24 +0530314 suspend_stats.name = "system_suspend";
315 suspend_stats.first_bucket_time =
316 CONFIG_MSM_SUSPEND_STATS_FIRST_BUCKET;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530317
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530318 d_entry = proc_create_data("msm_pm_stats", S_IRUGO | S_IWUSR | S_IWGRP,
319 NULL, &msm_pm_stats_fops, NULL);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530320}