2011-05-25 22:43:31 +04:00
/*
* fam15h_power . c - AMD Family 15 h processor power monitoring
*
* Copyright ( c ) 2011 Advanced Micro Devices , Inc .
2012-10-29 21:50:47 +04:00
* Author : Andreas Herrmann < herrmann . der . user @ googlemail . com >
2011-05-25 22:43:31 +04:00
*
*
* This driver is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License ; either
* version 2 of the License , or ( at your option ) any later version .
*
* This driver is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
* See the GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this driver ; if not , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/err.h>
# include <linux/hwmon.h>
# include <linux/hwmon-sysfs.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/bitops.h>
# include <asm/processor.h>
2015-10-30 12:56:57 +03:00
# include <asm/msr.h>
2011-05-25 22:43:31 +04:00
MODULE_DESCRIPTION ( " AMD Family 15h CPU processor power monitor " ) ;
2012-10-29 21:50:47 +04:00
MODULE_AUTHOR ( " Andreas Herrmann <herrmann.der.user@googlemail.com> " ) ;
2011-05-25 22:43:31 +04:00
MODULE_LICENSE ( " GPL " ) ;
/* D18F3 */
# define REG_NORTHBRIDGE_CAP 0xe8
/* D18F4 */
# define REG_PROCESSOR_TDP 0x1b8
/* D18F5 */
# define REG_TDP_RUNNING_AVERAGE 0xe0
# define REG_TDP_LIMIT3 0xe8
2015-10-30 12:56:55 +03:00
# define FAM15H_MIN_NUM_ATTRS 2
# define FAM15H_NUM_GROUPS 2
2015-10-30 12:56:57 +03:00
# define MSR_F15H_CU_MAX_PWR_ACCUMULATOR 0xc001007b
2011-05-25 22:43:31 +04:00
struct fam15h_power_data {
2014-06-19 19:29:11 +04:00
struct pci_dev * pdev ;
2011-05-25 22:43:31 +04:00
unsigned int tdp_to_watts ;
unsigned int base_tdp ;
unsigned int processor_pwr_watts ;
2015-08-27 11:07:38 +03:00
unsigned int cpu_pwr_sample_ratio ;
2015-10-30 12:56:55 +03:00
const struct attribute_group * groups [ FAM15H_NUM_GROUPS ] ;
struct attribute_group group ;
2015-10-30 12:56:57 +03:00
/* maximum accumulated power of a compute unit */
u64 max_cu_acc_power ;
2011-05-25 22:43:31 +04:00
} ;
static ssize_t show_power ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
u32 val , tdp_limit , running_avg_range ;
s32 running_avg_capture ;
u64 curr_pwr_watts ;
struct fam15h_power_data * data = dev_get_drvdata ( dev ) ;
2014-06-19 19:29:11 +04:00
struct pci_dev * f4 = data - > pdev ;
2011-05-25 22:43:31 +04:00
pci_bus_read_config_dword ( f4 - > bus , PCI_DEVFN ( PCI_SLOT ( f4 - > devfn ) , 5 ) ,
REG_TDP_RUNNING_AVERAGE , & val ) ;
2015-08-27 11:07:35 +03:00
/*
* On Carrizo and later platforms , TdpRunAvgAccCap bit field
* is extended to 4 : 31 from 4 : 25.
*/
if ( boot_cpu_data . x86 = = 0x15 & & boot_cpu_data . x86_model > = 0x60 ) {
running_avg_capture = val > > 4 ;
running_avg_capture = sign_extend32 ( running_avg_capture , 27 ) ;
} else {
running_avg_capture = ( val > > 4 ) & 0x3fffff ;
running_avg_capture = sign_extend32 ( running_avg_capture , 21 ) ;
}
2012-03-23 13:02:17 +04:00
running_avg_range = ( val & 0xf ) + 1 ;
2011-05-25 22:43:31 +04:00
pci_bus_read_config_dword ( f4 - > bus , PCI_DEVFN ( PCI_SLOT ( f4 - > devfn ) , 5 ) ,
REG_TDP_LIMIT3 , & val ) ;
tdp_limit = val > > 16 ;
2012-06-21 17:26:12 +04:00
curr_pwr_watts = ( ( u64 ) ( tdp_limit +
data - > base_tdp ) ) < < running_avg_range ;
2012-03-23 13:02:17 +04:00
curr_pwr_watts - = running_avg_capture ;
2011-05-25 22:43:31 +04:00
curr_pwr_watts * = data - > tdp_to_watts ;
/*
* Convert to microWatt
*
* power is in Watt provided as fixed point integer with
* scaling factor 1 / ( 2 ^ 16 ) . For conversion we use
* ( 10 ^ 6 ) / ( 2 ^ 16 ) = 15625 / ( 2 ^ 10 )
*/
2012-03-23 13:02:17 +04:00
curr_pwr_watts = ( curr_pwr_watts * 15625 ) > > ( 10 + running_avg_range ) ;
2011-05-25 22:43:31 +04:00
return sprintf ( buf , " %u \n " , ( unsigned int ) curr_pwr_watts ) ;
}
static DEVICE_ATTR ( power1_input , S_IRUGO , show_power , NULL ) ;
static ssize_t show_power_crit ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct fam15h_power_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %u \n " , data - > processor_pwr_watts ) ;
}
static DEVICE_ATTR ( power1_crit , S_IRUGO , show_power_crit , NULL ) ;
2015-10-30 12:56:55 +03:00
static int fam15h_power_init_attrs ( struct pci_dev * pdev ,
struct fam15h_power_data * data )
2014-09-16 23:58:04 +04:00
{
2015-10-30 12:56:55 +03:00
int n = FAM15H_MIN_NUM_ATTRS ;
struct attribute * * fam15h_power_attrs ;
2015-10-30 12:56:56 +03:00
struct cpuinfo_x86 * c = & boot_cpu_data ;
2014-09-16 23:58:04 +04:00
2015-10-30 12:56:56 +03:00
if ( c - > x86 = = 0x15 & &
( c - > x86_model < = 0xf | |
( c - > x86_model > = 0x60 & & c - > x86_model < = 0x6f ) ) )
2015-10-30 12:56:55 +03:00
n + = 1 ;
2014-09-16 23:58:04 +04:00
2015-10-30 12:56:55 +03:00
fam15h_power_attrs = devm_kcalloc ( & pdev - > dev , n ,
sizeof ( * fam15h_power_attrs ) ,
GFP_KERNEL ) ;
2011-05-25 22:43:31 +04:00
2015-10-30 12:56:55 +03:00
if ( ! fam15h_power_attrs )
return - ENOMEM ;
n = 0 ;
fam15h_power_attrs [ n + + ] = & dev_attr_power1_crit . attr ;
2015-10-30 12:56:56 +03:00
if ( c - > x86 = = 0x15 & &
( c - > x86_model < = 0xf | |
( c - > x86_model > = 0x60 & & c - > x86_model < = 0x6f ) ) )
2015-10-30 12:56:55 +03:00
fam15h_power_attrs [ n + + ] = & dev_attr_power1_input . attr ;
data - > group . attrs = fam15h_power_attrs ;
return 0 ;
}
2011-05-25 22:43:31 +04:00
2015-08-27 11:07:33 +03:00
static bool should_load_on_this_node ( struct pci_dev * f4 )
2011-05-25 22:43:31 +04:00
{
u32 val ;
pci_bus_read_config_dword ( f4 - > bus , PCI_DEVFN ( PCI_SLOT ( f4 - > devfn ) , 3 ) ,
REG_NORTHBRIDGE_CAP , & val ) ;
if ( ( val & BIT ( 29 ) ) & & ( ( val > > 30 ) & 3 ) )
return false ;
return true ;
}
2012-04-10 02:16:34 +04:00
/*
* Newer BKDG versions have an updated recommendation on how to properly
* initialize the running average range ( was : 0xE , now : 0x9 ) . This avoids
* counter saturations resulting in bogus power readings .
* We correct this value ourselves to cope with older BIOSes .
*/
2012-09-23 22:27:32 +04:00
static const struct pci_device_id affected_device [ ] = {
2012-04-26 00:44:20 +04:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_15H_NB_F4 ) } ,
{ 0 }
} ;
2012-09-23 22:27:32 +04:00
static void tweak_runavg_range ( struct pci_dev * pdev )
2012-04-10 02:16:34 +04:00
{
u32 val ;
/*
* let this quirk apply only to the current version of the
* northbridge , since future versions may change the behavior
*/
2012-04-26 00:44:20 +04:00
if ( ! pci_match_id ( affected_device , pdev ) )
2012-04-10 02:16:34 +04:00
return ;
pci_bus_read_config_dword ( pdev - > bus ,
PCI_DEVFN ( PCI_SLOT ( pdev - > devfn ) , 5 ) ,
REG_TDP_RUNNING_AVERAGE , & val ) ;
if ( ( val & 0xf ) ! = 0xe )
return ;
val & = ~ 0xf ;
val | = 0x9 ;
pci_bus_write_config_dword ( pdev - > bus ,
PCI_DEVFN ( PCI_SLOT ( pdev - > devfn ) , 5 ) ,
REG_TDP_RUNNING_AVERAGE , val ) ;
}
2012-09-23 22:27:32 +04:00
# ifdef CONFIG_PM
static int fam15h_power_resume ( struct pci_dev * pdev )
{
tweak_runavg_range ( pdev ) ;
return 0 ;
}
# else
# define fam15h_power_resume NULL
# endif
2015-10-30 12:56:55 +03:00
static int fam15h_power_init_data ( struct pci_dev * f4 ,
struct fam15h_power_data * data )
2011-05-25 22:43:31 +04:00
{
2015-08-27 11:07:38 +03:00
u32 val , eax , ebx , ecx , edx ;
2011-05-25 22:43:31 +04:00
u64 tmp ;
2015-10-30 12:56:55 +03:00
int ret ;
2011-05-25 22:43:31 +04:00
pci_read_config_dword ( f4 , REG_PROCESSOR_TDP , & val ) ;
data - > base_tdp = val > > 16 ;
tmp = val & 0xffff ;
pci_bus_read_config_dword ( f4 - > bus , PCI_DEVFN ( PCI_SLOT ( f4 - > devfn ) , 5 ) ,
REG_TDP_LIMIT3 , & val ) ;
data - > tdp_to_watts = ( ( val & 0x3ff ) < < 6 ) | ( ( val > > 10 ) & 0x3f ) ;
tmp * = data - > tdp_to_watts ;
/* result not allowed to be >= 256W */
if ( ( tmp > > 16 ) > = 256 )
2013-01-10 22:01:24 +04:00
dev_warn ( & f4 - > dev ,
" Bogus value for ProcessorPwrWatts (processor_pwr_watts>=%u) \n " ,
2011-05-25 22:43:31 +04:00
( unsigned int ) ( tmp > > 16 ) ) ;
/* convert to microWatt */
data - > processor_pwr_watts = ( tmp * 15625 ) > > 10 ;
2015-08-27 11:07:38 +03:00
2015-10-30 12:56:55 +03:00
ret = fam15h_power_init_attrs ( f4 , data ) ;
if ( ret )
return ret ;
2015-08-27 11:07:38 +03:00
cpuid ( 0x80000007 , & eax , & ebx , & ecx , & edx ) ;
/* CPUID Fn8000_0007:EDX[12] indicates to support accumulated power */
if ( ! ( edx & BIT ( 12 ) ) )
2015-10-30 12:56:55 +03:00
return 0 ;
2015-08-27 11:07:38 +03:00
/*
* determine the ratio of the compute unit power accumulator
* sample period to the PTSC counter period by executing CPUID
* Fn8000_0007 : ECX
*/
data - > cpu_pwr_sample_ratio = ecx ;
2015-10-30 12:56:55 +03:00
2015-10-30 12:56:57 +03:00
if ( rdmsrl_safe ( MSR_F15H_CU_MAX_PWR_ACCUMULATOR , & tmp ) ) {
pr_err ( " Failed to read max compute unit power accumulator MSR \n " ) ;
return - ENODEV ;
}
data - > max_cu_acc_power = tmp ;
2015-10-30 12:56:55 +03:00
return 0 ;
2011-05-25 22:43:31 +04:00
}
2012-11-19 22:22:35 +04:00
static int fam15h_power_probe ( struct pci_dev * pdev ,
2015-10-30 12:56:55 +03:00
const struct pci_device_id * id )
2011-05-25 22:43:31 +04:00
{
struct fam15h_power_data * data ;
2012-06-02 20:58:06 +04:00
struct device * dev = & pdev - > dev ;
2014-06-19 19:29:11 +04:00
struct device * hwmon_dev ;
2015-10-30 12:56:55 +03:00
int ret ;
2011-05-25 22:43:31 +04:00
2012-04-10 02:16:34 +04:00
/*
* though we ignore every other northbridge , we still have to
* do the tweaking on _each_ node in MCM processors as the counters
* are working hand - in - hand
*/
tweak_runavg_range ( pdev ) ;
2015-08-27 11:07:33 +03:00
if ( ! should_load_on_this_node ( pdev ) )
2012-06-02 20:58:06 +04:00
return - ENODEV ;
data = devm_kzalloc ( dev , sizeof ( struct fam15h_power_data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2011-05-25 22:43:31 +04:00
2015-10-30 12:56:55 +03:00
ret = fam15h_power_init_data ( pdev , data ) ;
if ( ret )
return ret ;
2014-06-19 19:29:11 +04:00
data - > pdev = pdev ;
2011-05-25 22:43:31 +04:00
2015-10-30 12:56:55 +03:00
data - > groups [ 0 ] = & data - > group ;
2014-06-19 19:29:11 +04:00
hwmon_dev = devm_hwmon_device_register_with_groups ( dev , " fam15h_power " ,
data ,
2015-10-30 12:56:55 +03:00
& data - > groups [ 0 ] ) ;
2014-06-19 19:29:11 +04:00
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2011-05-25 22:43:31 +04:00
}
2013-12-03 11:10:29 +04:00
static const struct pci_device_id fam15h_power_id_table [ ] = {
2011-05-25 22:43:31 +04:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_15H_NB_F4 ) } ,
2014-09-16 23:58:16 +04:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_15H_M30H_NB_F4 ) } ,
2015-08-27 11:07:32 +03:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_15H_M60H_NB_F4 ) } ,
2012-12-05 15:12:42 +04:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_16H_NB_F4 ) } ,
2014-11-04 20:49:02 +03:00
{ PCI_VDEVICE ( AMD , PCI_DEVICE_ID_AMD_16H_M30H_NB_F4 ) } ,
2011-05-25 22:43:31 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( pci , fam15h_power_id_table ) ;
static struct pci_driver fam15h_power_driver = {
. name = " fam15h_power " ,
. id_table = fam15h_power_id_table ,
. probe = fam15h_power_probe ,
2012-09-23 22:27:32 +04:00
. resume = fam15h_power_resume ,
2011-05-25 22:43:31 +04:00
} ;
2012-04-03 05:25:46 +04:00
module_pci_driver ( fam15h_power_driver ) ;