cpufreq_stats.c 10.6 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4
/*
 *  drivers/cpufreq/cpufreq_stats.c
 *
 *  Copyright (C) 2003-2004 Venkatesh Pallipadi <venkatesh.pallipadi@intel.com>.
5
 *  (C) 2004 Zou Nan hai <nanhai.zou@intel.com>.
L
Linus Torvalds 已提交
6 7 8 9 10 11 12
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */

#include <linux/kernel.h>
13
#include <linux/slab.h>
L
Linus Torvalds 已提交
14 15 16
#include <linux/cpu.h>
#include <linux/sysfs.h>
#include <linux/cpufreq.h>
17
#include <linux/module.h>
L
Linus Torvalds 已提交
18 19 20 21
#include <linux/jiffies.h>
#include <linux/percpu.h>
#include <linux/kobject.h>
#include <linux/spinlock.h>
22
#include <linux/notifier.h>
23
#include <asm/cputime.h>
L
Linus Torvalds 已提交
24 25 26 27 28 29

static spinlock_t cpufreq_stats_lock;

struct cpufreq_stats {
	unsigned int cpu;
	unsigned int total_trans;
30
	unsigned long long last_time;
L
Linus Torvalds 已提交
31 32 33
	unsigned int max_state;
	unsigned int state_num;
	unsigned int last_index;
34
	u64 *time_in_state;
L
Linus Torvalds 已提交
35 36 37 38 39 40
	unsigned int *freq_table;
#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
	unsigned int *trans_table;
#endif
};

41
static DEFINE_PER_CPU(struct cpufreq_stats *, cpufreq_stats_table);
L
Linus Torvalds 已提交
42 43 44 45 46 47

struct cpufreq_stats_attribute {
	struct attribute attr;
	ssize_t(*show) (struct cpufreq_stats *, char *);
};

48
static int cpufreq_stats_update(unsigned int cpu)
L
Linus Torvalds 已提交
49 50
{
	struct cpufreq_stats *stat;
51 52 53
	unsigned long long cur_time;

	cur_time = get_jiffies_64();
L
Linus Torvalds 已提交
54
	spin_lock(&cpufreq_stats_lock);
55
	stat = per_cpu(cpufreq_stats_table, cpu);
L
Linus Torvalds 已提交
56
	if (stat->time_in_state)
57 58
		stat->time_in_state[stat->last_index] +=
			cur_time - stat->last_time;
59
	stat->last_time = cur_time;
L
Linus Torvalds 已提交
60 61 62 63
	spin_unlock(&cpufreq_stats_lock);
	return 0;
}

64
static ssize_t show_total_trans(struct cpufreq_policy *policy, char *buf)
L
Linus Torvalds 已提交
65
{
66
	struct cpufreq_stats *stat = per_cpu(cpufreq_stats_table, policy->cpu);
67
	if (!stat)
L
Linus Torvalds 已提交
68 69
		return 0;
	return sprintf(buf, "%d\n",
70
			per_cpu(cpufreq_stats_table, stat->cpu)->total_trans);
L
Linus Torvalds 已提交
71 72
}

73
static ssize_t show_time_in_state(struct cpufreq_policy *policy, char *buf)
L
Linus Torvalds 已提交
74 75 76
{
	ssize_t len = 0;
	int i;
77
	struct cpufreq_stats *stat = per_cpu(cpufreq_stats_table, policy->cpu);
78
	if (!stat)
L
Linus Torvalds 已提交
79 80 81
		return 0;
	cpufreq_stats_update(stat->cpu);
	for (i = 0; i < stat->state_num; i++) {
82
		len += sprintf(buf + len, "%u %llu\n", stat->freq_table[i],
83 84
			(unsigned long long)
			cputime64_to_clock_t(stat->time_in_state[i]));
L
Linus Torvalds 已提交
85 86 87 88 89
	}
	return len;
}

