blob: ac4ed253febb3df2f8b9b1b4ea8c4c9913807431 [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
39static DEFINE_SPINLOCK(msm_pm_stats_lock);
40static DEFINE_PER_CPU_SHARED_ALIGNED(
41 struct msm_pm_cpu_time_stats, msm_pm_stats);
42
43/*
44 * Add the given time data to the statistics collection.
45 */
46void msm_pm_add_stat(enum msm_pm_time_stats_id id, int64_t t)
47{
48 unsigned long flags;
49 struct msm_pm_time_stats *stats;
50 int64_t bt;
51 int i;
52
53 spin_lock_irqsave(&msm_pm_stats_lock, flags);
54 stats = __get_cpu_var(msm_pm_stats).stats;
55
56 if (!stats[id].enabled)
57 goto add_bail;
58
59 stats[id].total_time += t;
60 stats[id].count++;
61
62 bt = t;
63 do_div(bt, stats[id].first_bucket_time);
64
65 if (bt < 1ULL << (CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT *
66 (CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1)))
67 i = DIV_ROUND_UP(fls((uint32_t)bt),
68 CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT);
69 else
70 i = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
71
72 if (i >= CONFIG_MSM_IDLE_STATS_BUCKET_COUNT)
73 i = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
74
75 stats[id].bucket[i]++;
76
77 if (t < stats[id].min_time[i] || !stats[id].max_time[i])
78 stats[id].min_time[i] = t;
79 if (t > stats[id].max_time[i])
80 stats[id].max_time[i] = t;
81
82add_bail:
83 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
84}
85
86/*
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053087 * Write out the power management statistics.
88 */
Anji Jonnala6137b8f2013-07-16 11:39:15 +053089
90static int msm_pm_stats_show(struct seq_file *m, void *v)
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053091{
Anji Jonnala6137b8f2013-07-16 11:39:15 +053092 int cpu;
93 int bucket_count = CONFIG_MSM_IDLE_STATS_BUCKET_COUNT - 1;
94 int bucket_shift = CONFIG_MSM_IDLE_STATS_BUCKET_SHIFT;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053095
Anji Jonnala6137b8f2013-07-16 11:39:15 +053096 for_each_possible_cpu(cpu) {
Praveen Chidambaram3895bde2012-05-14 19:42:40 +053097 unsigned long flags;
98 struct msm_pm_time_stats *stats;
Anji Jonnala6137b8f2013-07-16 11:39:15 +053099 int i, id;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530100 int64_t bucket_time;
101 int64_t s;
102 uint32_t ns;
103
104 spin_lock_irqsave(&msm_pm_stats_lock, flags);
105 stats = per_cpu(msm_pm_stats, cpu).stats;
106
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530107 for (id = 0; id < MSM_PM_STAT_COUNT; id++) {
108 /* Skip the disabled ones */
109 if (!stats[id].enabled)
110 continue;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530111
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530112 s = stats[id].total_time;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530113 ns = do_div(s, NSEC_PER_SEC);
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530114 seq_printf(m,
115 "[cpu %u] %s:\n"
116 " count: %7d\n"
117 " total_time: %lld.%09u\n",
118 cpu, stats[id].name,
119 stats[id].count,
120 s, ns);
121
122 bucket_time = stats[id].first_bucket_time;
123 for (i = 0; i < bucket_count; i++) {
124 s = bucket_time;
125 ns = do_div(s, NSEC_PER_SEC);
126 seq_printf(m,
127 " <%6lld.%09u: %7d (%lld-%lld)\n",
128 s, ns, stats[id].bucket[i],
129 stats[id].min_time[i],
130 stats[id].max_time[i]);
131
132 bucket_time <<= bucket_shift;
133 }
134
135 seq_printf(m, " >=%6lld.%09u: %7d (%lld-%lld)\n",
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530136 s, ns, stats[id].bucket[i],
137 stats[id].min_time[i],
138 stats[id].max_time[i]);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530139 }
140
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530141 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
142 }
143
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530144 return 0;
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530145}
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530146
147#define MSM_PM_STATS_RESET "reset"
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530148/*
149 * Reset the power management statistics values.
150 */
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530151static ssize_t msm_pm_write_proc(struct file *file, const char __user *buffer,
152 size_t count, loff_t *off)
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530153{
154 char buf[sizeof(MSM_PM_STATS_RESET)];
155 int ret;
156 unsigned long flags;
157 unsigned int cpu;
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600158 size_t len = strnlen(MSM_PM_STATS_RESET, sizeof(MSM_PM_STATS_RESET));
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530159
160 if (count < sizeof(MSM_PM_STATS_RESET)) {
161 ret = -EINVAL;
162 goto write_proc_failed;
163 }
164
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600165 if (copy_from_user(buf, buffer, len)) {
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530166 ret = -EFAULT;
167 goto write_proc_failed;
168 }
169
Mahesh Sivasubramanianad4a5752012-06-04 17:45:29 -0600170 if (strncmp(buf, MSM_PM_STATS_RESET, len)) {
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530171 ret = -EINVAL;
172 goto write_proc_failed;
173 }
174
175 spin_lock_irqsave(&msm_pm_stats_lock, flags);
176 for_each_possible_cpu(cpu) {
177 struct msm_pm_time_stats *stats;
178 int i;
179
180 stats = per_cpu(msm_pm_stats, cpu).stats;
181 for (i = 0; i < MSM_PM_STAT_COUNT; i++) {
182 memset(stats[i].bucket,
183 0, sizeof(stats[i].bucket));
184 memset(stats[i].min_time,
185 0, sizeof(stats[i].min_time));
186 memset(stats[i].max_time,
187 0, sizeof(stats[i].max_time));
188 stats[i].count = 0;
189 stats[i].total_time = 0;
190 }
191 }
192
193 spin_unlock_irqrestore(&msm_pm_stats_lock, flags);
194 return count;
195
196write_proc_failed:
197 return ret;
198}
199#undef MSM_PM_STATS_RESET
200
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530201static int msm_pm_stats_open(struct inode *inode, struct file *file)
202{
203 return single_open(file, msm_pm_stats_show, NULL);
204}
205
206static const struct file_operations msm_pm_stats_fops = {
207 .open = msm_pm_stats_open,
208 .read = seq_read,
209 .llseek = seq_lseek,
210 .release = single_release,
211 .write = msm_pm_write_proc,
212};
213
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530214void msm_pm_add_stats(enum msm_pm_time_stats_id *enable_stats, int size)
215{
216 unsigned int cpu;
217 struct proc_dir_entry *d_entry;
218 int i = 0;
219
220 for_each_possible_cpu(cpu) {
221 struct msm_pm_time_stats *stats =
222 per_cpu(msm_pm_stats, cpu).stats;
223
224 stats[MSM_PM_STAT_REQUESTED_IDLE].name = "idle-request";
225 stats[MSM_PM_STAT_REQUESTED_IDLE].first_bucket_time =
226 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
227
228 stats[MSM_PM_STAT_IDLE_SPIN].name = "idle-spin";
229 stats[MSM_PM_STAT_IDLE_SPIN].first_bucket_time =
230 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
231
232 stats[MSM_PM_STAT_IDLE_WFI].name = "idle-wfi";
233 stats[MSM_PM_STAT_IDLE_WFI].first_bucket_time =
234 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
235
236 stats[MSM_PM_STAT_RETENTION].name = "retention";
237 stats[MSM_PM_STAT_RETENTION].first_bucket_time =
238 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
239
240 stats[MSM_PM_STAT_IDLE_STANDALONE_POWER_COLLAPSE].name =
241 "idle-standalone-power-collapse";
242 stats[MSM_PM_STAT_IDLE_STANDALONE_POWER_COLLAPSE].
243 first_bucket_time = CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
244
245 stats[MSM_PM_STAT_IDLE_FAILED_STANDALONE_POWER_COLLAPSE].name =
246 "idle-failed-standalone-power-collapse";
247 stats[MSM_PM_STAT_IDLE_FAILED_STANDALONE_POWER_COLLAPSE].
248 first_bucket_time =
249 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
250
251 stats[MSM_PM_STAT_IDLE_POWER_COLLAPSE].name =
252 "idle-power-collapse";
253 stats[MSM_PM_STAT_IDLE_POWER_COLLAPSE].first_bucket_time =
254 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
255
256 stats[MSM_PM_STAT_IDLE_FAILED_POWER_COLLAPSE].name =
257 "idle-failed-power-collapse";
258 stats[MSM_PM_STAT_IDLE_FAILED_POWER_COLLAPSE].
259 first_bucket_time =
260 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
261
262 stats[MSM_PM_STAT_SUSPEND].name = "suspend";
263 stats[MSM_PM_STAT_SUSPEND].first_bucket_time =
264 CONFIG_MSM_SUSPEND_STATS_FIRST_BUCKET;
265
266 stats[MSM_PM_STAT_FAILED_SUSPEND].name = "failed-suspend";
267 stats[MSM_PM_STAT_FAILED_SUSPEND].first_bucket_time =
268 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
269
270 stats[MSM_PM_STAT_NOT_IDLE].name = "not-idle";
271 stats[MSM_PM_STAT_NOT_IDLE].first_bucket_time =
272 CONFIG_MSM_IDLE_STATS_FIRST_BUCKET;
273
274 for (i = 0; i < size; i++)
275 stats[enable_stats[i]].enabled = true;
276
277 }
278
Anji Jonnala6137b8f2013-07-16 11:39:15 +0530279 d_entry = proc_create_data("msm_pm_stats", S_IRUGO | S_IWUSR | S_IWGRP,
280 NULL, &msm_pm_stats_fops, NULL);
Praveen Chidambaram3895bde2012-05-14 19:42:40 +0530281}