xref: /openbmc/linux/drivers/cpufreq/cpufreq_userspace.c (revision de1df26b7cef702a32ae876ed45c1112f523df48)
1c0672860SThomas Renninger 
21da177e4SLinus Torvalds /*
31da177e4SLinus Torvalds  *  linux/drivers/cpufreq/cpufreq_userspace.c
41da177e4SLinus Torvalds  *
51da177e4SLinus Torvalds  *  Copyright (C)  2001 Russell King
61da177e4SLinus Torvalds  *            (C)  2002 - 2004 Dominik Brodowski <linux@brodo.de>
71da177e4SLinus Torvalds  *
81da177e4SLinus Torvalds  * This program is free software; you can redistribute it and/or modify
91da177e4SLinus Torvalds  * it under the terms of the GNU General Public License version 2 as
101da177e4SLinus Torvalds  * published by the Free Software Foundation.
111da177e4SLinus Torvalds  *
121da177e4SLinus Torvalds  */
131da177e4SLinus Torvalds 
14db701151SViresh Kumar #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
15db701151SViresh Kumar 
161da177e4SLinus Torvalds #include <linux/cpufreq.h>
17d1922f02SViresh Kumar #include <linux/init.h>
18d1922f02SViresh Kumar #include <linux/module.h>
193fc54d37Sakpm@osdl.org #include <linux/mutex.h>
201da177e4SLinus Torvalds 
21b38868aaSMike Travis static DEFINE_PER_CPU(unsigned int, cpu_is_managed);
223fc54d37Sakpm@osdl.org static DEFINE_MUTEX(userspace_mutex);
231da177e4SLinus Torvalds 
241da177e4SLinus Torvalds /**
251da177e4SLinus Torvalds  * cpufreq_set - set the CPU frequency
269e76988eSVenki Pallipadi  * @policy: pointer to policy struct where freq is being set
271da177e4SLinus Torvalds  * @freq: target frequency in kHz
281da177e4SLinus Torvalds  *
291da177e4SLinus Torvalds  * Sets the CPU frequency to freq.
301da177e4SLinus Torvalds  */
319e76988eSVenki Pallipadi static int cpufreq_set(struct cpufreq_policy *policy, unsigned int freq)
321da177e4SLinus Torvalds {
331da177e4SLinus Torvalds 	int ret = -EINVAL;
341da177e4SLinus Torvalds 
352d06d8c4SDominik Brodowski 	pr_debug("cpufreq_set for cpu %u, freq %u kHz\n", policy->cpu, freq);
361da177e4SLinus Torvalds 
373fc54d37Sakpm@osdl.org 	mutex_lock(&userspace_mutex);
38b38868aaSMike Travis 	if (!per_cpu(cpu_is_managed, policy->cpu))
391da177e4SLinus Torvalds 		goto err;
401da177e4SLinus Torvalds 
41c0672860SThomas Renninger 	ret = __cpufreq_driver_target(policy, freq, CPUFREQ_RELATION_L);
421da177e4SLinus Torvalds  err:
433fc54d37Sakpm@osdl.org 	mutex_unlock(&userspace_mutex);
441da177e4SLinus Torvalds 	return ret;
451da177e4SLinus Torvalds }
461da177e4SLinus Torvalds 
471da177e4SLinus Torvalds static ssize_t show_speed(struct cpufreq_policy *policy, char *buf)
481da177e4SLinus Torvalds {
49d1922f02SViresh Kumar 	return sprintf(buf, "%u\n", policy->cur);
501da177e4SLinus Torvalds }
511da177e4SLinus Torvalds 
521da177e4SLinus Torvalds static int cpufreq_governor_userspace(struct cpufreq_policy *policy,
531da177e4SLinus Torvalds 				   unsigned int event)
541da177e4SLinus Torvalds {
551da177e4SLinus Torvalds 	unsigned int cpu = policy->cpu;
56914f7c31SJeff Garzik 	int rc = 0;
57914f7c31SJeff Garzik 
581da177e4SLinus Torvalds 	switch (event) {
591da177e4SLinus Torvalds 	case CPUFREQ_GOV_START:
601da177e4SLinus Torvalds 		BUG_ON(!policy->cur);
61d1922f02SViresh Kumar 		pr_debug("started managing cpu %u\n", cpu);
62d1922f02SViresh Kumar 
633fc54d37Sakpm@osdl.org 		mutex_lock(&userspace_mutex);
64b38868aaSMike Travis 		per_cpu(cpu_is_managed, cpu) = 1;
653fc54d37Sakpm@osdl.org 		mutex_unlock(&userspace_mutex);
661da177e4SLinus Torvalds 		break;
671da177e4SLinus Torvalds 	case CPUFREQ_GOV_STOP:
682d06d8c4SDominik Brodowski 		pr_debug("managing cpu %u stopped\n", cpu);
69d1922f02SViresh Kumar 
70d1922f02SViresh Kumar 		mutex_lock(&userspace_mutex);
71d1922f02SViresh Kumar 		per_cpu(cpu_is_managed, cpu) = 0;
723fc54d37Sakpm@osdl.org 		mutex_unlock(&userspace_mutex);
731da177e4SLinus Torvalds 		break;
741da177e4SLinus Torvalds 	case CPUFREQ_GOV_LIMITS:
753fc54d37Sakpm@osdl.org 		mutex_lock(&userspace_mutex);
76d1922f02SViresh Kumar 		pr_debug("limit event for cpu %u: %u - %u kHz, currently %u kHz\n",
77c0672860SThomas Renninger 			cpu, policy->min, policy->max,
78d1922f02SViresh Kumar 			policy->cur);
79d1922f02SViresh Kumar 
80d1922f02SViresh Kumar 		if (policy->max < policy->cur)
81c0672860SThomas Renninger 			__cpufreq_driver_target(policy, policy->max,
821da177e4SLinus Torvalds 						CPUFREQ_RELATION_H);
83d1922f02SViresh Kumar 		else if (policy->min > policy->cur)
84c0672860SThomas Renninger 			__cpufreq_driver_target(policy, policy->min,
851da177e4SLinus Torvalds 						CPUFREQ_RELATION_L);
863fc54d37Sakpm@osdl.org 		mutex_unlock(&userspace_mutex);
871da177e4SLinus Torvalds 		break;
881da177e4SLinus Torvalds 	}
89914f7c31SJeff Garzik 	return rc;
901da177e4SLinus Torvalds }
911da177e4SLinus Torvalds 
92*de1df26bSRafael J. Wysocki static struct cpufreq_governor cpufreq_gov_userspace = {
931da177e4SLinus Torvalds 	.name		= "userspace",
941da177e4SLinus Torvalds 	.governor	= cpufreq_governor_userspace,
959e76988eSVenki Pallipadi 	.store_setspeed	= cpufreq_set,
969e76988eSVenki Pallipadi 	.show_setspeed	= show_speed,
971da177e4SLinus Torvalds 	.owner		= THIS_MODULE,
981da177e4SLinus Torvalds };
991da177e4SLinus Torvalds 
1001da177e4SLinus Torvalds static int __init cpufreq_gov_userspace_init(void)
1011da177e4SLinus Torvalds {
1021da177e4SLinus Torvalds 	return cpufreq_register_governor(&cpufreq_gov_userspace);
1031da177e4SLinus Torvalds }
1041da177e4SLinus Torvalds 
1051da177e4SLinus Torvalds static void __exit cpufreq_gov_userspace_exit(void)
1061da177e4SLinus Torvalds {
1071da177e4SLinus Torvalds 	cpufreq_unregister_governor(&cpufreq_gov_userspace);
1081da177e4SLinus Torvalds }
1091da177e4SLinus Torvalds 
1101bceb8d1SDave Jones MODULE_AUTHOR("Dominik Brodowski <linux@brodo.de>, "
1111bceb8d1SDave Jones 		"Russell King <rmk@arm.linux.org.uk>");
1121da177e4SLinus Torvalds MODULE_DESCRIPTION("CPUfreq policy governor 'userspace'");
1131da177e4SLinus Torvalds MODULE_LICENSE("GPL");
1141da177e4SLinus Torvalds 
1156915719bSJohannes Weiner #ifdef CONFIG_CPU_FREQ_DEFAULT_GOV_USERSPACE
116*de1df26bSRafael J. Wysocki struct cpufreq_governor *cpufreq_default_governor(void)
117*de1df26bSRafael J. Wysocki {
118*de1df26bSRafael J. Wysocki 	return &cpufreq_gov_userspace;
119*de1df26bSRafael J. Wysocki }
120*de1df26bSRafael J. Wysocki 
1211da177e4SLinus Torvalds fs_initcall(cpufreq_gov_userspace_init);
1226915719bSJohannes Weiner #else
1236915719bSJohannes Weiner module_init(cpufreq_gov_userspace_init);
1246915719bSJohannes Weiner #endif
1251da177e4SLinus Torvalds module_exit(cpufreq_gov_userspace_exit);
126