2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2006-01-27 16:15:26 +01:00
2005-04-16 15:20:36 -07:00
/*
* linux / drivers / cpufreq / cpufreq_userspace . c
*
* Copyright ( C ) 2001 Russell King
* ( C ) 2002 - 2004 Dominik Brodowski < linux @ brodo . de >
*/
2012-10-23 01:29:03 +02:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-04-16 15:20:36 -07:00
# include <linux/cpufreq.h>
2013-06-05 11:47:38 +05:30
# include <linux/init.h>
# include <linux/module.h>
2006-01-13 15:54:22 -08:00
# include <linux/mutex.h>
2016-04-29 14:44:37 -07:00
# include <linux/slab.h>
2005-04-16 15:20:36 -07:00
2008-07-18 18:11:32 -07:00
static DEFINE_PER_CPU ( unsigned int , cpu_is_managed ) ;
2023-09-12 06:10:56 +00:00
struct userspace_policy {
unsigned int setspeed ;
struct mutex mutex ;
} ;
2005-04-16 15:20:36 -07:00
2006-02-28 00:43:23 -05:00
/**
2005-04-16 15:20:36 -07:00
* cpufreq_set - set the CPU frequency
2007-10-26 10:18:21 -07:00
* @ policy : pointer to policy struct where freq is being set
2005-04-16 15:20:36 -07:00
* @ freq : target frequency in kHz
*
* Sets the CPU frequency to freq .
*/
2007-10-26 10:18:21 -07:00
static int cpufreq_set ( struct cpufreq_policy * policy , unsigned int freq )
2005-04-16 15:20:36 -07:00
{
int ret = - EINVAL ;
2023-09-12 06:10:56 +00:00
struct userspace_policy * userspace = policy - > governor_data ;
2005-04-16 15:20:36 -07:00
2011-03-27 15:04:46 +02:00
pr_debug ( " cpufreq_set for cpu %u, freq %u kHz \n " , policy - > cpu , freq ) ;
2005-04-16 15:20:36 -07:00
2023-09-12 06:10:56 +00:00
mutex_lock ( & userspace - > mutex ) ;
2008-07-18 18:11:32 -07:00
if ( ! per_cpu ( cpu_is_managed , policy - > cpu ) )
2005-04-16 15:20:36 -07:00
goto err ;
2023-09-12 06:10:56 +00:00
userspace - > setspeed = freq ;
2016-04-29 14:44:37 -07:00
2006-01-27 16:15:26 +01:00
ret = __cpufreq_driver_target ( policy , freq , CPUFREQ_RELATION_L ) ;
2005-04-16 15:20:36 -07:00
err :
2023-09-12 06:10:56 +00:00
mutex_unlock ( & userspace - > mutex ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
2007-10-26 10:18:21 -07:00
static ssize_t show_speed ( struct cpufreq_policy * policy , char * buf )
2005-04-16 15:20:36 -07:00
{
2013-06-05 11:47:38 +05:30
return sprintf ( buf , " %u \n " , policy - > cur ) ;
2005-04-16 15:20:36 -07:00
}
2016-04-29 14:44:37 -07:00
static int cpufreq_userspace_policy_init ( struct cpufreq_policy * policy )
{
2023-09-12 06:10:56 +00:00
struct userspace_policy * userspace ;
2016-04-29 14:44:37 -07:00
2023-09-12 06:10:56 +00:00
userspace = kzalloc ( sizeof ( * userspace ) , GFP_KERNEL ) ;
if ( ! userspace )
2016-04-29 14:44:37 -07:00
return - ENOMEM ;
2023-09-12 06:10:56 +00:00
mutex_init ( & userspace - > mutex ) ;
policy - > governor_data = userspace ;
2016-04-29 14:44:37 -07:00
return 0 ;
}
2023-09-12 06:10:56 +00:00
/*
* Any routine that writes to the policy struct will hold the " rwsem " of
* policy struct that means it is free to free " governor_data " here .
*/
2016-06-02 23:24:15 +02:00
static void cpufreq_userspace_policy_exit ( struct cpufreq_policy * policy )
{
kfree ( policy - > governor_data ) ;
policy - > governor_data = NULL ;
}
static int cpufreq_userspace_policy_start ( struct cpufreq_policy * policy )
2005-04-16 15:20:36 -07:00
{
2023-09-12 06:10:56 +00:00
struct userspace_policy * userspace = policy - > governor_data ;
2006-10-20 14:31:00 -07:00
2016-06-02 23:24:15 +02:00
BUG_ON ( ! policy - > cur ) ;
pr_debug ( " started managing cpu %u \n " , policy - > cpu ) ;
2016-04-29 14:44:37 -07:00
2023-09-12 06:10:56 +00:00
mutex_lock ( & userspace - > mutex ) ;
2016-06-02 23:24:15 +02:00
per_cpu ( cpu_is_managed , policy - > cpu ) = 1 ;
2023-09-12 06:10:56 +00:00
userspace - > setspeed = policy - > cur ;
mutex_unlock ( & userspace - > mutex ) ;
2016-06-02 23:24:15 +02:00
return 0 ;
}
static void cpufreq_userspace_policy_stop ( struct cpufreq_policy * policy )
{
2023-09-12 06:10:56 +00:00
struct userspace_policy * userspace = policy - > governor_data ;
2016-06-02 23:24:15 +02:00
pr_debug ( " managing cpu %u stopped \n " , policy - > cpu ) ;
2023-09-12 06:10:56 +00:00
mutex_lock ( & userspace - > mutex ) ;
2016-06-02 23:24:15 +02:00
per_cpu ( cpu_is_managed , policy - > cpu ) = 0 ;
2023-09-12 06:10:56 +00:00
userspace - > setspeed = 0 ;
mutex_unlock ( & userspace - > mutex ) ;
2016-06-02 23:24:15 +02:00
}
static void cpufreq_userspace_policy_limits ( struct cpufreq_policy * policy )
{
2023-09-12 06:10:56 +00:00
struct userspace_policy * userspace = policy - > governor_data ;
2016-06-02 23:24:15 +02:00
2023-09-12 06:10:56 +00:00
mutex_lock ( & userspace - > mutex ) ;
2016-06-02 23:24:15 +02:00
pr_debug ( " limit event for cpu %u: %u - %u kHz, currently %u kHz, last set to %u kHz \n " ,
2023-09-12 06:10:56 +00:00
policy - > cpu , policy - > min , policy - > max , policy - > cur , userspace - > setspeed ) ;
if ( policy - > max < userspace - > setspeed )
__cpufreq_driver_target ( policy , policy - > max ,
CPUFREQ_RELATION_H ) ;
else if ( policy - > min > userspace - > setspeed )
__cpufreq_driver_target ( policy , policy - > min ,
CPUFREQ_RELATION_L ) ;
2016-06-02 23:24:15 +02:00
else
2023-09-12 06:10:56 +00:00
__cpufreq_driver_target ( policy , userspace - > setspeed ,
CPUFREQ_RELATION_L ) ;
2016-06-02 23:24:15 +02:00
2023-09-12 06:10:56 +00:00
mutex_unlock ( & userspace - > mutex ) ;
2005-04-16 15:20:36 -07:00
}
2016-02-05 02:37:42 +01:00
static struct cpufreq_governor cpufreq_gov_userspace = {
2005-04-16 15:20:36 -07:00
. name = " userspace " ,
2016-06-02 23:24:15 +02:00
. init = cpufreq_userspace_policy_init ,
. exit = cpufreq_userspace_policy_exit ,
. start = cpufreq_userspace_policy_start ,
. stop = cpufreq_userspace_policy_stop ,
. limits = cpufreq_userspace_policy_limits ,
2007-10-26 10:18:21 -07:00
. store_setspeed = cpufreq_set ,
. show_setspeed = show_speed ,
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
} ;
2009-01-18 01:51:46 -05:00
MODULE_AUTHOR ( " Dominik Brodowski <linux@brodo.de>, "
" Russell King <rmk@arm.linux.org.uk> " ) ;
MODULE_DESCRIPTION ( " CPUfreq policy governor 'userspace' " ) ;
MODULE_LICENSE ( " GPL " ) ;
2005-04-16 15:20:36 -07:00
2008-01-17 15:21:08 -08:00
# ifdef CONFIG_CPU_FREQ_DEFAULT_GOV_USERSPACE
2016-02-05 02:37:42 +01:00
struct cpufreq_governor * cpufreq_default_governor ( void )
{
return & cpufreq_gov_userspace ;
}
2008-01-17 15:21:08 -08:00
# endif
2020-06-29 13:54:59 +05:30
cpufreq_governor_init ( cpufreq_gov_userspace ) ;
cpufreq_governor_exit ( cpufreq_gov_userspace ) ;