2005-04-16 15:20:36 -07:00
/*
* drivers / cpufreq / cpufreq_stats . c
*
* Copyright ( C ) 2003 - 2004 Venkatesh Pallipadi < venkatesh . pallipadi @ intel . com > .
2009-01-18 01:49:04 -05:00
* ( C ) 2004 Zou Nan hai < nanhai . zou @ intel . com > .
2005-04-16 15:20:36 -07: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 .
*/
# include <linux/cpu.h>
# include <linux/cpufreq.h>
2011-05-27 13:23:32 -04:00
# include <linux/module.h>
2013-08-06 22:53:03 +05:30
# include <linux/slab.h>
2005-04-16 15:20:36 -07:00
2016-05-31 22:14:44 +02:00
static DEFINE_SPINLOCK ( cpufreq_stats_lock ) ;
2005-04-16 15:20:36 -07:00
struct cpufreq_stats {
unsigned int total_trans ;
2013-06-19 14:19:33 +05:30
unsigned long long last_time ;
2005-04-16 15:20:36 -07:00
unsigned int max_state ;
unsigned int state_num ;
unsigned int last_index ;
2012-10-26 00:51:21 +02:00
u64 * time_in_state ;
2005-04-16 15:20:36 -07:00
unsigned int * freq_table ;
unsigned int * trans_table ;
} ;
2018-01-04 08:53:54 +05:30
static void cpufreq_stats_update ( struct cpufreq_stats * stats )
2005-04-16 15:20:36 -07:00
{
2015-01-06 21:09:03 +05:30
unsigned long long cur_time = get_jiffies_64 ( ) ;
2005-05-25 14:46:50 -07:00
2005-04-16 15:20:36 -07:00
spin_lock ( & cpufreq_stats_lock ) ;
2015-01-06 21:09:12 +05:30
stats - > time_in_state [ stats - > last_index ] + = cur_time - stats - > last_time ;
2015-01-06 21:09:07 +05:30
stats - > last_time = cur_time ;
2005-04-16 15:20:36 -07:00
spin_unlock ( & cpufreq_stats_lock ) ;
}
2016-11-07 10:02:23 -08:00
static void cpufreq_stats_clear_table ( struct cpufreq_stats * stats )
{
unsigned int count = stats - > max_state ;
memset ( stats - > time_in_state , 0 , count * sizeof ( u64 ) ) ;
memset ( stats - > trans_table , 0 , count * count * sizeof ( int ) ) ;
stats - > last_time = get_jiffies_64 ( ) ;
stats - > total_trans = 0 ;
}
2009-01-18 01:49:04 -05:00
static ssize_t show_total_trans ( struct cpufreq_policy * policy , char * buf )
2005-04-16 15:20:36 -07:00
{
2015-01-13 11:34:00 +05:30
return sprintf ( buf , " %d \n " , policy - > stats - > total_trans ) ;
2005-04-16 15:20:36 -07:00
}
2009-01-18 01:49:04 -05:00
static ssize_t show_time_in_state ( struct cpufreq_policy * policy , char * buf )
2005-04-16 15:20:36 -07:00
{
2015-01-06 21:09:07 +05:30
struct cpufreq_stats * stats = policy - > stats ;
2005-04-16 15:20:36 -07:00
ssize_t len = 0 ;
int i ;
2015-01-13 11:34:00 +05:30
2016-05-31 22:14:44 +02:00
if ( policy - > fast_switch_enabled )
return 0 ;
2015-01-06 21:09:07 +05:30
cpufreq_stats_update ( stats ) ;
for ( i = 0 ; i < stats - > state_num ; i + + ) {
len + = sprintf ( buf + len , " %u %llu \n " , stats - > freq_table [ i ] ,
2009-01-18 01:49:04 -05:00
( unsigned long long )
2015-01-06 21:09:07 +05:30
jiffies_64_to_clock_t ( stats - > time_in_state [ i ] ) ) ;
2005-04-16 15:20:36 -07:00
}
return len ;
}
2016-11-07 10:02:23 -08:00
static ssize_t store_reset ( struct cpufreq_policy * policy , const char * buf ,
size_t count )
{
/* We don't care what is written to the attribute. */
cpufreq_stats_clear_table ( policy - > stats ) ;
return count ;
}
2009-01-18 01:49:04 -05:00
static ssize_t show_trans_table ( struct cpufreq_policy * policy , char * buf )
2005-04-16 15:20:36 -07:00
{
2015-01-06 21:09:07 +05:30
struct cpufreq_stats * stats = policy - > stats ;
2005-04-16 15:20:36 -07:00
ssize_t len = 0 ;
int i , j ;
2016-05-31 22:14:44 +02:00
if ( policy - > fast_switch_enabled )
return 0 ;
2005-05-25 14:46:50 -07:00
len + = snprintf ( buf + len , PAGE_SIZE - len , " From : To \n " ) ;
len + = snprintf ( buf + len , PAGE_SIZE - len , " : " ) ;
2015-01-06 21:09:07 +05:30
for ( i = 0 ; i < stats - > state_num ; i + + ) {
2005-05-25 14:46:50 -07:00
if ( len > = PAGE_SIZE )
break ;
len + = snprintf ( buf + len , PAGE_SIZE - len , " %9u " ,
2015-01-06 21:09:07 +05:30
stats - > freq_table [ i ] ) ;
2005-05-25 14:46:50 -07:00
}
if ( len > = PAGE_SIZE )
2008-02-16 08:41:25 -02:00
return PAGE_SIZE ;
2005-05-25 14:46:50 -07:00
len + = snprintf ( buf + len , PAGE_SIZE - len , " \n " ) ;
2015-01-06 21:09:07 +05:30
for ( i = 0 ; i < stats - > state_num ; i + + ) {
2005-04-16 15:20:36 -07:00
if ( len > = PAGE_SIZE )
break ;
2005-05-25 14:46:50 -07:00
len + = snprintf ( buf + len , PAGE_SIZE - len , " %9u: " ,
2015-01-06 21:09:07 +05:30
stats - > freq_table [ i ] ) ;
2005-04-16 15:20:36 -07:00
2015-01-06 21:09:07 +05:30
for ( j = 0 ; j < stats - > state_num ; j + + ) {
2005-04-16 15:20:36 -07:00
if ( len > = PAGE_SIZE )
break ;
2005-05-25 14:46:50 -07:00
len + = snprintf ( buf + len , PAGE_SIZE - len , " %9u " ,
2015-01-06 21:09:07 +05:30
stats - > trans_table [ i * stats - > max_state + j ] ) ;
2005-04-16 15:20:36 -07:00
}
2008-02-16 08:41:25 -02:00
if ( len > = PAGE_SIZE )
break ;
2005-04-16 15:20:36 -07:00
len + = snprintf ( buf + len , PAGE_SIZE - len , " \n " ) ;
}
2017-11-07 13:39:29 +05:30
if ( len > = PAGE_SIZE ) {
pr_warn_once ( " cpufreq transition table exceeds PAGE_SIZE. Disabling \n " ) ;
return - EFBIG ;
}
2005-04-16 15:20:36 -07:00
return len ;
}
2013-02-04 11:38:52 +00:00
cpufreq_freq_attr_ro ( trans_table ) ;
2005-04-16 15:20:36 -07:00
2013-02-04 11:38:52 +00:00
cpufreq_freq_attr_ro ( total_trans ) ;
cpufreq_freq_attr_ro ( time_in_state ) ;
2016-11-07 10:02:23 -08:00
cpufreq_freq_attr_wo ( reset ) ;
2005-04-16 15:20:36 -07:00
static struct attribute * default_attrs [ ] = {
2013-02-04 11:38:52 +00:00
& total_trans . attr ,
& time_in_state . attr ,
2016-11-07 10:02:23 -08:00
& reset . attr ,
2013-02-04 11:38:52 +00:00
& trans_table . attr ,
2005-04-16 15:20:36 -07:00
NULL
} ;
2017-07-03 13:29:04 +05:30
static const struct attribute_group stats_attr_group = {
2005-04-16 15:20:36 -07:00
. attrs = default_attrs ,
. name = " stats "
} ;
2015-01-06 21:09:07 +05:30
static int freq_table_get_index ( struct cpufreq_stats * stats , unsigned int freq )
2005-04-16 15:20:36 -07:00
{
int index ;
2015-01-06 21:09:07 +05:30
for ( index = 0 ; index < stats - > max_state ; index + + )
if ( stats - > freq_table [ index ] = = freq )
2005-04-16 15:20:36 -07:00
return index ;
return - 1 ;
}
2016-05-31 22:14:44 +02:00
void cpufreq_stats_free_table ( struct cpufreq_policy * policy )
2005-04-16 15:20:36 -07:00
{
2015-01-06 21:09:07 +05:30
struct cpufreq_stats * stats = policy - > stats ;
2013-01-14 13:23:03 +00:00
2015-01-13 11:34:00 +05:30
/* Already freed */
2015-01-06 21:09:07 +05:30
if ( ! stats )
2014-01-07 07:10:12 +05:30
return ;
2015-01-06 21:09:07 +05:30
pr_debug ( " %s: Free stats table \n " , __func__ ) ;
2014-01-07 07:10:12 +05:30
sysfs_remove_group ( & policy - > kobj , & stats_attr_group ) ;
2015-01-06 21:09:07 +05:30
kfree ( stats - > time_in_state ) ;
kfree ( stats ) ;
2015-01-13 11:34:00 +05:30
policy - > stats = NULL ;
2011-05-02 11:29:17 -07:00
}
2016-05-31 22:14:44 +02:00
void cpufreq_stats_create_table ( struct cpufreq_policy * policy )
2005-04-16 15:20:36 -07:00
{
2015-01-06 21:09:11 +05:30
unsigned int i = 0 , count = 0 , ret = - ENOMEM ;
2015-01-06 21:09:07 +05:30
struct cpufreq_stats * stats ;
2005-04-16 15:20:36 -07:00
unsigned int alloc_size ;
2017-04-25 15:57:15 +05:30
struct cpufreq_frequency_table * pos ;
2014-02-27 17:58:36 -08:00
2017-04-25 15:57:15 +05:30
count = cpufreq_table_count_valid_entries ( policy ) ;
if ( ! count )
2016-05-31 22:14:44 +02:00
return ;
2014-02-27 17:58:36 -08:00
2015-01-06 21:09:01 +05:30
/* stats already initialized */
2015-01-13 11:34:00 +05:30
if ( policy - > stats )
2016-05-31 22:14:44 +02:00
return ;
2015-01-06 21:09:01 +05:30
2015-01-06 21:09:07 +05:30
stats = kzalloc ( sizeof ( * stats ) , GFP_KERNEL ) ;
2015-01-06 21:09:11 +05:30
if ( ! stats )
2016-05-31 22:14:44 +02:00
return ;
2005-04-16 15:20:36 -07:00
2012-10-26 00:51:21 +02:00
alloc_size = count * sizeof ( int ) + count * sizeof ( u64 ) ;
2005-04-16 15:20:36 -07:00
alloc_size + = count * count * sizeof ( int ) ;
2015-01-06 21:09:11 +05:30
/* Allocate memory for time_in_state/freq_table/trans_table in one go */
2015-01-06 21:09:07 +05:30
stats - > time_in_state = kzalloc ( alloc_size , GFP_KERNEL ) ;
2015-01-06 21:09:11 +05:30
if ( ! stats - > time_in_state )
goto free_stat ;
2015-01-06 21:09:07 +05:30
stats - > freq_table = ( unsigned int * ) ( stats - > time_in_state + count ) ;
2005-04-16 15:20:36 -07:00
2015-01-06 21:09:07 +05:30
stats - > trans_table = stats - > freq_table + count ;
2015-01-06 21:09:11 +05:30
stats - > max_state = count ;
/* Find valid-unique entries */
2017-04-25 15:57:15 +05:30
cpufreq_for_each_valid_entry ( pos , policy - > freq_table )
2015-01-06 21:09:07 +05:30
if ( freq_table_get_index ( stats , pos - > frequency ) = = - 1 )
stats - > freq_table [ i + + ] = pos - > frequency ;
2015-01-06 21:09:11 +05:30
2015-01-06 21:09:15 +05:30
stats - > state_num = i ;
2015-01-06 21:09:07 +05:30
stats - > last_time = get_jiffies_64 ( ) ;
stats - > last_index = freq_table_get_index ( stats , policy - > cur ) ;
2015-01-06 21:09:11 +05:30
policy - > stats = stats ;
ret = sysfs_create_group ( & policy - > kobj , & stats_attr_group ) ;
if ( ! ret )
2016-05-31 22:14:44 +02:00
return ;
2015-01-06 21:09:11 +05:30
/* We failed, release resources */
2015-01-13 11:34:00 +05:30
policy - > stats = NULL ;
2015-01-06 21:09:11 +05:30
kfree ( stats - > time_in_state ) ;
free_stat :
kfree ( stats ) ;
2014-01-07 07:10:13 +05:30
}
2016-05-31 22:14:44 +02:00
void cpufreq_stats_record_transition ( struct cpufreq_policy * policy ,
unsigned int new_freq )
2005-04-16 15:20:36 -07:00
{
2016-05-31 22:14:44 +02:00
struct cpufreq_stats * stats = policy - > stats ;
2005-04-16 15:20:36 -07:00
int old_index , new_index ;
2016-05-31 22:14:44 +02:00
if ( ! stats ) {
2015-01-13 11:34:00 +05:30
pr_debug ( " %s: No stats found \n " , __func__ ) ;
2016-05-31 22:14:44 +02:00
return ;
2015-01-13 11:34:00 +05:30
}
2015-01-06 21:09:07 +05:30
old_index = stats - > last_index ;
2016-05-31 22:14:44 +02:00
new_index = freq_table_get_index ( stats , new_freq ) ;
2005-04-16 15:20:36 -07:00
2015-01-06 21:09:07 +05:30
/* We can't do stats->time_in_state[-1]= .. */
2016-05-31 22:14:44 +02:00
if ( old_index = = - 1 | | new_index = = - 1 | | old_index = = new_index )
return ;
2006-12-19 12:58:55 -08:00
2015-01-06 21:09:14 +05:30
cpufreq_stats_update ( stats ) ;
2015-01-06 21:09:07 +05:30
stats - > last_index = new_index ;
stats - > trans_table [ old_index * stats - > max_state + new_index ] + + ;
stats - > total_trans + + ;
2005-04-16 15:20:36 -07:00
}