2019-05-19 15:51:31 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2006-08-28 14:40:17 +02:00
/*
* k8temp . c - Linux kernel module for hardware monitoring
*
2006-12-12 18:18:30 +01:00
* Copyright ( C ) 2006 Rudolf Marek < r . marek @ assembler . cz >
2006-08-28 14:40:17 +02:00
*
* Inspired from the w83785 and amd756 drivers .
*/
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/pci.h>
# include <linux/hwmon.h>
# include <linux/err.h>
# include <linux/mutex.h>
2009-01-15 22:27:46 +01:00
# include <asm/processor.h>
2006-08-28 14:40:17 +02:00
# define TEMP_FROM_REG(val) (((((val) >> 16) & 0xff) - 49) * 1000)
# define REG_TEMP 0xe4
# define SEL_PLACE 0x40
# define SEL_CORE 0x04
struct k8temp_data {
struct mutex update_lock ;
/* registers values */
2012-01-10 23:01:39 +01:00
u8 sensorsp ; /* sensor presence bits - SEL_CORE, SEL_PLACE */
2009-01-15 22:27:47 +01:00
u8 swap_core_select ; /* meaning of SEL_CORE is inverted */
2009-01-15 22:27:47 +01:00
u32 temp_offset ;
2006-08-28 14:40:17 +02:00
} ;
2013-12-03 07:10:29 +00:00
static const struct pci_device_id k8temp_ids [ ] = {
2006-08-28 14:40:17 +02:00
{ PCI_DEVICE ( PCI_VENDOR_ID_AMD , PCI_DEVICE_ID_AMD_K8_NB_MISC ) } ,
{ 0 } ,
} ;
2006-08-28 14:41:03 +02:00
MODULE_DEVICE_TABLE ( pci , k8temp_ids ) ;
2012-11-19 13:22:35 -05:00
static int is_rev_g_desktop ( u8 model )
2010-08-25 15:42:12 +02:00
{
u32 brandidx ;
if ( model < 0x69 )
return 0 ;
if ( model = = 0xc1 | | model = = 0x6c | | model = = 0x7c )
return 0 ;
/*
* Differentiate between AM2 and ASB1 .
* See " Constructing the processor Name String " in " Revision
* Guide for AMD NPT Family 0F h Processors " (33610).
*/
brandidx = cpuid_ebx ( 0x80000001 ) ;
brandidx = ( brandidx > > 9 ) & 0x1f ;
/* Single core */
if ( ( model = = 0x6f | | model = = 0x7f ) & &
( brandidx = = 0x7 | | brandidx = = 0x9 | | brandidx = = 0xc ) )
return 0 ;
/* Dual core */
if ( model = = 0x6b & &
( brandidx = = 0xb | | brandidx = = 0xc ) )
return 0 ;
return 1 ;
}
2019-07-21 14:00:51 +02:00
static umode_t
k8temp_is_visible ( const void * drvdata , enum hwmon_sensor_types type ,
u32 attr , int channel )
{
const struct k8temp_data * data = drvdata ;
if ( ( channel & 1 ) & & ! ( data - > sensorsp & SEL_PLACE ) )
return 0 ;
if ( ( channel & 2 ) & & ! ( data - > sensorsp & SEL_CORE ) )
return 0 ;
return 0444 ;
}
static int
k8temp_read ( struct device * dev , enum hwmon_sensor_types type ,
u32 attr , int channel , long * val )
{
struct k8temp_data * data = dev_get_drvdata ( dev ) ;
struct pci_dev * pdev = to_pci_dev ( dev - > parent ) ;
int core , place ;
u32 temp ;
u8 tmp ;
core = ( channel > > 1 ) & 1 ;
place = channel & 1 ;
core ^ = data - > swap_core_select ;
mutex_lock ( & data - > update_lock ) ;
pci_read_config_byte ( pdev , REG_TEMP , & tmp ) ;
tmp & = ~ ( SEL_PLACE | SEL_CORE ) ;
if ( core )
tmp | = SEL_CORE ;
if ( place )
tmp | = SEL_PLACE ;
pci_write_config_byte ( pdev , REG_TEMP , tmp ) ;
pci_read_config_dword ( pdev , REG_TEMP , & temp ) ;
mutex_unlock ( & data - > update_lock ) ;
* val = TEMP_FROM_REG ( temp ) + data - > temp_offset ;
return 0 ;
}
static const struct hwmon_ops k8temp_ops = {
. is_visible = k8temp_is_visible ,
. read = k8temp_read ,
} ;
2023-04-06 22:30:19 +02:00
static const struct hwmon_channel_info * const k8temp_info [ ] = {
2019-07-21 14:00:51 +02:00
HWMON_CHANNEL_INFO ( temp ,
HWMON_T_INPUT , HWMON_T_INPUT , HWMON_T_INPUT , HWMON_T_INPUT ) ,
NULL
} ;
static const struct hwmon_chip_info k8temp_chip_info = {
. ops = & k8temp_ops ,
. info = k8temp_info ,
} ;
2012-11-19 13:22:35 -05:00
static int k8temp_probe ( struct pci_dev * pdev ,
2006-08-28 14:40:17 +02:00
const struct pci_device_id * id )
{
u8 scfg ;
u32 temp ;
2009-01-15 22:27:46 +01:00
u8 model , stepping ;
2006-08-28 14:40:17 +02:00
struct k8temp_data * data ;
2019-07-21 14:00:51 +02:00
struct device * hwmon_dev ;
2006-08-28 14:40:17 +02:00
2012-06-02 12:04:06 -07:00
data = devm_kzalloc ( & pdev - > dev , sizeof ( struct k8temp_data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2006-08-28 14:40:17 +02:00
2009-01-15 22:27:46 +01:00
model = boot_cpu_data . x86_model ;
2018-01-01 09:52:10 +08:00
stepping = boot_cpu_data . x86_stepping ;
2009-01-15 22:27:46 +01:00
2010-10-28 20:31:42 +02:00
/* feature available since SH-C0, exclude older revisions */
2012-06-02 12:04:06 -07:00
if ( ( model = = 4 & & stepping = = 0 ) | |
( model = = 5 & & stepping < = 1 ) )
return - ENODEV ;
2009-01-15 22:27:47 +01:00
2010-10-28 20:31:42 +02:00
/*
* AMD NPT family 0f h , i . e . RevF and RevG :
* meaning of SEL_CORE bit is inverted
*/
if ( model > = 0x40 ) {
data - > swap_core_select = 1 ;
2013-01-10 10:01:24 -08:00
dev_warn ( & pdev - > dev ,
" Temperature readouts might be wrong - check erratum #141 \n " ) ;
2009-01-15 22:27:46 +01:00
}
2010-10-28 20:31:42 +02:00
/*
* RevG desktop CPUs ( i . e . no socket S1G1 or ASB1 parts ) need
* additional offset , otherwise reported temperature is below
* ambient temperature
*/
if ( is_rev_g_desktop ( model ) )
data - > temp_offset = 21000 ;
2006-08-28 14:40:17 +02:00
pci_read_config_byte ( pdev , REG_TEMP , & scfg ) ;
2012-01-10 23:01:39 +01:00
scfg & = ~ ( SEL_PLACE | SEL_CORE ) ; /* Select sensor 0, core0 */
2006-08-28 14:40:17 +02:00
pci_write_config_byte ( pdev , REG_TEMP , scfg ) ;
pci_read_config_byte ( pdev , REG_TEMP , & scfg ) ;
if ( scfg & ( SEL_PLACE | SEL_CORE ) ) {
dev_err ( & pdev - > dev , " Configuration bit(s) stuck at 1! \n " ) ;
2012-06-02 12:04:06 -07:00
return - ENODEV ;
2006-08-28 14:40:17 +02:00
}
scfg | = ( SEL_PLACE | SEL_CORE ) ;
pci_write_config_byte ( pdev , REG_TEMP , scfg ) ;
/* now we know if we can change core and/or sensor */
pci_read_config_byte ( pdev , REG_TEMP , & data - > sensorsp ) ;
if ( data - > sensorsp & SEL_PLACE ) {
scfg & = ~ SEL_CORE ; /* Select sensor 1, core0 */
pci_write_config_byte ( pdev , REG_TEMP , scfg ) ;
pci_read_config_dword ( pdev , REG_TEMP , & temp ) ;
scfg | = SEL_CORE ; /* prepare for next selection */
2012-01-10 23:01:39 +01:00
if ( ! ( ( temp > > 16 ) & 0xff ) ) /* if temp is 0 -49C is unlikely */
2006-08-28 14:40:17 +02:00
data - > sensorsp & = ~ SEL_PLACE ;
}
if ( data - > sensorsp & SEL_CORE ) {
scfg & = ~ SEL_PLACE ; /* Select sensor 0, core1 */
pci_write_config_byte ( pdev , REG_TEMP , scfg ) ;
pci_read_config_dword ( pdev , REG_TEMP , & temp ) ;
2012-01-10 23:01:39 +01:00
if ( ! ( ( temp > > 16 ) & 0xff ) ) /* if temp is 0 -49C is unlikely */
2006-08-28 14:40:17 +02:00
data - > sensorsp & = ~ SEL_CORE ;
}
mutex_init ( & data - > update_lock ) ;
2019-07-21 14:00:51 +02:00
hwmon_dev = devm_hwmon_device_register_with_info ( & pdev - > dev ,
" k8temp " ,
data ,
& k8temp_chip_info ,
NULL ) ;
2006-08-28 14:40:17 +02:00
2019-07-21 14:00:51 +02:00
return PTR_ERR_OR_ZERO ( hwmon_dev ) ;
2006-08-28 14:40:17 +02:00
}
static struct pci_driver k8temp_driver = {
. name = " k8temp " ,
. id_table = k8temp_ids ,
. probe = k8temp_probe ,
} ;
2012-04-02 21:25:46 -04:00
module_pci_driver ( k8temp_driver ) ;
2006-08-28 14:40:17 +02:00
2006-12-12 18:18:30 +01:00
MODULE_AUTHOR ( " Rudolf Marek <r.marek@assembler.cz> " ) ;
2006-08-28 14:40:17 +02:00
MODULE_DESCRIPTION ( " AMD K8 core temperature monitor " ) ;
MODULE_LICENSE ( " GPL " ) ;