#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
90
static ssize_t show_trans_table(struct cpufreq_policy *policy, char *buf)
L
Linus Torvalds 已提交
91 92 93 94
{
	ssize_t len = 0;
	int i, j;

95
	struct cpufreq_stats *stat = per_cpu(cpufreq_stats_table, policy->cpu);
96
	if (!stat)
L
Linus Torvalds 已提交
97 98
		return 0;
	cpufreq_stats_update(stat->cpu);
99 100 101 102 103 104 105 106 107
	len += snprintf(buf + len, PAGE_SIZE - len, "   From  :    To\n");
	len += snprintf(buf + len, PAGE_SIZE - len, "         : ");
	for (i = 0; i < stat->state_num; i++) {
		if (len >= PAGE_SIZE)
			break;
		len += snprintf(buf + len, PAGE_SIZE - len, "%9u ",
				stat->freq_table[i]);
	}
	if (len >= PAGE_SIZE)
108
		return PAGE_SIZE;
109 110 111

	len += snprintf(buf + len, PAGE_SIZE - len, "\n");

L
Linus Torvalds 已提交
112 113 114
	for (i = 0; i < stat->state_num; i++) {
		if (len >= PAGE_SIZE)
			break;
115 116

		len += snprintf(buf + len, PAGE_SIZE - len, "%9u: ",
L
Linus Torvalds 已提交
117 118
				stat->freq_table[i]);

119
		for (j = 0; j < stat->state_num; j++) {
L
Linus Torvalds 已提交
120 121
			if (len >= PAGE_SIZE)
				break;
122
			len += snprintf(buf + len, PAGE_SIZE - len, "%9u ",
L
Linus Torvalds 已提交
123 124
					stat->trans_table[i*stat->max_state+j]);
		}
125 126
		if (len >= PAGE_SIZE)
			break;
L
Linus Torvalds 已提交
127 128
		len += snprintf(buf + len, PAGE_SIZE - len, "\n");
	}
129 130
	if (len >= PAGE_SIZE)
		return PAGE_SIZE;
L
Linus Torvalds 已提交
131 132
	return len;
}
133
cpufreq_freq_attr_ro(trans_table);
L
Linus Torvalds 已提交
134 135
#endif

136 137
cpufreq_freq_attr_ro(total_trans);
cpufreq_freq_attr_ro(time_in_state);
L
Linus Torvalds 已提交
138 139

static struct attribute *default_attrs[] = {
140 141
	&total_trans.attr,
	&time_in_state.attr,
L
Linus Torvalds 已提交
142
#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
143
	&trans_table.attr,
L
Linus Torvalds 已提交
144 145 146 147 148 149 150 151
#endif
	NULL
};
static struct attribute_group stats_attr_group = {
	.attrs = default_attrs,
	.name = "stats"
};

152
static int freq_table_get_index(struct cpufreq_stats *stat, unsigned int freq)
L
Linus Torvalds 已提交
153 154 155 156 157 158 159 160
{
	int index;
	for (index = 0; index < stat->max_state; index++)
		if (stat->freq_table[index] == freq)
			return index;
	return -1;
}

161 162 163
/* should be called late in the CPU removal sequence so that the stats
 * memory is still available in case someone tries to use it.
 */
164
static void cpufreq_stats_free_table(unsigned int cpu)
L
Linus Torvalds 已提交
165
{
166
	struct cpufreq_stats *stat = per_cpu(cpufreq_stats_table, cpu);
167

L
Linus Torvalds 已提交
168
	if (stat) {
169
		pr_debug("%s: Free stat table\n", __func__);
L
Linus Torvalds 已提交
170 171
		kfree(stat->time_in_state);
		kfree(stat);
172
		per_cpu(cpufreq_stats_table, cpu) = NULL;
L
Linus Torvalds 已提交
173
	}
174 175 176 177 178 179 180 181
}

/* must be called early in the CPU removal sequence (before
 * cpufreq_remove_dev) so that policy is still valid.
 */
static void cpufreq_stats_free_sysfs(unsigned int cpu)
{
	struct cpufreq_policy *policy = cpufreq_cpu_get(cpu);
182

183
	if (!policy)
184 185
		return;

186 187 188 189
	if (!cpufreq_frequency_get_table(cpu))
		goto put_ref;

	if (!policy_is_shared(policy)) {
190
		pr_debug("%s: Free sysfs stat\n", __func__);
191
		sysfs_remove_group(&policy->kobj, &stats_attr_group);
192
	}
193 194 195

put_ref:
	cpufreq_cpu_put(policy);
L
Linus Torvalds 已提交
196 197
}

198
static int cpufreq_stats_create_table(struct cpufreq_policy *policy,
L
Linus Torvalds 已提交
199 200 201 202 203 204 205
		struct cpufreq_frequency_table *table)
{
	unsigned int i, j, count = 0, ret = 0;
	struct cpufreq_stats *stat;
	struct cpufreq_policy *data;
	unsigned int alloc_size;
	unsigned int cpu = policy->cpu;
206
	if (per_cpu(cpufreq_stats_table, cpu))
L
Linus Torvalds 已提交
207
		return -EBUSY;
208 209
	stat = kzalloc(sizeof(struct cpufreq_stats), GFP_KERNEL);
	if ((stat) == NULL)
L
Linus Torvalds 已提交
210 211 212
		return -ENOMEM;

