2005-04-17 02:20:36 +04:00
/*
* cpufreq driver for Enhanced SpeedStep , as found in Intel ' s Pentium
* M ( part of the Centrino chipset ) .
*
2006-06-22 00:15:48 +04:00
* Since the original Pentium M , most new Intel CPUs support Enhanced
* SpeedStep .
*
2005-04-17 02:20:36 +04:00
* Despite the " SpeedStep " in the name , this is almost entirely unlike
* traditional SpeedStep .
*
* Modelled on speedstep . c
*
* Copyright ( C ) 2003 Jeremy Fitzhardinge < jeremy @ goop . org >
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/cpufreq.h>
2005-10-31 02:03:48 +03:00
# include <linux/sched.h> /* current */
2005-04-17 02:20:36 +04:00
# include <linux/delay.h>
# include <linux/compiler.h>
# include <asm/msr.h>
# include <asm/processor.h>
# include <asm/cpufeature.h>
# define PFX "speedstep-centrino: "
2008-10-09 16:59:17 +04:00
# define MAINTAINER "cpufreq@vger.kernel.org"
2005-04-17 02:20:36 +04:00
2008-07-19 05:11:34 +04:00
# define dprintk(msg...) \
cpufreq_debug_printk ( CPUFREQ_DEBUG_DRIVER , " speedstep-centrino " , msg )
2005-04-17 02:20:36 +04:00
2006-11-10 22:20:47 +03:00
# define INTEL_MSR_RANGE (0xffff)
2005-04-17 02:20:36 +04:00
struct cpu_id
{
__u8 x86 ; /* CPU family */
__u8 x86_model ; /* model */
__u8 x86_mask ; /* stepping */
} ;
enum {
CPU_BANIAS ,
CPU_DOTHAN_A1 ,
CPU_DOTHAN_A2 ,
CPU_DOTHAN_B0 ,
2005-06-01 06:03:43 +04:00
CPU_MP4HT_D0 ,
CPU_MP4HT_E0 ,
2005-04-17 02:20:36 +04:00
} ;
static const struct cpu_id cpu_ids [ ] = {
[ CPU_BANIAS ] = { 6 , 9 , 5 } ,
[ CPU_DOTHAN_A1 ] = { 6 , 13 , 1 } ,
[ CPU_DOTHAN_A2 ] = { 6 , 13 , 2 } ,
[ CPU_DOTHAN_B0 ] = { 6 , 13 , 6 } ,
2005-06-01 06:03:43 +04:00
[ CPU_MP4HT_D0 ] = { 15 , 3 , 4 } ,
[ CPU_MP4HT_E0 ] = { 15 , 4 , 1 } ,
2005-04-17 02:20:36 +04:00
} ;
2005-11-07 11:58:31 +03:00
# define N_IDS ARRAY_SIZE(cpu_ids)
2005-04-17 02:20:36 +04:00
struct cpu_model
{
const struct cpu_id * cpu_id ;
const char * model_name ;
unsigned max_freq ; /* max clock in kHz */
struct cpufreq_frequency_table * op_points ; /* clock/voltage pairs */
} ;
2008-07-19 05:11:34 +04:00
static int centrino_verify_cpu_id ( const struct cpuinfo_x86 * c ,
const struct cpu_id * x ) ;
2005-04-17 02:20:36 +04:00
/* Operating points for current CPU */
2008-07-19 05:11:34 +04:00
static DEFINE_PER_CPU ( struct cpu_model * , centrino_model ) ;
static DEFINE_PER_CPU ( const struct cpu_id * , centrino_cpu ) ;
2005-04-17 02:20:36 +04:00
static struct cpufreq_driver centrino_driver ;
# ifdef CONFIG_X86_SPEEDSTEP_CENTRINO_TABLE
/* Computes the correct form for IA32_PERF_CTL MSR for a particular
frequency / voltage operating point ; frequency in MHz , volts in mV .
This is stored as " index " in the structure . */
# define OP(mhz, mv) \
{ \
. frequency = ( mhz ) * 1000 , \
. index = ( ( ( mhz ) / 100 ) < < 8 ) | ( ( mv - 700 ) / 16 ) \
}
/*
* These voltage tables were derived from the Intel Pentium M
* datasheet , document 25261202. pdf , Table 5. I have verified they
* are consistent with my IBM ThinkPad X31 , which has a 1.3 GHz Pentium
* M .
*/
/* Ultra Low Voltage Intel Pentium M processor 900MHz (Banias) */
static struct cpufreq_frequency_table banias_900 [ ] =
{
OP ( 600 , 844 ) ,
OP ( 800 , 988 ) ,
OP ( 900 , 1004 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Ultra Low Voltage Intel Pentium M processor 1000MHz (Banias) */
static struct cpufreq_frequency_table banias_1000 [ ] =
{
OP ( 600 , 844 ) ,
OP ( 800 , 972 ) ,
OP ( 900 , 988 ) ,
OP ( 1000 , 1004 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Low Voltage Intel Pentium M processor 1.10GHz (Banias) */
static struct cpufreq_frequency_table banias_1100 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1020 ) ,
OP ( 900 , 1100 ) ,
OP ( 1000 , 1164 ) ,
OP ( 1100 , 1180 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Low Voltage Intel Pentium M processor 1.20GHz (Banias) */
static struct cpufreq_frequency_table banias_1200 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1004 ) ,
OP ( 900 , 1020 ) ,
OP ( 1000 , 1100 ) ,
OP ( 1100 , 1164 ) ,
OP ( 1200 , 1180 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Intel Pentium M processor 1.30GHz (Banias) */
static struct cpufreq_frequency_table banias_1300 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1260 ) ,
OP ( 1000 , 1292 ) ,
OP ( 1200 , 1356 ) ,
OP ( 1300 , 1388 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Intel Pentium M processor 1.40GHz (Banias) */
static struct cpufreq_frequency_table banias_1400 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1180 ) ,
OP ( 1000 , 1308 ) ,
OP ( 1200 , 1436 ) ,
OP ( 1400 , 1484 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Intel Pentium M processor 1.50GHz (Banias) */
static struct cpufreq_frequency_table banias_1500 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1116 ) ,
OP ( 1000 , 1228 ) ,
OP ( 1200 , 1356 ) ,
OP ( 1400 , 1452 ) ,
OP ( 1500 , 1484 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Intel Pentium M processor 1.60GHz (Banias) */
static struct cpufreq_frequency_table banias_1600 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1036 ) ,
OP ( 1000 , 1164 ) ,
OP ( 1200 , 1276 ) ,
OP ( 1400 , 1420 ) ,
OP ( 1600 , 1484 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
/* Intel Pentium M processor 1.70GHz (Banias) */
static struct cpufreq_frequency_table banias_1700 [ ] =
{
OP ( 600 , 956 ) ,
OP ( 800 , 1004 ) ,
OP ( 1000 , 1116 ) ,
OP ( 1200 , 1228 ) ,
OP ( 1400 , 1308 ) ,
OP ( 1700 , 1484 ) ,
{ . frequency = CPUFREQ_TABLE_END }
} ;
# undef OP
# define _BANIAS(cpuid, max, name) \
{ . cpu_id = cpuid , \
. model_name = " Intel(R) Pentium(R) M processor " name " MHz " , \
. max_freq = ( max ) * 1000 , \
. op_points = banias_ # # max , \
}
# define BANIAS(max) _BANIAS(&cpu_ids[CPU_BANIAS], max, #max)
/* CPU models, their operating frequency range, and freq/voltage
operating points */
static struct cpu_model models [ ] =
{
_BANIAS ( & cpu_ids [ CPU_BANIAS ] , 900 , " 900 " ) ,
BANIAS ( 1000 ) ,
BANIAS ( 1100 ) ,
BANIAS ( 1200 ) ,
BANIAS ( 1300 ) ,
BANIAS ( 1400 ) ,
BANIAS ( 1500 ) ,
BANIAS ( 1600 ) ,
BANIAS ( 1700 ) ,
/* NULL model_name is a wildcard */
{ & cpu_ids [ CPU_DOTHAN_A1 ] , NULL , 0 , NULL } ,
{ & cpu_ids [ CPU_DOTHAN_A2 ] , NULL , 0 , NULL } ,
{ & cpu_ids [ CPU_DOTHAN_B0 ] , NULL , 0 , NULL } ,
2005-06-01 06:03:43 +04:00
{ & cpu_ids [ CPU_MP4HT_D0 ] , NULL , 0 , NULL } ,
{ & cpu_ids [ CPU_MP4HT_E0 ] , NULL , 0 , NULL } ,
2005-04-17 02:20:36 +04:00
{ NULL , }
} ;
# undef _BANIAS
# undef BANIAS
static int centrino_cpu_init_table ( struct cpufreq_policy * policy )
{
2007-10-19 22:35:04 +04:00
struct cpuinfo_x86 * cpu = & cpu_data ( policy - > cpu ) ;
2005-04-17 02:20:36 +04:00
struct cpu_model * model ;
for ( model = models ; model - > cpu_id ! = NULL ; model + + )
if ( centrino_verify_cpu_id ( cpu , model - > cpu_id ) & &
( model - > model_name = = NULL | |
strcmp ( cpu - > x86_model_id , model - > model_name ) = = 0 ) )
break ;
if ( model - > cpu_id = = NULL ) {
/* No match at all */
2006-04-26 17:41:22 +04:00
dprintk ( " no support for CPU model \" %s \" : "
2005-04-17 02:20:36 +04:00
" send /proc/cpuinfo to " MAINTAINER " \n " ,
cpu - > x86_model_id ) ;
return - ENOENT ;
}
if ( model - > op_points = = NULL ) {
/* Matched a non-match */
2006-04-26 17:41:22 +04:00
dprintk ( " no table support for CPU model \" %s \" \n " ,
2005-04-17 02:20:36 +04:00
cpu - > x86_model_id ) ;
2007-07-09 01:39:14 +04:00
dprintk ( " try using the acpi-cpufreq driver \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENOENT ;
}
2008-07-19 05:11:34 +04:00
per_cpu ( centrino_model , policy - > cpu ) = model ;
2005-04-17 02:20:36 +04:00
dprintk ( " found \" %s \" : max frequency: %dkHz \n " ,
model - > model_name , model - > max_freq ) ;
return 0 ;
}
# else
2008-07-19 05:11:34 +04:00
static inline int centrino_cpu_init_table ( struct cpufreq_policy * policy )
{
return - ENODEV ;
}
2005-04-17 02:20:36 +04:00
# endif /* CONFIG_X86_SPEEDSTEP_CENTRINO_TABLE */
2008-07-19 05:11:34 +04:00
static int centrino_verify_cpu_id ( const struct cpuinfo_x86 * c ,
const struct cpu_id * x )
2005-04-17 02:20:36 +04:00
{
if ( ( c - > x86 = = x - > x86 ) & &
( c - > x86_model = = x - > x86_model ) & &
( c - > x86_mask = = x - > x86_mask ) )
return 1 ;
return 0 ;
}
/* To be called only after centrino_model is initialized */
static unsigned extract_clock ( unsigned msr , unsigned int cpu , int failsafe )
{
int i ;
/*
* Extract clock in kHz from PERF_CTL value
* for centrino , as some DSDTs are buggy .
* Ideally , this can be done using the acpi_data structure .
*/
2008-07-19 05:11:34 +04:00
if ( ( per_cpu ( centrino_cpu , cpu ) = = & cpu_ids [ CPU_BANIAS ] ) | |
( per_cpu ( centrino_cpu , cpu ) = = & cpu_ids [ CPU_DOTHAN_A1 ] ) | |
( per_cpu ( centrino_cpu , cpu ) = = & cpu_ids [ CPU_DOTHAN_B0 ] ) ) {
2005-04-17 02:20:36 +04:00
msr = ( msr > > 8 ) & 0xff ;
return msr * 100000 ;
}
2008-07-19 05:11:34 +04:00
if ( ( ! per_cpu ( centrino_model , cpu ) ) | |
( ! per_cpu ( centrino_model , cpu ) - > op_points ) )
2005-04-17 02:20:36 +04:00
return 0 ;
msr & = 0xffff ;
2008-07-19 05:11:34 +04:00
for ( i = 0 ;
per_cpu ( centrino_model , cpu ) - > op_points [ i ] . frequency
! = CPUFREQ_TABLE_END ;
i + + ) {
if ( msr = = per_cpu ( centrino_model , cpu ) - > op_points [ i ] . index )
return per_cpu ( centrino_model , cpu ) - >
op_points [ i ] . frequency ;
2005-04-17 02:20:36 +04:00
}
if ( failsafe )
2008-07-19 05:11:34 +04:00
return per_cpu ( centrino_model , cpu ) - > op_points [ i - 1 ] . frequency ;
2005-04-17 02:20:36 +04:00
else
return 0 ;
}
/* Return the current CPU frequency in kHz */
static unsigned int get_cur_freq ( unsigned int cpu )
{
unsigned l , h ;
unsigned clock_freq ;
cpumask_t saved_mask ;
saved_mask = current - > cpus_allowed ;
2008-07-25 05:21:31 +04:00
set_cpus_allowed_ptr ( current , & cpumask_of_cpu ( cpu ) ) ;
2005-04-17 02:20:36 +04:00
if ( smp_processor_id ( ) ! = cpu )
return 0 ;
rdmsr ( MSR_IA32_PERF_STATUS , l , h ) ;
clock_freq = extract_clock ( l , cpu , 0 ) ;
if ( unlikely ( clock_freq = = 0 ) ) {
/*
* On some CPUs , we can see transient MSR values ( which are
* not present in _PSS ) , while CPU is doing some automatic
* P - state transition ( like TM2 ) . Get the last freq set
* in PERF_CTL .
*/
rdmsr ( MSR_IA32_PERF_CTL , l , h ) ;
clock_freq = extract_clock ( l , cpu , 1 ) ;
}
2008-04-05 05:11:05 +04:00
set_cpus_allowed_ptr ( current , & saved_mask ) ;
2005-04-17 02:20:36 +04:00
return clock_freq ;
}
static int centrino_cpu_init ( struct cpufreq_policy * policy )
{
2007-10-19 22:35:04 +04:00
struct cpuinfo_x86 * cpu = & cpu_data ( policy - > cpu ) ;
2005-04-17 02:20:36 +04:00
unsigned freq ;
unsigned l , h ;
int ret ;
int i ;
/* Only Intel makes Enhanced Speedstep-capable CPUs */
2008-07-19 05:11:34 +04:00
if ( cpu - > x86_vendor ! = X86_VENDOR_INTEL | |
! cpu_has ( cpu , X86_FEATURE_EST ) )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
2006-02-28 08:37:44 +03:00
if ( cpu_has ( cpu , X86_FEATURE_CONSTANT_TSC ) )
2005-04-17 02:20:36 +04:00
centrino_driver . flags | = CPUFREQ_CONST_LOOPS ;
2007-07-09 01:39:14 +04:00
if ( policy - > cpu ! = 0 )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2007-07-09 01:39:14 +04:00
for ( i = 0 ; i < N_IDS ; i + + )
if ( centrino_verify_cpu_id ( cpu , & cpu_ids [ i ] ) )
break ;
2005-08-30 00:54:55 +04:00
2007-07-09 01:39:14 +04:00
if ( i ! = N_IDS )
2008-07-19 05:11:34 +04:00
per_cpu ( centrino_cpu , policy - > cpu ) = & cpu_ids [ i ] ;
2005-08-30 00:54:55 +04:00
2008-07-19 05:11:34 +04:00
if ( ! per_cpu ( centrino_cpu , policy - > cpu ) ) {
2007-07-09 01:39:14 +04:00
dprintk ( " found unsupported CPU with "
" Enhanced SpeedStep: send /proc/cpuinfo to "
MAINTAINER " \n " ) ;
return - ENODEV ;
}
2005-04-17 02:20:36 +04:00
2007-07-09 01:39:14 +04:00
if ( centrino_cpu_init_table ( policy ) ) {
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
/* Check to see if Enhanced SpeedStep is enabled, and try to
enable it if not . */
rdmsr ( MSR_IA32_MISC_ENABLE , l , h ) ;
2009-02-20 13:56:38 +03:00
if ( ! ( l & MSR_IA32_MISC_ENABLE_ENHANCED_SPEEDSTEP ) ) {
l | = MSR_IA32_MISC_ENABLE_ENHANCED_SPEEDSTEP ;
2005-04-17 02:20:36 +04:00
dprintk ( " trying to enable Enhanced SpeedStep (%x) \n " , l ) ;
wrmsr ( MSR_IA32_MISC_ENABLE , l , h ) ;
/* check to see if it stuck */
rdmsr ( MSR_IA32_MISC_ENABLE , l , h ) ;
2009-02-20 13:56:38 +03:00
if ( ! ( l & MSR_IA32_MISC_ENABLE_ENHANCED_SPEEDSTEP ) ) {
2008-07-19 05:11:34 +04:00
printk ( KERN_INFO PFX
" couldn't enable Enhanced SpeedStep \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
}
freq = get_cur_freq ( policy - > cpu ) ;
2008-07-19 05:11:34 +04:00
policy - > cpuinfo . transition_latency = 10000 ;
/* 10uS transition latency */
2005-04-17 02:20:36 +04:00
policy - > cur = freq ;
dprintk ( " centrino_cpu_init: cur=%dkHz \n " , policy - > cur ) ;
2008-07-19 05:11:34 +04:00
ret = cpufreq_frequency_table_cpuinfo ( policy ,
per_cpu ( centrino_model , policy - > cpu ) - > op_points ) ;
2005-04-17 02:20:36 +04:00
if ( ret )
return ( ret ) ;
2008-07-19 05:11:34 +04:00
cpufreq_frequency_table_get_attr (
per_cpu ( centrino_model , policy - > cpu ) - > op_points , policy - > cpu ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int centrino_cpu_exit ( struct cpufreq_policy * policy )
{
unsigned int cpu = policy - > cpu ;
2008-07-19 05:11:34 +04:00
if ( ! per_cpu ( centrino_model , cpu ) )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
cpufreq_frequency_table_put_attr ( cpu ) ;
2008-07-19 05:11:34 +04:00
per_cpu ( centrino_model , cpu ) = NULL ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/**
* centrino_verify - verifies a new CPUFreq policy
* @ policy : new policy
*
* Limit must be within this model ' s frequency range at least one
* border included .
*/
static int centrino_verify ( struct cpufreq_policy * policy )
{
2008-07-19 05:11:34 +04:00
return cpufreq_frequency_table_verify ( policy ,
per_cpu ( centrino_model , policy - > cpu ) - > op_points ) ;
2005-04-17 02:20:36 +04:00
}
/**
* centrino_setpolicy - set a new CPUFreq policy
* @ policy : new policy
* @ target_freq : the target frequency
2008-07-19 05:11:34 +04:00
* @ relation : how that frequency relates to achieved frequency
* ( CPUFREQ_RELATION_L or CPUFREQ_RELATION_H )
2005-04-17 02:20:36 +04:00
*
* Sets a new CPUFreq policy .
*/
static int centrino_target ( struct cpufreq_policy * policy ,
unsigned int target_freq ,
unsigned int relation )
{
unsigned int newstate = 0 ;
2005-12-14 23:05:00 +03:00
unsigned int msr , oldmsr = 0 , h = 0 , cpu = policy - > cpu ;
2005-04-17 02:20:36 +04:00
struct cpufreq_freqs freqs ;
2005-12-14 23:05:00 +03:00
int retval = 0 ;
unsigned int j , k , first_cpu , tmp ;
2009-01-04 16:18:05 +03:00
cpumask_var_t saved_mask , covered_cpus ;
2008-07-16 01:14:37 +04:00
2009-01-04 16:18:05 +03:00
if ( unlikely ( ! alloc_cpumask_var ( & saved_mask , GFP_KERNEL ) ) )
2008-07-16 01:14:37 +04:00
return - ENOMEM ;
2009-06-07 01:51:36 +04:00
if ( unlikely ( ! zalloc_cpumask_var ( & covered_cpus , GFP_KERNEL ) ) ) {
2009-01-04 16:18:05 +03:00
free_cpumask_var ( saved_mask ) ;
return - ENOMEM ;
}
cpumask_copy ( saved_mask , & current - > cpus_allowed ) ;
2008-07-16 01:14:37 +04:00
2008-07-19 05:11:34 +04:00
if ( unlikely ( per_cpu ( centrino_model , cpu ) = = NULL ) ) {
2008-07-16 01:14:37 +04:00
retval = - ENODEV ;
goto out ;
}
2005-04-17 02:20:36 +04:00
2005-12-14 23:05:00 +03:00
if ( unlikely ( cpufreq_frequency_table_target ( policy ,
2008-07-19 05:11:34 +04:00
per_cpu ( centrino_model , cpu ) - > op_points ,
2005-12-14 23:05:00 +03:00
target_freq ,
relation ,
& newstate ) ) ) {
2008-07-16 01:14:37 +04:00
retval = - EINVAL ;
goto out ;
2005-04-17 02:20:36 +04:00
}
2005-12-14 23:05:00 +03:00
first_cpu = 1 ;
2009-01-04 16:18:06 +03:00
for_each_cpu ( j , policy - > cpus ) {
const struct cpumask * mask ;
2008-11-21 13:37:16 +03:00
/* cpufreq holds the hotplug lock, so we are safe here */
if ( ! cpu_online ( j ) )
continue ;
2005-12-14 23:05:00 +03:00
/*
* Support for SMP systems .
* Make sure we are running on CPU that wants to change freq
*/
if ( policy - > shared_type = = CPUFREQ_SHARED_TYPE_ANY )
2009-01-04 16:18:06 +03:00
mask = policy - > cpus ;
2005-12-14 23:05:00 +03:00
else
2009-01-04 16:18:06 +03:00
mask = cpumask_of ( j ) ;
2005-12-14 23:05:00 +03:00
2008-11-21 13:37:16 +03:00
set_cpus_allowed_ptr ( current , mask ) ;
2007-04-12 02:22:34 +04:00
preempt_disable ( ) ;
2008-11-21 13:37:16 +03:00
if ( unlikely ( ! cpu_isset ( smp_processor_id ( ) , * mask ) ) ) {
2005-12-14 23:05:00 +03:00
dprintk ( " couldn't limit to CPUs in this domain \n " ) ;
retval = - EAGAIN ;
if ( first_cpu ) {
/* We haven't started the transition yet. */
goto migrate_end ;
}
2007-04-12 02:22:34 +04:00
preempt_enable ( ) ;
2005-12-14 23:05:00 +03:00
break ;
}
2005-04-17 02:20:36 +04:00
2008-07-19 05:11:34 +04:00
msr = per_cpu ( centrino_model , cpu ) - > op_points [ newstate ] . index ;
2005-12-14 23:05:00 +03:00
if ( first_cpu ) {
rdmsr ( MSR_IA32_PERF_CTL , oldmsr , h ) ;
if ( msr = = ( oldmsr & 0xffff ) ) {
dprintk ( " no change needed - msr was and needs "
" to be %x \n " , oldmsr ) ;
retval = 0 ;
goto migrate_end ;
}
freqs . old = extract_clock ( oldmsr , cpu , 0 ) ;
freqs . new = extract_clock ( msr , cpu , 0 ) ;
dprintk ( " target=%dkHz old=%d new=%d msr=%04x \n " ,
target_freq , freqs . old , freqs . new , msr ) ;
2009-01-04 16:18:06 +03:00
for_each_cpu ( k , policy - > cpus ) {
2008-11-21 13:37:16 +03:00
if ( ! cpu_online ( k ) )
continue ;
2005-12-14 23:05:00 +03:00
freqs . cpu = k ;
cpufreq_notify_transition ( & freqs ,
CPUFREQ_PRECHANGE ) ;
}
first_cpu = 0 ;
/* all but 16 LSB are reserved, treat them with care */
oldmsr & = ~ 0xffff ;
msr & = 0xffff ;
oldmsr | = msr ;
}
2005-04-17 02:20:36 +04:00
2005-12-14 23:05:00 +03:00
wrmsr ( MSR_IA32_PERF_CTL , oldmsr , h ) ;
2007-04-12 02:22:34 +04:00
if ( policy - > shared_type = = CPUFREQ_SHARED_TYPE_ANY ) {
preempt_enable ( ) ;
2005-12-14 23:05:00 +03:00
break ;
2007-04-12 02:22:34 +04:00
}
2005-04-17 02:20:36 +04:00
2008-07-16 01:14:37 +04:00
cpu_set ( j , * covered_cpus ) ;
2007-04-12 02:22:34 +04:00
preempt_enable ( ) ;
2005-12-14 23:05:00 +03:00
}
2005-04-17 02:20:36 +04:00
2009-01-04 16:18:06 +03:00
for_each_cpu ( k , policy - > cpus ) {
2008-11-21 13:37:16 +03:00
if ( ! cpu_online ( k ) )
continue ;
2005-12-14 23:05:00 +03:00
freqs . cpu = k ;
cpufreq_notify_transition ( & freqs , CPUFREQ_POSTCHANGE ) ;
}
2005-04-17 02:20:36 +04:00
2005-12-14 23:05:00 +03:00
if ( unlikely ( retval ) ) {
/*
* We have failed halfway through the frequency change .
* We have sent callbacks to policy - > cpus and
* MSRs have already been written on coverd_cpus .
* Best effort undo . .
*/
2005-04-17 02:20:36 +04:00
2008-11-21 13:37:16 +03:00
for_each_cpu_mask_nr ( j , * covered_cpus ) {
set_cpus_allowed_ptr ( current , & cpumask_of_cpu ( j ) ) ;
wrmsr ( MSR_IA32_PERF_CTL , oldmsr , h ) ;
}
2005-04-17 02:20:36 +04:00
2005-12-14 23:05:00 +03:00
tmp = freqs . new ;
freqs . new = freqs . old ;
freqs . old = tmp ;
2009-01-04 16:18:06 +03:00
for_each_cpu ( j , policy - > cpus ) {
2008-11-21 13:37:16 +03:00
if ( ! cpu_online ( j ) )
continue ;
2005-12-14 23:05:00 +03:00
cpufreq_notify_transition ( & freqs , CPUFREQ_PRECHANGE ) ;
cpufreq_notify_transition ( & freqs , CPUFREQ_POSTCHANGE ) ;
}
}
2008-07-16 01:14:37 +04:00
set_cpus_allowed_ptr ( current , saved_mask ) ;
retval = 0 ;
goto out ;
2005-04-17 02:20:36 +04:00
migrate_end :
2007-04-12 02:22:34 +04:00
preempt_enable ( ) ;
2008-07-16 01:14:37 +04:00
set_cpus_allowed_ptr ( current , saved_mask ) ;
out :
2009-01-04 16:18:05 +03:00
free_cpumask_var ( saved_mask ) ;
free_cpumask_var ( covered_cpus ) ;
2008-07-16 01:14:37 +04:00
return retval ;
2005-04-17 02:20:36 +04:00
}
static struct freq_attr * centrino_attr [ ] = {
& cpufreq_freq_attr_scaling_available_freqs ,
NULL ,
} ;
static struct cpufreq_driver centrino_driver = {
. name = " centrino " , /* should be speedstep-centrino,
but there ' s a 16 char limit */
. init = centrino_cpu_init ,
. exit = centrino_cpu_exit ,
. verify = centrino_verify ,
. target = centrino_target ,
. get = get_cur_freq ,
. attr = centrino_attr ,
. owner = THIS_MODULE ,
} ;
/**
* centrino_init - initializes the Enhanced SpeedStep CPUFreq driver
*
* Initializes the Enhanced SpeedStep support . Returns - ENODEV on
* unsupported devices , - ENOENT if there ' s no voltage table for this
* particular CPU model , - EINVAL on problems during initiatization ,
* and zero on success .
*
* This is quite picky . Not only does the CPU have to advertise the
* " est " flag in the cpuid capability flags , we look for a specific
* CPU model and stepping , and we need to have the exact model name in
* our voltage tables . That is , be paranoid about not releasing
* someone ' s valuable magic smoke .
*/
static int __init centrino_init ( void )
{
2007-10-19 22:35:04 +04:00
struct cpuinfo_x86 * cpu = & cpu_data ( 0 ) ;
2005-04-17 02:20:36 +04:00
if ( ! cpu_has ( cpu , X86_FEATURE_EST ) )
return - ENODEV ;
return cpufreq_register_driver ( & centrino_driver ) ;
}
static void __exit centrino_exit ( void )
{
cpufreq_unregister_driver ( & centrino_driver ) ;
}
MODULE_AUTHOR ( " Jeremy Fitzhardinge <jeremy@goop.org> " ) ;
MODULE_DESCRIPTION ( " Enhanced SpeedStep driver for Intel Pentium M processors. " ) ;
MODULE_LICENSE ( " GPL " ) ;
late_initcall ( centrino_init ) ;
module_exit ( centrino_exit ) ;