2013-02-06 21:02:13 +04:00
/*
2013-04-10 02:38:18 +04:00
* intel_pstate . c : Native P state management for Intel processors
2013-02-06 21:02:13 +04:00
*
* ( C ) Copyright 2012 Intel Corporation
* Author : Dirk Brandewie < dirk . j . brandewie @ intel . com >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; version 2
* of the License .
*/
# include <linux/kernel.h>
# include <linux/kernel_stat.h>
# include <linux/module.h>
# include <linux/ktime.h>
# include <linux/hrtimer.h>
# include <linux/tick.h>
# include <linux/slab.h>
# include <linux/sched.h>
# include <linux/list.h>
# include <linux/cpu.h>
# include <linux/cpufreq.h>
# include <linux/sysfs.h>
# include <linux/types.h>
# include <linux/fs.h>
# include <linux/debugfs.h>
2013-10-31 19:24:05 +04:00
# include <linux/acpi.h>
2013-02-06 21:02:13 +04:00
# include <trace/events/power.h>
# include <asm/div64.h>
# include <asm/msr.h>
# include <asm/cpu_device_id.h>
# define SAMPLE_COUNT 3
2014-02-12 22:01:07 +04:00
# define BYT_RATIOS 0x66a
# define BYT_VIDS 0x66b
# define BYT_TURBO_RATIOS 0x66c
2014-05-08 23:57:23 +04:00
# define BYT_TURBO_VIDS 0x66d
2014-02-12 22:01:07 +04:00
2013-10-21 20:20:35 +04:00
2014-02-25 22:35:37 +04:00
# define FRAC_BITS 6
2013-02-06 21:02:13 +04:00
# define int_tofp(X) ((int64_t)(X) << FRAC_BITS)
# define fp_toint(X) ((X) >> FRAC_BITS)
2014-02-25 22:35:37 +04:00
# define FP_ROUNDUP(X) ((X) += 1 << FRAC_BITS)
2013-02-06 21:02:13 +04:00
static inline int32_t mul_fp ( int32_t x , int32_t y )
{
return ( ( int64_t ) x * ( int64_t ) y ) > > FRAC_BITS ;
}
static inline int32_t div_fp ( int32_t x , int32_t y )
{
return div_s64 ( ( int64_t ) x < < FRAC_BITS , ( int64_t ) y ) ;
}
struct sample {
2013-10-21 20:20:32 +04:00
int32_t core_pct_busy ;
2013-02-06 21:02:13 +04:00
u64 aperf ;
u64 mperf ;
2014-02-03 20:55:31 +04:00
unsigned long long tsc ;
2013-02-06 21:02:13 +04:00
int freq ;
} ;
struct pstate_data {
int current_pstate ;
int min_pstate ;
int max_pstate ;
int turbo_pstate ;
} ;
2013-12-18 22:32:39 +04:00
struct vid_data {
2014-05-08 23:57:23 +04:00
int min ;
int max ;
int turbo ;
2013-12-18 22:32:39 +04:00
int32_t ratio ;
} ;
2013-02-06 21:02:13 +04:00
struct _pid {
int setpoint ;
int32_t integral ;
int32_t p_gain ;
int32_t i_gain ;
int32_t d_gain ;
int deadband ;
2013-10-21 20:20:32 +04:00
int32_t last_err ;
2013-02-06 21:02:13 +04:00
} ;
struct cpudata {
int cpu ;
char name [ 64 ] ;
struct timer_list timer ;
struct pstate_data pstate ;
2013-12-18 22:32:39 +04:00
struct vid_data vid ;
2013-02-06 21:02:13 +04:00
struct _pid pid ;
u64 prev_aperf ;
u64 prev_mperf ;
2014-02-03 20:55:31 +04:00
unsigned long long prev_tsc ;
2014-02-12 22:01:04 +04:00
struct sample sample ;
2013-02-06 21:02:13 +04:00
} ;
static struct cpudata * * all_cpu_data ;
struct pstate_adjust_policy {
int sample_rate_ms ;
int deadband ;
int setpoint ;
int p_gain_pct ;
int d_gain_pct ;
int i_gain_pct ;
} ;
2013-10-21 20:20:34 +04:00
struct pstate_funcs {
int ( * get_max ) ( void ) ;
int ( * get_min ) ( void ) ;
int ( * get_turbo ) ( void ) ;
2013-12-18 22:32:39 +04:00
void ( * set ) ( struct cpudata * , int pstate ) ;
void ( * get_vid ) ( struct cpudata * ) ;
2013-02-06 21:02:13 +04:00
} ;
2013-10-21 20:20:34 +04:00
struct cpu_defaults {
struct pstate_adjust_policy pid_policy ;
struct pstate_funcs funcs ;
2013-02-06 21:02:13 +04:00
} ;
2013-10-21 20:20:34 +04:00
static struct pstate_adjust_policy pid_params ;
static struct pstate_funcs pstate_funcs ;
2013-02-06 21:02:13 +04:00
struct perf_limits {
int no_turbo ;
int max_perf_pct ;
int min_perf_pct ;
int32_t max_perf ;
int32_t min_perf ;
2013-05-07 19:20:26 +04:00
int max_policy_pct ;
int max_sysfs_pct ;
2013-02-06 21:02:13 +04:00
} ;
static struct perf_limits limits = {
. no_turbo = 0 ,
. max_perf_pct = 100 ,
. max_perf = int_tofp ( 1 ) ,
. min_perf_pct = 0 ,
. min_perf = 0 ,
2013-05-07 19:20:26 +04:00
. max_policy_pct = 100 ,
. max_sysfs_pct = 100 ,
2013-02-06 21:02:13 +04:00
} ;
static inline void pid_reset ( struct _pid * pid , int setpoint , int busy ,
int deadband , int integral ) {
pid - > setpoint = setpoint ;
pid - > deadband = deadband ;
pid - > integral = int_tofp ( integral ) ;
2014-02-12 22:01:05 +04:00
pid - > last_err = int_tofp ( setpoint ) - int_tofp ( busy ) ;
2013-02-06 21:02:13 +04:00
}
static inline void pid_p_gain_set ( struct _pid * pid , int percent )
{
pid - > p_gain = div_fp ( int_tofp ( percent ) , int_tofp ( 100 ) ) ;
}
static inline void pid_i_gain_set ( struct _pid * pid , int percent )
{
pid - > i_gain = div_fp ( int_tofp ( percent ) , int_tofp ( 100 ) ) ;
}
static inline void pid_d_gain_set ( struct _pid * pid , int percent )
{
pid - > d_gain = div_fp ( int_tofp ( percent ) , int_tofp ( 100 ) ) ;
}
2013-10-21 20:20:32 +04:00
static signed int pid_calc ( struct _pid * pid , int32_t busy )
2013-02-06 21:02:13 +04:00
{
2013-10-21 20:20:32 +04:00
signed int result ;
2013-02-06 21:02:13 +04:00
int32_t pterm , dterm , fp_error ;
int32_t integral_limit ;
2013-10-21 20:20:32 +04:00
fp_error = int_tofp ( pid - > setpoint ) - busy ;
2013-02-06 21:02:13 +04:00
2013-10-21 20:20:32 +04:00
if ( abs ( fp_error ) < = int_tofp ( pid - > deadband ) )
2013-02-06 21:02:13 +04:00
return 0 ;
pterm = mul_fp ( pid - > p_gain , fp_error ) ;
pid - > integral + = fp_error ;
/* limit the integral term */
integral_limit = int_tofp ( 30 ) ;
if ( pid - > integral > integral_limit )
pid - > integral = integral_limit ;
if ( pid - > integral < - integral_limit )
pid - > integral = - integral_limit ;
2013-10-21 20:20:32 +04:00
dterm = mul_fp ( pid - > d_gain , fp_error - pid - > last_err ) ;
pid - > last_err = fp_error ;
2013-02-06 21:02:13 +04:00
result = pterm + mul_fp ( pid - > integral , pid - > i_gain ) + dterm ;
return ( signed int ) fp_toint ( result ) ;
}
static inline void intel_pstate_busy_pid_reset ( struct cpudata * cpu )
{
2013-10-21 20:20:34 +04:00
pid_p_gain_set ( & cpu - > pid , pid_params . p_gain_pct ) ;
pid_d_gain_set ( & cpu - > pid , pid_params . d_gain_pct ) ;
pid_i_gain_set ( & cpu - > pid , pid_params . i_gain_pct ) ;
2013-02-06 21:02:13 +04:00
pid_reset ( & cpu - > pid ,
2013-10-21 20:20:34 +04:00
pid_params . setpoint ,
2013-02-06 21:02:13 +04:00
100 ,
2013-10-21 20:20:34 +04:00
pid_params . deadband ,
2013-02-06 21:02:13 +04:00
0 ) ;
}
static inline void intel_pstate_reset_all_pid ( void )
{
unsigned int cpu ;
for_each_online_cpu ( cpu ) {
if ( all_cpu_data [ cpu ] )
intel_pstate_busy_pid_reset ( all_cpu_data [ cpu ] ) ;
}
}
/************************** debugfs begin ************************/
static int pid_param_set ( void * data , u64 val )
{
* ( u32 * ) data = val ;
intel_pstate_reset_all_pid ( ) ;
return 0 ;
}
static int pid_param_get ( void * data , u64 * val )
{
* val = * ( u32 * ) data ;
return 0 ;
}
DEFINE_SIMPLE_ATTRIBUTE ( fops_pid_param , pid_param_get ,
pid_param_set , " %llu \n " ) ;
struct pid_param {
char * name ;
void * value ;
} ;
static struct pid_param pid_files [ ] = {
2013-10-21 20:20:34 +04:00
{ " sample_rate_ms " , & pid_params . sample_rate_ms } ,
{ " d_gain_pct " , & pid_params . d_gain_pct } ,
{ " i_gain_pct " , & pid_params . i_gain_pct } ,
{ " deadband " , & pid_params . deadband } ,
{ " setpoint " , & pid_params . setpoint } ,
{ " p_gain_pct " , & pid_params . p_gain_pct } ,
2013-02-06 21:02:13 +04:00
{ NULL , NULL }
} ;
static struct dentry * debugfs_parent ;
static void intel_pstate_debug_expose_params ( void )
{
int i = 0 ;
debugfs_parent = debugfs_create_dir ( " pstate_snb " , NULL ) ;
if ( IS_ERR_OR_NULL ( debugfs_parent ) )
return ;
while ( pid_files [ i ] . name ) {
debugfs_create_file ( pid_files [ i ] . name , 0660 ,
debugfs_parent , pid_files [ i ] . value ,
& fops_pid_param ) ;
i + + ;
}
}
/************************** debugfs end ************************/
/************************** sysfs begin ************************/
# define show_one(file_name, object) \
static ssize_t show_ # # file_name \
( struct kobject * kobj , struct attribute * attr , char * buf ) \
{ \
return sprintf ( buf , " %u \n " , limits . object ) ; \
}
static ssize_t store_no_turbo ( struct kobject * a , struct attribute * b ,
const char * buf , size_t count )
{
unsigned int input ;
int ret ;
ret = sscanf ( buf , " %u " , & input ) ;
if ( ret ! = 1 )
return - EINVAL ;
limits . no_turbo = clamp_t ( int , input , 0 , 1 ) ;
return count ;
}
static ssize_t store_max_perf_pct ( struct kobject * a , struct attribute * b ,
const char * buf , size_t count )
{
unsigned int input ;
int ret ;
ret = sscanf ( buf , " %u " , & input ) ;
if ( ret ! = 1 )
return - EINVAL ;
2013-05-07 19:20:26 +04:00
limits . max_sysfs_pct = clamp_t ( int , input , 0 , 100 ) ;
limits . max_perf_pct = min ( limits . max_policy_pct , limits . max_sysfs_pct ) ;
2013-02-06 21:02:13 +04:00
limits . max_perf = div_fp ( int_tofp ( limits . max_perf_pct ) , int_tofp ( 100 ) ) ;
return count ;
}
static ssize_t store_min_perf_pct ( struct kobject * a , struct attribute * b ,
const char * buf , size_t count )
{
unsigned int input ;
int ret ;
ret = sscanf ( buf , " %u " , & input ) ;
if ( ret ! = 1 )
return - EINVAL ;
limits . min_perf_pct = clamp_t ( int , input , 0 , 100 ) ;
limits . min_perf = div_fp ( int_tofp ( limits . min_perf_pct ) , int_tofp ( 100 ) ) ;
return count ;
}
show_one ( no_turbo , no_turbo ) ;
show_one ( max_perf_pct , max_perf_pct ) ;
show_one ( min_perf_pct , min_perf_pct ) ;
define_one_global_rw ( no_turbo ) ;
define_one_global_rw ( max_perf_pct ) ;
define_one_global_rw ( min_perf_pct ) ;
static struct attribute * intel_pstate_attributes [ ] = {
& no_turbo . attr ,
& max_perf_pct . attr ,
& min_perf_pct . attr ,
NULL
} ;
static struct attribute_group intel_pstate_attr_group = {
. attrs = intel_pstate_attributes ,
} ;
static struct kobject * intel_pstate_kobject ;
static void intel_pstate_sysfs_expose_params ( void )
{
int rc ;
intel_pstate_kobject = kobject_create_and_add ( " intel_pstate " ,
& cpu_subsys . dev_root - > kobj ) ;
BUG_ON ( ! intel_pstate_kobject ) ;
rc = sysfs_create_group ( intel_pstate_kobject ,
& intel_pstate_attr_group ) ;
BUG_ON ( rc ) ;
}
/************************** sysfs end ************************/
2013-10-21 20:20:35 +04:00
static int byt_get_min_pstate ( void )
{
u64 value ;
rdmsrl ( BYT_RATIOS , value ) ;
2014-05-08 23:57:23 +04:00
return ( value > > 8 ) & 0x3F ;
2013-10-21 20:20:35 +04:00
}
static int byt_get_max_pstate ( void )
{
u64 value ;
rdmsrl ( BYT_RATIOS , value ) ;
2014-05-08 23:57:23 +04:00
return ( value > > 16 ) & 0x3F ;
2013-10-21 20:20:35 +04:00
}
2013-02-06 21:02:13 +04:00
2014-02-12 22:01:07 +04:00
static int byt_get_turbo_pstate ( void )
{
u64 value ;
rdmsrl ( BYT_TURBO_RATIOS , value ) ;
return value & 0x3F ;
}
2013-12-18 22:32:39 +04:00
static void byt_set_pstate ( struct cpudata * cpudata , int pstate )
{
u64 val ;
int32_t vid_fp ;
u32 vid ;
val = pstate < < 8 ;
if ( limits . no_turbo )
val | = ( u64 ) 1 < < 32 ;
vid_fp = cpudata - > vid . min + mul_fp (
int_tofp ( pstate - cpudata - > pstate . min_pstate ) ,
cpudata - > vid . ratio ) ;
vid_fp = clamp_t ( int32_t , vid_fp , cpudata - > vid . min , cpudata - > vid . max ) ;
vid = fp_toint ( vid_fp ) ;
2014-05-08 23:57:23 +04:00
if ( pstate > cpudata - > pstate . max_pstate )
vid = cpudata - > vid . turbo ;
2013-12-18 22:32:39 +04:00
val | = vid ;
wrmsrl ( MSR_IA32_PERF_CTL , val ) ;
}
static void byt_get_vid ( struct cpudata * cpudata )
{
u64 value ;
2014-05-08 23:57:23 +04:00
2013-12-18 22:32:39 +04:00
rdmsrl ( BYT_VIDS , value ) ;
2014-05-08 23:57:23 +04:00
cpudata - > vid . min = int_tofp ( ( value > > 8 ) & 0x3f ) ;
cpudata - > vid . max = int_tofp ( ( value > > 16 ) & 0x3f ) ;
2013-12-18 22:32:39 +04:00
cpudata - > vid . ratio = div_fp (
cpudata - > vid . max - cpudata - > vid . min ,
int_tofp ( cpudata - > pstate . max_pstate -
cpudata - > pstate . min_pstate ) ) ;
2014-05-08 23:57:23 +04:00
rdmsrl ( BYT_TURBO_VIDS , value ) ;
cpudata - > vid . turbo = value & 0x7f ;
2013-12-18 22:32:39 +04:00
}
2013-10-21 20:20:34 +04:00
static int core_get_min_pstate ( void )
2013-02-06 21:02:13 +04:00
{
u64 value ;
2013-03-20 18:21:10 +04:00
rdmsrl ( MSR_PLATFORM_INFO , value ) ;
2013-02-06 21:02:13 +04:00
return ( value > > 40 ) & 0xFF ;
}
2013-10-21 20:20:34 +04:00
static int core_get_max_pstate ( void )
2013-02-06 21:02:13 +04:00
{
u64 value ;
2013-03-20 18:21:10 +04:00
rdmsrl ( MSR_PLATFORM_INFO , value ) ;
2013-02-06 21:02:13 +04:00
return ( value > > 8 ) & 0xFF ;
}
2013-10-21 20:20:34 +04:00
static int core_get_turbo_pstate ( void )
2013-02-06 21:02:13 +04:00
{
u64 value ;
int nont , ret ;
2013-03-20 18:21:10 +04:00
rdmsrl ( MSR_NHM_TURBO_RATIO_LIMIT , value ) ;
2013-10-21 20:20:34 +04:00
nont = core_get_max_pstate ( ) ;
2013-02-06 21:02:13 +04:00
ret = ( ( value ) & 255 ) ;
if ( ret < = nont )
ret = nont ;
return ret ;
}
2013-12-18 22:32:39 +04:00
static void core_set_pstate ( struct cpudata * cpudata , int pstate )
2013-10-21 20:20:34 +04:00
{
u64 val ;
val = pstate < < 8 ;
if ( limits . no_turbo )
val | = ( u64 ) 1 < < 32 ;
2014-03-19 19:45:54 +04:00
wrmsrl_on_cpu ( cpudata - > cpu , MSR_IA32_PERF_CTL , val ) ;
2013-10-21 20:20:34 +04:00
}
static struct cpu_defaults core_params = {
. pid_policy = {
. sample_rate_ms = 10 ,
. deadband = 0 ,
. setpoint = 97 ,
. p_gain_pct = 20 ,
. d_gain_pct = 0 ,
. i_gain_pct = 0 ,
} ,
. funcs = {
. get_max = core_get_max_pstate ,
. get_min = core_get_min_pstate ,
. get_turbo = core_get_turbo_pstate ,
. set = core_set_pstate ,
} ,
} ;
2013-10-21 20:20:35 +04:00
static struct cpu_defaults byt_params = {
. pid_policy = {
. sample_rate_ms = 10 ,
. deadband = 0 ,
. setpoint = 97 ,
. p_gain_pct = 14 ,
. d_gain_pct = 0 ,
. i_gain_pct = 4 ,
} ,
. funcs = {
. get_max = byt_get_max_pstate ,
. get_min = byt_get_min_pstate ,
2014-02-12 22:01:07 +04:00
. get_turbo = byt_get_turbo_pstate ,
2013-12-18 22:32:39 +04:00
. set = byt_set_pstate ,
. get_vid = byt_get_vid ,
2013-10-21 20:20:35 +04:00
} ,
} ;
2013-02-06 21:02:13 +04:00
static void intel_pstate_get_min_max ( struct cpudata * cpu , int * min , int * max )
{
int max_perf = cpu - > pstate . turbo_pstate ;
2013-10-21 20:20:33 +04:00
int max_perf_adj ;
2013-02-06 21:02:13 +04:00
int min_perf ;
if ( limits . no_turbo )
max_perf = cpu - > pstate . max_pstate ;
2013-10-21 20:20:33 +04:00
max_perf_adj = fp_toint ( mul_fp ( int_tofp ( max_perf ) , limits . max_perf ) ) ;
* max = clamp_t ( int , max_perf_adj ,
2013-02-06 21:02:13 +04:00
cpu - > pstate . min_pstate , cpu - > pstate . turbo_pstate ) ;
min_perf = fp_toint ( mul_fp ( int_tofp ( max_perf ) , limits . min_perf ) ) ;
* min = clamp_t ( int , min_perf ,
cpu - > pstate . min_pstate , max_perf ) ;
}
static void intel_pstate_set_pstate ( struct cpudata * cpu , int pstate )
{
int max_perf , min_perf ;
intel_pstate_get_min_max ( cpu , & min_perf , & max_perf ) ;
pstate = clamp_t ( int , pstate , min_perf , max_perf ) ;
if ( pstate = = cpu - > pstate . current_pstate )
return ;
trace_cpu_frequency ( pstate * 100000 , cpu - > cpu ) ;
2013-05-07 19:20:30 +04:00
2013-02-06 21:02:13 +04:00
cpu - > pstate . current_pstate = pstate ;
2013-12-18 22:32:39 +04:00
pstate_funcs . set ( cpu , pstate ) ;
2013-02-06 21:02:13 +04:00
}
static inline void intel_pstate_pstate_increase ( struct cpudata * cpu , int steps )
{
int target ;
target = cpu - > pstate . current_pstate + steps ;
intel_pstate_set_pstate ( cpu , target ) ;
}
static inline void intel_pstate_pstate_decrease ( struct cpudata * cpu , int steps )
{
int target ;
target = cpu - > pstate . current_pstate - steps ;
intel_pstate_set_pstate ( cpu , target ) ;
}
static void intel_pstate_get_cpu_pstates ( struct cpudata * cpu )
{
sprintf ( cpu - > name , " Intel 2nd generation core " ) ;
2013-10-21 20:20:34 +04:00
cpu - > pstate . min_pstate = pstate_funcs . get_min ( ) ;
cpu - > pstate . max_pstate = pstate_funcs . get_max ( ) ;
cpu - > pstate . turbo_pstate = pstate_funcs . get_turbo ( ) ;
2013-02-06 21:02:13 +04:00
2013-12-18 22:32:39 +04:00
if ( pstate_funcs . get_vid )
pstate_funcs . get_vid ( cpu ) ;
2014-05-08 23:57:24 +04:00
intel_pstate_set_pstate ( cpu , cpu - > pstate . min_pstate ) ;
2013-02-06 21:02:13 +04:00
}
static inline void intel_pstate_calc_busy ( struct cpudata * cpu ,
struct sample * sample )
{
2014-02-25 22:35:37 +04:00
int32_t core_pct ;
int32_t c0_pct ;
2013-02-06 21:02:13 +04:00
2014-02-25 22:35:37 +04:00
core_pct = div_fp ( int_tofp ( ( sample - > aperf ) ) ,
int_tofp ( ( sample - > mperf ) ) ) ;
core_pct = mul_fp ( core_pct , int_tofp ( 100 ) ) ;
FP_ROUNDUP ( core_pct ) ;
c0_pct = div_fp ( int_tofp ( sample - > mperf ) , int_tofp ( sample - > tsc ) ) ;
2014-02-03 20:55:31 +04:00
sample - > freq = fp_toint (
2014-02-25 22:35:37 +04:00
mul_fp ( int_tofp ( cpu - > pstate . max_pstate * 1000 ) , core_pct ) ) ;
2014-02-03 20:55:31 +04:00
2014-02-25 22:35:37 +04:00
sample - > core_pct_busy = mul_fp ( core_pct , c0_pct ) ;
2013-02-06 21:02:13 +04:00
}
static inline void intel_pstate_sample ( struct cpudata * cpu )
{
u64 aperf , mperf ;
2014-02-03 20:55:31 +04:00
unsigned long long tsc ;
2013-02-06 21:02:13 +04:00
rdmsrl ( MSR_IA32_APERF , aperf ) ;
rdmsrl ( MSR_IA32_MPERF , mperf ) ;
2014-02-03 20:55:31 +04:00
tsc = native_read_tsc ( ) ;
2014-01-16 22:32:25 +04:00
2014-02-25 22:35:37 +04:00
aperf = aperf > > FRAC_BITS ;
mperf = mperf > > FRAC_BITS ;
tsc = tsc > > FRAC_BITS ;
2014-02-12 22:01:04 +04:00
cpu - > sample . aperf = aperf ;
cpu - > sample . mperf = mperf ;
cpu - > sample . tsc = tsc ;
cpu - > sample . aperf - = cpu - > prev_aperf ;
cpu - > sample . mperf - = cpu - > prev_mperf ;
cpu - > sample . tsc - = cpu - > prev_tsc ;
2013-05-07 19:20:25 +04:00
2014-02-12 22:01:04 +04:00
intel_pstate_calc_busy ( cpu , & cpu - > sample ) ;
2013-02-06 21:02:13 +04:00
cpu - > prev_aperf = aperf ;
cpu - > prev_mperf = mperf ;
2014-02-03 20:55:31 +04:00
cpu - > prev_tsc = tsc ;
2013-02-06 21:02:13 +04:00
}
static inline void intel_pstate_set_sample_time ( struct cpudata * cpu )
{
int sample_time , delay ;
2013-10-21 20:20:34 +04:00
sample_time = pid_params . sample_rate_ms ;
2013-02-06 21:02:13 +04:00
delay = msecs_to_jiffies ( sample_time ) ;
mod_timer_pinned ( & cpu - > timer , jiffies + delay ) ;
}
2013-10-21 20:20:32 +04:00
static inline int32_t intel_pstate_get_scaled_busy ( struct cpudata * cpu )
2013-02-06 21:02:13 +04:00
{
2013-07-18 19:48:42 +04:00
int32_t core_busy , max_pstate , current_pstate ;
2013-02-06 21:02:13 +04:00
2014-02-12 22:01:04 +04:00
core_busy = cpu - > sample . core_pct_busy ;
2013-07-18 19:48:42 +04:00
max_pstate = int_tofp ( cpu - > pstate . max_pstate ) ;
2013-02-06 21:02:13 +04:00
current_pstate = int_tofp ( cpu - > pstate . current_pstate ) ;
2014-02-25 22:35:37 +04:00
core_busy = mul_fp ( core_busy , div_fp ( max_pstate , current_pstate ) ) ;
return FP_ROUNDUP ( core_busy ) ;
2013-02-06 21:02:13 +04:00
}
static inline void intel_pstate_adjust_busy_pstate ( struct cpudata * cpu )
{
2013-10-21 20:20:32 +04:00
int32_t busy_scaled ;
2013-02-06 21:02:13 +04:00
struct _pid * pid ;
signed int ctl = 0 ;
int steps ;
pid = & cpu - > pid ;
busy_scaled = intel_pstate_get_scaled_busy ( cpu ) ;
ctl = pid_calc ( pid , busy_scaled ) ;
steps = abs ( ctl ) ;
2014-01-16 22:32:25 +04:00
2013-02-06 21:02:13 +04:00
if ( ctl < 0 )
intel_pstate_pstate_increase ( cpu , steps ) ;
else
intel_pstate_pstate_decrease ( cpu , steps ) ;
}
static void intel_pstate_timer_func ( unsigned long __data )
{
struct cpudata * cpu = ( struct cpudata * ) __data ;
2014-01-16 22:32:25 +04:00
struct sample * sample ;
2013-02-06 21:02:13 +04:00
intel_pstate_sample ( cpu ) ;
2014-01-16 22:32:25 +04:00
2014-02-12 22:01:04 +04:00
sample = & cpu - > sample ;
2014-01-16 22:32:25 +04:00
2013-05-07 19:20:27 +04:00
intel_pstate_adjust_busy_pstate ( cpu ) ;
2014-01-16 22:32:25 +04:00
trace_pstate_sample ( fp_toint ( sample - > core_pct_busy ) ,
fp_toint ( intel_pstate_get_scaled_busy ( cpu ) ) ,
cpu - > pstate . current_pstate ,
sample - > mperf ,
sample - > aperf ,
sample - > freq ) ;
2013-02-06 21:02:13 +04:00
intel_pstate_set_sample_time ( cpu ) ;
}
# define ICPU(model, policy) \
2014-01-06 22:59:16 +04:00
{ X86_VENDOR_INTEL , 6 , model , X86_FEATURE_APERFMPERF , \
( unsigned long ) & policy }
2013-02-06 21:02:13 +04:00
static const struct x86_cpu_id intel_pstate_cpu_ids [ ] = {
2013-10-21 20:20:34 +04:00
ICPU ( 0x2a , core_params ) ,
ICPU ( 0x2d , core_params ) ,
2013-10-21 20:20:35 +04:00
ICPU ( 0x37 , byt_params ) ,
2013-10-21 20:20:34 +04:00
ICPU ( 0x3a , core_params ) ,
ICPU ( 0x3c , core_params ) ,
ICPU ( 0x3e , core_params ) ,
ICPU ( 0x3f , core_params ) ,
ICPU ( 0x45 , core_params ) ,
ICPU ( 0x46 , core_params ) ,
2013-02-06 21:02:13 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( x86cpu , intel_pstate_cpu_ids ) ;
static int intel_pstate_init_cpu ( unsigned int cpunum )
{
const struct x86_cpu_id * id ;
struct cpudata * cpu ;
id = x86_match_cpu ( intel_pstate_cpu_ids ) ;
if ( ! id )
return - ENODEV ;
all_cpu_data [ cpunum ] = kzalloc ( sizeof ( struct cpudata ) , GFP_KERNEL ) ;
if ( ! all_cpu_data [ cpunum ] )
return - ENOMEM ;
cpu = all_cpu_data [ cpunum ] ;
intel_pstate_get_cpu_pstates ( cpu ) ;
cpu - > cpu = cpunum ;
2013-10-21 20:20:34 +04:00
2013-02-06 21:02:13 +04:00
init_timer_deferrable ( & cpu - > timer ) ;
cpu - > timer . function = intel_pstate_timer_func ;
cpu - > timer . data =
( unsigned long ) cpu ;
cpu - > timer . expires = jiffies + HZ / 100 ;
intel_pstate_busy_pid_reset ( cpu ) ;
intel_pstate_sample ( cpu ) ;
add_timer_on ( & cpu - > timer , cpunum ) ;
pr_info ( " Intel pstate controlling: cpu %d \n " , cpunum ) ;
return 0 ;
}
static unsigned int intel_pstate_get ( unsigned int cpu_num )
{
struct sample * sample ;
struct cpudata * cpu ;
cpu = all_cpu_data [ cpu_num ] ;
if ( ! cpu )
return 0 ;
2014-02-12 22:01:04 +04:00
sample = & cpu - > sample ;
2013-02-06 21:02:13 +04:00
return sample - > freq ;
}
static int intel_pstate_set_policy ( struct cpufreq_policy * policy )
{
struct cpudata * cpu ;
cpu = all_cpu_data [ policy - > cpu ] ;
2013-03-06 02:15:26 +04:00
if ( ! policy - > cpuinfo . max_freq )
return - ENODEV ;
2013-02-06 21:02:13 +04:00
if ( policy - > policy = = CPUFREQ_POLICY_PERFORMANCE ) {
limits . min_perf_pct = 100 ;
limits . min_perf = int_tofp ( 1 ) ;
limits . max_perf_pct = 100 ;
limits . max_perf = int_tofp ( 1 ) ;
limits . no_turbo = 0 ;
2013-04-10 02:38:18 +04:00
return 0 ;
2013-02-06 21:02:13 +04:00
}
2013-04-10 02:38:18 +04:00
limits . min_perf_pct = ( policy - > min * 100 ) / policy - > cpuinfo . max_freq ;
limits . min_perf_pct = clamp_t ( int , limits . min_perf_pct , 0 , 100 ) ;
limits . min_perf = div_fp ( int_tofp ( limits . min_perf_pct ) , int_tofp ( 100 ) ) ;
2013-05-07 19:20:26 +04:00
limits . max_policy_pct = policy - > max * 100 / policy - > cpuinfo . max_freq ;
limits . max_policy_pct = clamp_t ( int , limits . max_policy_pct , 0 , 100 ) ;
limits . max_perf_pct = min ( limits . max_policy_pct , limits . max_sysfs_pct ) ;
2013-04-10 02:38:18 +04:00
limits . max_perf = div_fp ( int_tofp ( limits . max_perf_pct ) , int_tofp ( 100 ) ) ;
2013-02-06 21:02:13 +04:00
return 0 ;
}
static int intel_pstate_verify_policy ( struct cpufreq_policy * policy )
{
2013-10-02 12:43:19 +04:00
cpufreq_verify_within_cpu_limits ( policy ) ;
2013-02-06 21:02:13 +04:00
if ( ( policy - > policy ! = CPUFREQ_POLICY_POWERSAVE ) & &
( policy - > policy ! = CPUFREQ_POLICY_PERFORMANCE ) )
return - EINVAL ;
return 0 ;
}
2014-03-19 19:45:54 +04:00
static void intel_pstate_stop_cpu ( struct cpufreq_policy * policy )
2013-02-06 21:02:13 +04:00
{
2014-03-19 19:45:54 +04:00
int cpu_num = policy - > cpu ;
struct cpudata * cpu = all_cpu_data [ cpu_num ] ;
2013-02-06 21:02:13 +04:00
2014-03-19 19:45:54 +04:00
pr_info ( " intel_pstate CPU %d exiting \n " , cpu_num ) ;
2014-03-24 18:41:29 +04:00
del_timer_sync ( & all_cpu_data [ cpu_num ] - > timer ) ;
2014-03-19 19:45:54 +04:00
intel_pstate_set_pstate ( cpu , cpu - > pstate . min_pstate ) ;
kfree ( all_cpu_data [ cpu_num ] ) ;
all_cpu_data [ cpu_num ] = NULL ;
2013-02-06 21:02:13 +04:00
}
2013-06-19 21:54:04 +04:00
static int intel_pstate_cpu_init ( struct cpufreq_policy * policy )
2013-02-06 21:02:13 +04:00
{
struct cpudata * cpu ;
2013-10-15 22:06:14 +04:00
int rc ;
2013-02-06 21:02:13 +04:00
rc = intel_pstate_init_cpu ( policy - > cpu ) ;
if ( rc )
return rc ;
cpu = all_cpu_data [ policy - > cpu ] ;
if ( ! limits . no_turbo & &
limits . min_perf_pct = = 100 & & limits . max_perf_pct = = 100 )
policy - > policy = CPUFREQ_POLICY_PERFORMANCE ;
else
policy - > policy = CPUFREQ_POLICY_POWERSAVE ;
2013-10-15 22:06:14 +04:00
policy - > min = cpu - > pstate . min_pstate * 100000 ;
policy - > max = cpu - > pstate . turbo_pstate * 100000 ;
2013-02-06 21:02:13 +04:00
/* cpuinfo and default policy values */
policy - > cpuinfo . min_freq = cpu - > pstate . min_pstate * 100000 ;
policy - > cpuinfo . max_freq = cpu - > pstate . turbo_pstate * 100000 ;
policy - > cpuinfo . transition_latency = CPUFREQ_ETERNAL ;
cpumask_set_cpu ( policy - > cpu , policy - > cpus ) ;
return 0 ;
}
static struct cpufreq_driver intel_pstate_driver = {
. flags = CPUFREQ_CONST_LOOPS ,
. verify = intel_pstate_verify_policy ,
. setpolicy = intel_pstate_set_policy ,
. get = intel_pstate_get ,
. init = intel_pstate_cpu_init ,
2014-03-19 19:45:54 +04:00
. stop_cpu = intel_pstate_stop_cpu ,
2013-02-06 21:02:13 +04:00
. name = " intel_pstate " ,
} ;
2013-02-16 01:55:10 +04:00
static int __initdata no_load ;
2013-03-22 04:29:28 +04:00
static int intel_pstate_msrs_not_valid ( void )
{
/* Check that all the msr's we are using are valid. */
u64 aperf , mperf , tmp ;
rdmsrl ( MSR_IA32_APERF , aperf ) ;
rdmsrl ( MSR_IA32_MPERF , mperf ) ;
2013-10-21 20:20:34 +04:00
if ( ! pstate_funcs . get_max ( ) | |
! pstate_funcs . get_min ( ) | |
! pstate_funcs . get_turbo ( ) )
2013-03-22 04:29:28 +04:00
return - ENODEV ;
rdmsrl ( MSR_IA32_APERF , tmp ) ;
if ( ! ( tmp - aperf ) )
return - ENODEV ;
rdmsrl ( MSR_IA32_MPERF , tmp ) ;
if ( ! ( tmp - mperf ) )
return - ENODEV ;
return 0 ;
}
2013-10-21 20:20:34 +04:00
2013-10-30 19:38:32 +04:00
static void copy_pid_params ( struct pstate_adjust_policy * policy )
2013-10-21 20:20:34 +04:00
{
pid_params . sample_rate_ms = policy - > sample_rate_ms ;
pid_params . p_gain_pct = policy - > p_gain_pct ;
pid_params . i_gain_pct = policy - > i_gain_pct ;
pid_params . d_gain_pct = policy - > d_gain_pct ;
pid_params . deadband = policy - > deadband ;
pid_params . setpoint = policy - > setpoint ;
}
2013-10-30 19:38:32 +04:00
static void copy_cpu_funcs ( struct pstate_funcs * funcs )
2013-10-21 20:20:34 +04:00
{
pstate_funcs . get_max = funcs - > get_max ;
pstate_funcs . get_min = funcs - > get_min ;
pstate_funcs . get_turbo = funcs - > get_turbo ;
pstate_funcs . set = funcs - > set ;
2013-12-18 22:32:39 +04:00
pstate_funcs . get_vid = funcs - > get_vid ;
2013-10-21 20:20:34 +04:00
}
2013-10-31 19:24:05 +04:00
# if IS_ENABLED(CONFIG_ACPI)
# include <acpi/processor.h>
static bool intel_pstate_no_acpi_pss ( void )
{
int i ;
for_each_possible_cpu ( i ) {
acpi_status status ;
union acpi_object * pss ;
struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER , NULL } ;
struct acpi_processor * pr = per_cpu ( processors , i ) ;
if ( ! pr )
continue ;
status = acpi_evaluate_object ( pr - > handle , " _PSS " , NULL , & buffer ) ;
if ( ACPI_FAILURE ( status ) )
continue ;
pss = buffer . pointer ;
if ( pss & & pss - > type = = ACPI_TYPE_PACKAGE ) {
kfree ( pss ) ;
return false ;
}
kfree ( pss ) ;
}
return true ;
}
struct hw_vendor_info {
u16 valid ;
char oem_id [ ACPI_OEM_ID_SIZE ] ;
char oem_table_id [ ACPI_OEM_TABLE_ID_SIZE ] ;
} ;
/* Hardware vendor-specific info that has its own power management modes */
static struct hw_vendor_info vendor_info [ ] = {
{ 1 , " HP " , " ProLiant " } ,
{ 0 , " " , " " } ,
} ;
static bool intel_pstate_platform_pwr_mgmt_exists ( void )
{
struct acpi_table_header hdr ;
struct hw_vendor_info * v_info ;
if ( acpi_disabled
| | ACPI_FAILURE ( acpi_get_table_header ( ACPI_SIG_FADT , 0 , & hdr ) ) )
return false ;
for ( v_info = vendor_info ; v_info - > valid ; v_info + + ) {
if ( ! strncmp ( hdr . oem_id , v_info - > oem_id , ACPI_OEM_ID_SIZE )
& & ! strncmp ( hdr . oem_table_id , v_info - > oem_table_id , ACPI_OEM_TABLE_ID_SIZE )
& & intel_pstate_no_acpi_pss ( ) )
return true ;
}
return false ;
}
# else /* CONFIG_ACPI not enabled */
static inline bool intel_pstate_platform_pwr_mgmt_exists ( void ) { return false ; }
# endif /* CONFIG_ACPI */
2013-02-06 21:02:13 +04:00
static int __init intel_pstate_init ( void )
{
2013-03-06 02:15:27 +04:00
int cpu , rc = 0 ;
2013-02-06 21:02:13 +04:00
const struct x86_cpu_id * id ;
2013-10-21 20:20:34 +04:00
struct cpu_defaults * cpu_info ;
2013-02-06 21:02:13 +04:00
2013-02-16 01:55:10 +04:00
if ( no_load )
return - ENODEV ;
2013-02-06 21:02:13 +04:00
id = x86_match_cpu ( intel_pstate_cpu_ids ) ;
if ( ! id )
return - ENODEV ;
2013-10-31 19:24:05 +04:00
/*
* The Intel pstate driver will be ignored if the platform
* firmware has its own power management modes .
*/
if ( intel_pstate_platform_pwr_mgmt_exists ( ) )
return - ENODEV ;
2013-10-21 20:20:34 +04:00
cpu_info = ( struct cpu_defaults * ) id - > driver_data ;
copy_pid_params ( & cpu_info - > pid_policy ) ;
copy_cpu_funcs ( & cpu_info - > funcs ) ;
2013-03-22 04:29:28 +04:00
if ( intel_pstate_msrs_not_valid ( ) )
return - ENODEV ;
2013-02-06 21:02:13 +04:00
pr_info ( " Intel P-state driver initializing. \n " ) ;
2013-05-13 12:03:43 +04:00
all_cpu_data = vzalloc ( sizeof ( void * ) * num_possible_cpus ( ) ) ;
2013-02-06 21:02:13 +04:00
if ( ! all_cpu_data )
return - ENOMEM ;
rc = cpufreq_register_driver ( & intel_pstate_driver ) ;
if ( rc )
goto out ;
intel_pstate_debug_expose_params ( ) ;
intel_pstate_sysfs_expose_params ( ) ;
2014-01-16 22:32:25 +04:00
2013-02-06 21:02:13 +04:00
return rc ;
out :
2013-03-06 02:15:27 +04:00
get_online_cpus ( ) ;
for_each_online_cpu ( cpu ) {
if ( all_cpu_data [ cpu ] ) {
del_timer_sync ( & all_cpu_data [ cpu ] - > timer ) ;
kfree ( all_cpu_data [ cpu ] ) ;
}
}
put_online_cpus ( ) ;
vfree ( all_cpu_data ) ;
2013-02-06 21:02:13 +04:00
return - ENODEV ;
}
device_initcall ( intel_pstate_init ) ;
2013-02-16 01:55:10 +04:00
static int __init intel_pstate_setup ( char * str )
{
if ( ! str )
return - EINVAL ;
if ( ! strcmp ( str , " disable " ) )
no_load = 1 ;
return 0 ;
}
early_param ( " intel_pstate " , intel_pstate_setup ) ;
2013-02-06 21:02:13 +04:00
MODULE_AUTHOR ( " Dirk Brandewie <dirk.j.brandewie@intel.com> " ) ;
MODULE_DESCRIPTION ( " 'intel_pstate' - P state driver Intel Core processors " ) ;
MODULE_LICENSE ( " GPL " ) ;