	data = cpufreq_cpu_get(cpu);
213 214 215 216 217
	if (data == NULL) {
		ret = -EINVAL;
		goto error_get_fail;
	}

218 219
	ret = sysfs_create_group(&data->kobj, &stats_attr_group);
	if (ret)
L
Linus Torvalds 已提交
220 221 222
		goto error_out;

	stat->cpu = cpu;
223
	per_cpu(cpufreq_stats_table, cpu) = stat;
L
Linus Torvalds 已提交
224

225
	for (i = 0; table[i].frequency != CPUFREQ_TABLE_END; i++) {
L
Linus Torvalds 已提交
226 227 228 229 230 231
		unsigned int freq = table[i].frequency;
		if (freq == CPUFREQ_ENTRY_INVALID)
			continue;
		count++;
	}

232
	alloc_size = count * sizeof(int) + count * sizeof(u64);
L
Linus Torvalds 已提交
233 234 235 236 237

#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
	alloc_size += count * count * sizeof(int);
#endif
	stat->max_state = count;
238
	stat->time_in_state = kzalloc(alloc_size, GFP_KERNEL);
L
Linus Torvalds 已提交
239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
	if (!stat->time_in_state) {
		ret = -ENOMEM;
		goto error_out;
	}
	stat->freq_table = (unsigned int *)(stat->time_in_state + count);

#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
	stat->trans_table = stat->freq_table + count;
#endif
	j = 0;
	for (i = 0; table[i].frequency != CPUFREQ_TABLE_END; i++) {
		unsigned int freq = table[i].frequency;
		if (freq == CPUFREQ_ENTRY_INVALID)
			continue;
		if (freq_table_get_index(stat, freq) == -1)
			stat->freq_table[j++] = freq;
	}
	stat->state_num = j;
	spin_lock(&cpufreq_stats_lock);
258
	stat->last_time = get_jiffies_64();
L
Linus Torvalds 已提交
259 260 261 262 263 264
	stat->last_index = freq_table_get_index(stat, policy->cur);
	spin_unlock(&cpufreq_stats_lock);
	cpufreq_cpu_put(data);
	return 0;
error_out:
	cpufreq_cpu_put(data);
265
error_get_fail:
L
Linus Torvalds 已提交
266
	kfree(stat);
267
	per_cpu(cpufreq_stats_table, cpu) = NULL;
L
Linus Torvalds 已提交
268 269 270
	return ret;
}

271 272 273 274 275 276 277 278 279 280 281 282 283
static void cpufreq_stats_update_policy_cpu(struct cpufreq_policy *policy)
{
	struct cpufreq_stats *stat = per_cpu(cpufreq_stats_table,
			policy->last_cpu);

	pr_debug("Updating stats_table for new_cpu %u from last_cpu %u\n",
			policy->cpu, policy->last_cpu);
	per_cpu(cpufreq_stats_table, policy->cpu) = per_cpu(cpufreq_stats_table,
			policy->last_cpu);
	per_cpu(cpufreq_stats_table, policy->last_cpu) = NULL;
	stat->cpu = policy->cpu;
}

284 285
static int cpufreq_stat_notifier_policy(struct notifier_block *nb,
		unsigned long val, void *data)
L
Linus Torvalds 已提交
286 287 288 289 290
{
	int ret;
	struct cpufreq_policy *policy = data;
	struct cpufreq_frequency_table *table;
	unsigned int cpu = policy->cpu;
291 292 293 294 295 296

	if (val == CPUFREQ_UPDATE_POLICY_CPU) {
		cpufreq_stats_update_policy_cpu(policy);
		return 0;
	}

L
Linus Torvalds 已提交
297 298 299 300 301
	if (val != CPUFREQ_NOTIFY)
		return 0;
	table = cpufreq_frequency_get_table(cpu);
	if (!table)
		return 0;
302 303
	ret = cpufreq_stats_create_table(policy, table);
	if (ret)
L
Linus Torvalds 已提交
304 305 306 307
		return ret;
	return 0;
}

308 309
static int cpufreq_stat_notifier_trans(struct notifier_block *nb,
		unsigned long val, void *data)
