2005-04-16 15:20:36 -07:00
/*
* linux / drivers / cpufreq / freq_table . c
*
* Copyright ( C ) 2002 - 2003 Dominik Brodowski
2008-05-22 08:52:05 +02:00
*
* 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 .
*
2005-04-16 15:20:36 -07:00
*/
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-08-06 22:53:03 +05:30
# include <linux/module.h>
2005-04-16 15:20:36 -07:00
/*********************************************************************
* FREQUENCY TABLE HELPERS *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int cpufreq_frequency_table_cpuinfo ( struct cpufreq_policy * policy ,
struct cpufreq_frequency_table * table )
{
unsigned int min_freq = ~ 0 ;
unsigned int max_freq = 0 ;
2006-05-30 17:58:41 -04:00
unsigned int i ;
2005-04-16 15:20:36 -07:00
2009-01-18 01:56:41 -05:00
for ( i = 0 ; ( table [ i ] . frequency ! = CPUFREQ_TABLE_END ) ; i + + ) {
2005-04-16 15:20:36 -07:00
unsigned int freq = table [ i ] . frequency ;
if ( freq = = CPUFREQ_ENTRY_INVALID ) {
2011-03-27 15:04:46 +02:00
pr_debug ( " table entry %u is invalid, skipping \n " , i ) ;
2005-04-16 15:20:36 -07:00
continue ;
}
2013-12-20 15:24:49 +01:00
if ( ! cpufreq_boost_enabled ( )
2014-03-28 19:11:47 +05:30
& & ( table [ i ] . flags & CPUFREQ_BOOST_FREQ ) )
2013-12-20 15:24:49 +01:00
continue ;
2014-03-28 19:11:45 +05:30
pr_debug ( " table entry %u: %u kHz \n " , i , freq ) ;
2005-04-16 15:20:36 -07:00
if ( freq < min_freq )
min_freq = freq ;
if ( freq > max_freq )
max_freq = freq ;
}
policy - > min = policy - > cpuinfo . min_freq = min_freq ;
policy - > max = policy - > cpuinfo . max_freq = max_freq ;
if ( policy - > min = = ~ 0 )
return - EINVAL ;
else
return 0 ;
}
EXPORT_SYMBOL_GPL ( cpufreq_frequency_table_cpuinfo ) ;
int cpufreq_frequency_table_verify ( struct cpufreq_policy * policy ,
struct cpufreq_frequency_table * table )
{
2013-10-02 14:13:15 +05:30
unsigned int next_larger = ~ 0 , freq , i = 0 ;
bool found = false ;
2005-04-16 15:20:36 -07:00
2011-03-27 15:04:46 +02:00
pr_debug ( " request for verification of policy (%u - %u kHz) for cpu %u \n " ,
2006-10-26 16:20:58 +05:30
policy - > min , policy - > max , policy - > cpu ) ;
2005-04-16 15:20:36 -07:00
2013-10-02 14:13:19 +05:30
cpufreq_verify_within_cpu_limits ( policy ) ;
2005-04-16 15:20:36 -07:00
2013-10-02 14:13:15 +05:30
for ( ; freq = table [ i ] . frequency , freq ! = CPUFREQ_TABLE_END ; i + + ) {
2005-04-16 15:20:36 -07:00
if ( freq = = CPUFREQ_ENTRY_INVALID )
continue ;
2013-10-02 14:13:15 +05:30
if ( ( freq > = policy - > min ) & & ( freq < = policy - > max ) ) {
found = true ;
break ;
}
if ( ( next_larger > freq ) & & ( freq > policy - > max ) )
2005-04-16 15:20:36 -07:00
next_larger = freq ;
}
2013-10-02 14:13:15 +05:30
if ( ! found ) {
2005-04-16 15:20:36 -07:00
policy - > max = next_larger ;
2013-10-02 14:13:19 +05:30
cpufreq_verify_within_cpu_limits ( policy ) ;
2013-10-02 14:13:15 +05:30
}
2005-04-16 15:20:36 -07:00
2011-03-27 15:04:46 +02:00
pr_debug ( " verification lead to (%u - %u kHz) for cpu %u \n " ,
2006-10-26 16:20:58 +05:30
policy - > min , policy - > max , policy - > cpu ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( cpufreq_frequency_table_verify ) ;
2013-10-03 20:27:55 +05:30
/*
2014-03-10 14:53:33 +05:30
* Generic routine to verify policy & frequency table , requires driver to set
* policy - > freq_table prior to it .
2013-10-03 20:27:55 +05:30
*/
int cpufreq_generic_frequency_table_verify ( struct cpufreq_policy * policy )
{
struct cpufreq_frequency_table * table =
cpufreq_frequency_get_table ( policy - > cpu ) ;
if ( ! table )
return - ENODEV ;
return cpufreq_frequency_table_verify ( policy , table ) ;
}
EXPORT_SYMBOL_GPL ( cpufreq_generic_frequency_table_verify ) ;
2005-04-16 15:20:36 -07:00
int cpufreq_frequency_table_target ( struct cpufreq_policy * policy ,
struct cpufreq_frequency_table * table ,
unsigned int target_freq ,
unsigned int relation ,
unsigned int * index )
{
2006-05-30 18:09:31 -04:00
struct cpufreq_frequency_table optimal = {
2013-03-30 16:25:15 +05:30
. driver_data = ~ 0 ,
2006-05-30 18:09:31 -04:00
. frequency = 0 ,
} ;
struct cpufreq_frequency_table suboptimal = {
2013-03-30 16:25:15 +05:30
. driver_data = ~ 0 ,
2006-05-30 18:09:31 -04:00
. frequency = 0 ,
} ;
2005-04-16 15:20:36 -07:00
unsigned int i ;
2011-03-27 15:04:46 +02:00
pr_debug ( " request for target %u kHz (relation: %u) for cpu %u \n " ,
2006-10-26 16:20:58 +05:30
target_freq , relation , policy - > cpu ) ;
2005-04-16 15:20:36 -07:00
switch ( relation ) {
case CPUFREQ_RELATION_H :
suboptimal . frequency = ~ 0 ;
break ;
case CPUFREQ_RELATION_L :
optimal . frequency = ~ 0 ;
break ;
}
2009-01-18 01:56:41 -05:00
for ( i = 0 ; ( table [ i ] . frequency ! = CPUFREQ_TABLE_END ) ; i + + ) {
2005-04-16 15:20:36 -07:00
unsigned int freq = table [ i ] . frequency ;
if ( freq = = CPUFREQ_ENTRY_INVALID )
continue ;
if ( ( freq < policy - > min ) | | ( freq > policy - > max ) )
continue ;
2009-01-18 01:56:41 -05:00
switch ( relation ) {
2005-04-16 15:20:36 -07:00
case CPUFREQ_RELATION_H :
if ( freq < = target_freq ) {
if ( freq > = optimal . frequency ) {
optimal . frequency = freq ;
2013-03-30 16:25:15 +05:30
optimal . driver_data = i ;
2005-04-16 15:20:36 -07:00
}
} else {
if ( freq < = suboptimal . frequency ) {
suboptimal . frequency = freq ;
2013-03-30 16:25:15 +05:30
suboptimal . driver_data = i ;
2005-04-16 15:20:36 -07:00
}
}
break ;
case CPUFREQ_RELATION_L :
if ( freq > = target_freq ) {
if ( freq < = optimal . frequency ) {
optimal . frequency = freq ;
2013-03-30 16:25:15 +05:30
optimal . driver_data = i ;
2005-04-16 15:20:36 -07:00
}
} else {
if ( freq > = suboptimal . frequency ) {
suboptimal . frequency = freq ;
2013-03-30 16:25:15 +05:30
suboptimal . driver_data = i ;
2005-04-16 15:20:36 -07:00
}
}
break ;
}
}
2013-03-30 16:25:15 +05:30
if ( optimal . driver_data > i ) {
if ( suboptimal . driver_data > i )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2013-03-30 16:25:15 +05:30
* index = suboptimal . driver_data ;
2005-04-16 15:20:36 -07:00
} else
2013-03-30 16:25:15 +05:30
* index = optimal . driver_data ;
2005-04-16 15:20:36 -07:00
2014-03-28 19:11:45 +05:30
pr_debug ( " target index is %u, freq is:%u kHz \n " , * index ,
table [ * index ] . frequency ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( cpufreq_frequency_table_target ) ;
2013-12-03 11:20:46 +05:30
int cpufreq_frequency_table_get_index ( struct cpufreq_policy * policy ,
unsigned int freq )
{
struct cpufreq_frequency_table * table ;
int i ;
table = cpufreq_frequency_get_table ( policy - > cpu ) ;
if ( unlikely ( ! table ) ) {
pr_debug ( " %s: Unable to find frequency table \n " , __func__ ) ;
return - ENOENT ;
}
for ( i = 0 ; table [ i ] . frequency ! = CPUFREQ_TABLE_END ; i + + ) {
if ( table [ i ] . frequency = = freq )
return i ;
}
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( cpufreq_frequency_table_get_index ) ;
2005-04-16 15:20:36 -07:00
/**
2007-11-21 14:52:15 -08:00
* show_available_freqs - show available frequencies for the specified CPU
2005-04-16 15:20:36 -07:00
*/
2013-12-20 15:24:49 +01:00
static ssize_t show_available_freqs ( struct cpufreq_policy * policy , char * buf ,
bool show_boost )
2005-04-16 15:20:36 -07:00
{
unsigned int i = 0 ;
ssize_t count = 0 ;
2014-03-10 14:53:33 +05:30
struct cpufreq_frequency_table * table = policy - > freq_table ;
2005-04-16 15:20:36 -07:00
2014-03-10 14:53:33 +05:30
if ( ! table )
2005-04-16 15:20:36 -07:00
return - ENODEV ;
2009-01-18 01:56:41 -05:00
for ( i = 0 ; ( table [ i ] . frequency ! = CPUFREQ_TABLE_END ) ; i + + ) {
2005-04-16 15:20:36 -07:00
if ( table [ i ] . frequency = = CPUFREQ_ENTRY_INVALID )
continue ;
2013-12-20 15:24:49 +01:00
/*
* show_boost = true and driver_data = BOOST freq
* display BOOST freqs
*
* show_boost = false and driver_data = BOOST freq
* show_boost = true and driver_data ! = BOOST freq
* continue - do not display anything
*
* show_boost = false and driver_data ! = BOOST freq
* display NON BOOST freqs
*/
2014-03-28 19:11:47 +05:30
if ( show_boost ^ ( table [ i ] . flags & CPUFREQ_BOOST_FREQ ) )
2013-12-20 15:24:49 +01:00
continue ;
2005-04-16 15:20:36 -07:00
count + = sprintf ( & buf [ count ] , " %d " , table [ i ] . frequency ) ;
}
count + = sprintf ( & buf [ count ] , " \n " ) ;
return count ;
}
2013-12-20 15:24:49 +01:00
# define cpufreq_attr_available_freq(_name) \
struct freq_attr cpufreq_freq_attr_ # # _name # # _freqs = \
__ATTR_RO ( _name # # _frequencies )
/**
* show_scaling_available_frequencies - show available normal frequencies for
* the specified CPU
*/
static ssize_t scaling_available_frequencies_show ( struct cpufreq_policy * policy ,
char * buf )
{
return show_available_freqs ( policy , buf , false ) ;
}
cpufreq_attr_available_freq ( scaling_available ) ;
2005-04-16 15:20:36 -07:00
EXPORT_SYMBOL_GPL ( cpufreq_freq_attr_scaling_available_freqs ) ;
2013-12-20 15:24:49 +01:00
/**
* show_available_boost_freqs - show available boost frequencies for
* the specified CPU
*/
static ssize_t scaling_boost_frequencies_show ( struct cpufreq_policy * policy ,
char * buf )
{
return show_available_freqs ( policy , buf , true ) ;
}
cpufreq_attr_available_freq ( scaling_boost ) ;
EXPORT_SYMBOL_GPL ( cpufreq_freq_attr_scaling_boost_freqs ) ;
2013-10-03 20:27:55 +05:30
struct freq_attr * cpufreq_generic_attr [ ] = {
& cpufreq_freq_attr_scaling_available_freqs ,
2013-12-20 15:24:49 +01:00
# ifdef CONFIG_CPU_FREQ_BOOST_SW
& cpufreq_freq_attr_scaling_boost_freqs ,
# endif
2013-10-03 20:27:55 +05:30
NULL ,
} ;
EXPORT_SYMBOL_GPL ( cpufreq_generic_attr ) ;
2013-09-16 18:56:03 +05:30
int cpufreq_table_validate_and_show ( struct cpufreq_policy * policy ,
struct cpufreq_frequency_table * table )
{
int ret = cpufreq_frequency_table_cpuinfo ( policy , table ) ;
if ( ! ret )
2014-03-10 14:53:33 +05:30
policy - > freq_table = table ;
2013-09-16 18:56:03 +05:30
return ret ;
}
EXPORT_SYMBOL_GPL ( cpufreq_table_validate_and_show ) ;
2014-03-10 14:53:33 +05:30
struct cpufreq_policy * cpufreq_cpu_get_raw ( unsigned int cpu ) ;
2013-01-14 13:23:03 +00:00
2005-04-16 15:20:36 -07:00
struct cpufreq_frequency_table * cpufreq_frequency_get_table ( unsigned int cpu )
{
2014-03-10 14:53:33 +05:30
struct cpufreq_policy * policy = cpufreq_cpu_get_raw ( cpu ) ;
return policy ? policy - > freq_table : NULL ;
2005-04-16 15:20:36 -07:00
}
EXPORT_SYMBOL_GPL ( cpufreq_frequency_get_table ) ;
2009-01-18 01:56:41 -05:00
MODULE_AUTHOR ( " Dominik Brodowski <linux@brodo.de> " ) ;
MODULE_DESCRIPTION ( " CPUfreq frequency table helpers " ) ;
MODULE_LICENSE ( " GPL " ) ;