L
Linus Torvalds 已提交
310 311 312 313 314 315 316 317
{
	struct cpufreq_freqs *freq = data;
	struct cpufreq_stats *stat;
	int old_index, new_index;

	if (val != CPUFREQ_POSTCHANGE)
		return 0;

318
	stat = per_cpu(cpufreq_stats_table, freq->cpu);
L
Linus Torvalds 已提交
319 320
	if (!stat)
		return 0;
321

322
	old_index = stat->last_index;
L
Linus Torvalds 已提交
323 324
	new_index = freq_table_get_index(stat, freq->new);

325 326
	/* We can't do stat->time_in_state[-1]= .. */
	if (old_index == -1 || new_index == -1)
L
Linus Torvalds 已提交
327 328
		return 0;

329 330 331
	cpufreq_stats_update(freq->cpu);

	if (old_index == new_index)
332 333
		return 0;

L
Linus Torvalds 已提交
334 335 336 337 338 339 340 341 342 343
	spin_lock(&cpufreq_stats_lock);
	stat->last_index = new_index;
#ifdef CONFIG_CPU_FREQ_STAT_DETAILS
	stat->trans_table[old_index * stat->max_state + new_index]++;
#endif
	stat->total_trans++;
	spin_unlock(&cpufreq_stats_lock);
	return 0;
}

344 345 346
static int __cpuinit cpufreq_stat_cpu_callback(struct notifier_block *nfb,
					       unsigned long action,
					       void *hcpu)
347 348 349 350 351
{
	unsigned int cpu = (unsigned long)hcpu;

	switch (action) {
	case CPU_ONLINE:
352
	case CPU_ONLINE_FROZEN:
353 354
		cpufreq_update_policy(cpu);
		break;
355 356 357
	case CPU_DOWN_PREPARE:
		cpufreq_stats_free_sysfs(cpu);
		break;
358
	case CPU_DEAD:
359 360 361 362
		cpufreq_stats_free_table(cpu);
		break;
	case CPU_UP_CANCELED_FROZEN:
		cpufreq_stats_free_sysfs(cpu);
363 364 365 366 367 368
		cpufreq_stats_free_table(cpu);
		break;
	}
	return NOTIFY_OK;
}

369
/* priority=1 so this will get called before cpufreq_remove_dev */
370
static struct notifier_block cpufreq_stat_cpu_notifier __refdata = {
371
	.notifier_call = cpufreq_stat_cpu_callback,
372
	.priority = 1,
373 374
};

L
Linus Torvalds 已提交
375 376 377 378 379 380 381 382
static struct notifier_block notifier_policy_block = {
	.notifier_call = cpufreq_stat_notifier_policy
};

static struct notifier_block notifier_trans_block = {
	.notifier_call = cpufreq_stat_notifier_trans
};

383
static int __init cpufreq_stats_init(void)
L
Linus Torvalds 已提交
384 385 386
{
	int ret;
	unsigned int cpu;
387

L
Linus Torvalds 已提交
388
	spin_lock_init(&cpufreq_stats_lock);
389 390 391
	ret = cpufreq_register_notifier(&notifier_policy_block,
				CPUFREQ_POLICY_NOTIFIER);
	if (ret)
L
Linus Torvalds 已提交
392 393
		return ret;

394 395 396 397
	register_hotcpu_notifier(&cpufreq_stat_cpu_notifier);
	for_each_online_cpu(cpu)
		cpufreq_update_policy(cpu);

398 399 400
	ret = cpufreq_register_notifier(&notifier_trans_block,
				CPUFREQ_TRANSITION_NOTIFIER);
	if (ret) {
L
Linus Torvalds 已提交
401 402
		cpufreq_unregister_notifier(&notifier_policy_block,
				CPUFREQ_POLICY_NOTIFIER);
403 404 405
		unregister_hotcpu_notifier(&cpufreq_stat_cpu_notifier);
		for_each_online_cpu(cpu)
			cpufreq_stats_free_table(cpu);
L
Linus Torvalds 已提交
406 407 408 409 410
		return ret;
	}

	return 0;
}
411
static void __exit cpufreq_stats_exit(void)
L
Linus Torvalds 已提交
412 413
{
	unsigned int cpu;
414

L
Linus Torvalds 已提交
415 416 417 418
	cpufreq_unregister_notifier(&notifier_policy_block,
			CPUFREQ_POLICY_NOTIFIER);
	cpufreq_unregister_notifier(&notifier_trans_block,
			CPUFREQ_TRANSITION_NOTIFIER);
419
	unregister_hotcpu_notifier(&cpufreq_stat_cpu_notifier);
420
	for_each_online_cpu(cpu) {
421
		cpufreq_stats_free_table(cpu);
422
		cpufreq_stats_free_sysfs(cpu);
423
	}
L
Linus Torvalds 已提交
424 425
}

426 427
MODULE_AUTHOR("Zou Nan hai <nanhai.zou@intel.com>");
MODULE_DESCRIPTION("'cpufreq_stats' - A driver to export cpufreq stats "
428
				"through sysfs filesystem");
429
MODULE_LICENSE("GPL");
L
Linus Torvalds 已提交
430 431 432

module_init(cpufreq_stats_init);
module_exit(cpufreq_stats_exit);