2009-12-16 23:38:28 +03:00
/*
* via - cputemp . c - Driver for VIA CPU core temperature monitoring
* Copyright ( C ) 2009 VIA Technologies , Inc .
*
* based on existing coretemp . c , which is
*
* Copyright ( C ) 2007 Rudolf Marek < r . marek @ assembler . cz >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License .
*
* This program 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 program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston , MA
* 02110 - 1301 USA .
*/
2010-10-20 10:51:50 +04:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-12-16 23:38:28 +03:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/hwmon.h>
2011-07-25 23:46:10 +04:00
# include <linux/hwmon-vid.h>
2009-12-16 23:38:28 +03:00
# include <linux/sysfs.h>
# include <linux/hwmon-sysfs.h>
# include <linux/err.h>
# include <linux/mutex.h>
# include <linux/list.h>
# include <linux/platform_device.h>
# include <linux/cpu.h>
# include <asm/msr.h>
# include <asm/processor.h>
2012-01-26 03:09:09 +04:00
# include <asm/cpu_device_id.h>
2009-12-16 23:38:28 +03:00
# define DRVNAME "via_cputemp"
2010-08-14 23:09:02 +04:00
enum { SHOW_TEMP , SHOW_LABEL , SHOW_NAME } ;
2009-12-16 23:38:28 +03:00
/*
* Functions declaration
*/
struct via_cputemp_data {
struct device * hwmon_dev ;
const char * name ;
2011-07-25 23:46:10 +04:00
u8 vrm ;
2009-12-16 23:38:28 +03:00
u32 id ;
2011-07-25 23:46:10 +04:00
u32 msr_temp ;
u32 msr_vid ;
2009-12-16 23:38:28 +03:00
} ;
/*
* Sysfs stuff
*/
static ssize_t show_name ( struct device * dev , struct device_attribute
* devattr , char * buf )
{
int ret ;
struct sensor_device_attribute * attr = to_sensor_dev_attr ( devattr ) ;
struct via_cputemp_data * data = dev_get_drvdata ( dev ) ;
if ( attr - > index = = SHOW_NAME )
ret = sprintf ( buf , " %s \n " , data - > name ) ;
else /* show label */
ret = sprintf ( buf , " Core %d \n " , data - > id ) ;
return ret ;
}
static ssize_t show_temp ( struct device * dev ,
struct device_attribute * devattr , char * buf )
{
struct via_cputemp_data * data = dev_get_drvdata ( dev ) ;
u32 eax , edx ;
int err ;
2011-07-25 23:46:10 +04:00
err = rdmsr_safe_on_cpu ( data - > id , data - > msr_temp , & eax , & edx ) ;
2009-12-16 23:38:28 +03:00
if ( err )
return - EAGAIN ;
return sprintf ( buf , " %lu \n " , ( ( unsigned long ) eax & 0xffffff ) * 1000 ) ;
}
2011-07-25 23:46:10 +04:00
static ssize_t show_cpu_vid ( struct device * dev ,
struct device_attribute * devattr , char * buf )
{
struct via_cputemp_data * data = dev_get_drvdata ( dev ) ;
u32 eax , edx ;
int err ;
err = rdmsr_safe_on_cpu ( data - > id , data - > msr_vid , & eax , & edx ) ;
if ( err )
return - EAGAIN ;
return sprintf ( buf , " %d \n " , vid_from_reg ( ~ edx & 0x7f , data - > vrm ) ) ;
}
2009-12-16 23:38:28 +03:00
static SENSOR_DEVICE_ATTR ( temp1_input , S_IRUGO , show_temp , NULL ,
SHOW_TEMP ) ;
static SENSOR_DEVICE_ATTR ( temp1_label , S_IRUGO , show_name , NULL , SHOW_LABEL ) ;
static SENSOR_DEVICE_ATTR ( name , S_IRUGO , show_name , NULL , SHOW_NAME ) ;
static struct attribute * via_cputemp_attributes [ ] = {
& sensor_dev_attr_name . dev_attr . attr ,
& sensor_dev_attr_temp1_label . dev_attr . attr ,
& sensor_dev_attr_temp1_input . dev_attr . attr ,
NULL
} ;
static const struct attribute_group via_cputemp_group = {
. attrs = via_cputemp_attributes ,
} ;
2011-07-25 23:46:10 +04:00
/* Optional attributes */
static DEVICE_ATTR ( cpu0_vid , S_IRUGO , show_cpu_vid , NULL ) ;
2012-11-19 22:22:35 +04:00
static int via_cputemp_probe ( struct platform_device * pdev )
2009-12-16 23:38:28 +03:00
{
struct via_cputemp_data * data ;
struct cpuinfo_x86 * c = & cpu_data ( pdev - > id ) ;
int err ;
u32 eax , edx ;
2012-06-03 00:12:58 +04:00
data = devm_kzalloc ( & pdev - > dev , sizeof ( struct via_cputemp_data ) ,
GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
2009-12-16 23:38:28 +03:00
data - > id = pdev - > id ;
data - > name = " via_cputemp " ;
switch ( c - > x86_model ) {
case 0xA :
/* C7 A */
case 0xD :
/* C7 D */
2011-07-25 23:46:10 +04:00
data - > msr_temp = 0x1169 ;
data - > msr_vid = 0x198 ;
2009-12-16 23:38:28 +03:00
break ;
case 0xF :
/* Nano */
2011-07-25 23:46:10 +04:00
data - > msr_temp = 0x1423 ;
2009-12-16 23:38:28 +03:00
break ;
default :
2012-06-03 00:12:58 +04:00
return - ENODEV ;
2009-12-16 23:38:28 +03:00
}
/* test if we can access the TEMPERATURE MSR */
2011-07-25 23:46:10 +04:00
err = rdmsr_safe_on_cpu ( data - > id , data - > msr_temp , & eax , & edx ) ;
2009-12-16 23:38:28 +03:00
if ( err ) {
dev_err ( & pdev - > dev ,
" Unable to access TEMPERATURE MSR, giving up \n " ) ;
2012-06-03 00:12:58 +04:00
return err ;
2009-12-16 23:38:28 +03:00
}
platform_set_drvdata ( pdev , data ) ;
err = sysfs_create_group ( & pdev - > dev . kobj , & via_cputemp_group ) ;
if ( err )
2012-06-03 00:12:58 +04:00
return err ;
2009-12-16 23:38:28 +03:00
2011-07-25 23:46:10 +04:00
if ( data - > msr_vid )
data - > vrm = vid_which_vrm ( ) ;
if ( data - > vrm ) {
err = device_create_file ( & pdev - > dev , & dev_attr_cpu0_vid ) ;
if ( err )
goto exit_remove ;
}
2009-12-16 23:38:28 +03:00
data - > hwmon_dev = hwmon_device_register ( & pdev - > dev ) ;
if ( IS_ERR ( data - > hwmon_dev ) ) {
err = PTR_ERR ( data - > hwmon_dev ) ;
dev_err ( & pdev - > dev , " Class registration failed (%d) \n " ,
err ) ;
goto exit_remove ;
}
return 0 ;
exit_remove :
2011-07-25 23:46:10 +04:00
if ( data - > vrm )
device_remove_file ( & pdev - > dev , & dev_attr_cpu0_vid ) ;
2009-12-16 23:38:28 +03:00
sysfs_remove_group ( & pdev - > dev . kobj , & via_cputemp_group ) ;
return err ;
}
2012-11-19 22:25:51 +04:00
static int via_cputemp_remove ( struct platform_device * pdev )
2009-12-16 23:38:28 +03:00
{
struct via_cputemp_data * data = platform_get_drvdata ( pdev ) ;
hwmon_device_unregister ( data - > hwmon_dev ) ;
2011-07-25 23:46:10 +04:00
if ( data - > vrm )
device_remove_file ( & pdev - > dev , & dev_attr_cpu0_vid ) ;
2009-12-16 23:38:28 +03:00
sysfs_remove_group ( & pdev - > dev . kobj , & via_cputemp_group ) ;
return 0 ;
}
static struct platform_driver via_cputemp_driver = {
. driver = {
. name = DRVNAME ,
} ,
. probe = via_cputemp_probe ,
2012-11-19 22:21:20 +04:00
. remove = via_cputemp_remove ,
2009-12-16 23:38:28 +03:00
} ;
struct pdev_entry {
struct list_head list ;
struct platform_device * pdev ;
unsigned int cpu ;
} ;
static LIST_HEAD ( pdev_list ) ;
static DEFINE_MUTEX ( pdev_list_mutex ) ;
2013-06-19 22:02:20 +04:00
static int via_cputemp_device_add ( unsigned int cpu )
2009-12-16 23:38:28 +03:00
{
int err ;
struct platform_device * pdev ;
struct pdev_entry * pdev_entry ;
pdev = platform_device_alloc ( DRVNAME , cpu ) ;
if ( ! pdev ) {
err = - ENOMEM ;
2010-10-20 10:51:50 +04:00
pr_err ( " Device allocation failed \n " ) ;
2009-12-16 23:38:28 +03:00
goto exit ;
}
pdev_entry = kzalloc ( sizeof ( struct pdev_entry ) , GFP_KERNEL ) ;
if ( ! pdev_entry ) {
err = - ENOMEM ;
goto exit_device_put ;
}
err = platform_device_add ( pdev ) ;
if ( err ) {
2010-10-20 10:51:50 +04:00
pr_err ( " Device addition failed (%d) \n " , err ) ;
2009-12-16 23:38:28 +03:00
goto exit_device_free ;
}
pdev_entry - > pdev = pdev ;
pdev_entry - > cpu = cpu ;
mutex_lock ( & pdev_list_mutex ) ;
list_add_tail ( & pdev_entry - > list , & pdev_list ) ;
mutex_unlock ( & pdev_list_mutex ) ;
return 0 ;
exit_device_free :
kfree ( pdev_entry ) ;
exit_device_put :
platform_device_put ( pdev ) ;
exit :
return err ;
}
2013-06-19 22:02:20 +04:00
static void via_cputemp_device_remove ( unsigned int cpu )
2009-12-16 23:38:28 +03:00
{
2010-12-06 19:48:35 +03:00
struct pdev_entry * p ;
2009-12-16 23:38:28 +03:00
mutex_lock ( & pdev_list_mutex ) ;
2010-12-06 19:48:35 +03:00
list_for_each_entry ( p , & pdev_list , list ) {
2009-12-16 23:38:28 +03:00
if ( p - > cpu = = cpu ) {
platform_device_unregister ( p - > pdev ) ;
list_del ( & p - > list ) ;
2010-12-06 19:48:35 +03:00
mutex_unlock ( & pdev_list_mutex ) ;
2009-12-16 23:38:28 +03:00
kfree ( p ) ;
2010-12-06 19:48:35 +03:00
return ;
2009-12-16 23:38:28 +03:00
}
}
mutex_unlock ( & pdev_list_mutex ) ;
}
2013-06-19 22:02:20 +04:00
static int via_cputemp_cpu_callback ( struct notifier_block * nfb ,
unsigned long action , void * hcpu )
2009-12-16 23:38:28 +03:00
{
unsigned int cpu = ( unsigned long ) hcpu ;
switch ( action ) {
case CPU_ONLINE :
case CPU_DOWN_FAILED :
via_cputemp_device_add ( cpu ) ;
break ;
case CPU_DOWN_PREPARE :
via_cputemp_device_remove ( cpu ) ;
break ;
}
return NOTIFY_OK ;
}
static struct notifier_block via_cputemp_cpu_notifier __refdata = {
. notifier_call = via_cputemp_cpu_callback ,
} ;
2012-07-30 13:33:00 +04:00
static const struct x86_cpu_id __initconst cputemp_ids [ ] = {
2012-01-26 03:09:09 +04:00
{ X86_VENDOR_CENTAUR , 6 , 0xa , } , /* C7 A */
{ X86_VENDOR_CENTAUR , 6 , 0xd , } , /* C7 D */
{ X86_VENDOR_CENTAUR , 6 , 0xf , } , /* Nano */
{ }
} ;
MODULE_DEVICE_TABLE ( x86cpu , cputemp_ids ) ;
2009-12-16 23:38:28 +03:00
static int __init via_cputemp_init ( void )
{
int i , err ;
2012-01-26 03:09:09 +04:00
if ( ! x86_match_cpu ( cputemp_ids ) )
return - ENODEV ;
2009-12-16 23:38:28 +03:00
err = platform_driver_register ( & via_cputemp_driver ) ;
if ( err )
goto exit ;
2014-03-11 00:41:33 +04:00
cpu_notifier_register_begin ( ) ;
2009-12-16 23:38:28 +03:00
for_each_online_cpu ( i ) {
struct cpuinfo_x86 * c = & cpu_data ( i ) ;
if ( c - > x86 ! = 6 )
continue ;
if ( c - > x86_model < 0x0a )
continue ;
if ( c - > x86_model > 0x0f ) {
2010-10-20 10:51:50 +04:00
pr_warn ( " Unknown CPU model 0x%x \n " , c - > x86_model ) ;
2009-12-16 23:38:28 +03:00
continue ;
}
2010-12-06 19:48:35 +03:00
via_cputemp_device_add ( i ) ;
2009-12-16 23:38:28 +03:00
}
2010-12-06 19:48:35 +03:00
# ifndef CONFIG_HOTPLUG_CPU
2009-12-16 23:38:28 +03:00
if ( list_empty ( & pdev_list ) ) {
2014-03-11 00:41:33 +04:00
cpu_notifier_register_done ( ) ;
2009-12-16 23:38:28 +03:00
err = - ENODEV ;
goto exit_driver_unreg ;
}
2010-12-06 19:48:35 +03:00
# endif
2009-12-16 23:38:28 +03:00
2014-03-11 00:41:33 +04:00
__register_hotcpu_notifier ( & via_cputemp_cpu_notifier ) ;
cpu_notifier_register_done ( ) ;
2009-12-16 23:38:28 +03:00
return 0 ;
2010-12-06 19:48:35 +03:00
# ifndef CONFIG_HOTPLUG_CPU
2009-12-16 23:38:28 +03:00
exit_driver_unreg :
platform_driver_unregister ( & via_cputemp_driver ) ;
2010-12-06 19:48:35 +03:00
# endif
2009-12-16 23:38:28 +03:00
exit :
return err ;
}
static void __exit via_cputemp_exit ( void )
{
struct pdev_entry * p , * n ;
2010-10-09 06:01:48 +04:00
2014-03-11 00:41:33 +04:00
cpu_notifier_register_begin ( ) ;
__unregister_hotcpu_notifier ( & via_cputemp_cpu_notifier ) ;
2009-12-16 23:38:28 +03:00
mutex_lock ( & pdev_list_mutex ) ;
list_for_each_entry_safe ( p , n , & pdev_list , list ) {
platform_device_unregister ( p - > pdev ) ;
list_del ( & p - > list ) ;
kfree ( p ) ;
}
mutex_unlock ( & pdev_list_mutex ) ;
2014-03-11 00:41:33 +04:00
cpu_notifier_register_done ( ) ;
2009-12-16 23:38:28 +03:00
platform_driver_unregister ( & via_cputemp_driver ) ;
}
MODULE_AUTHOR ( " Harald Welte <HaraldWelte@viatech.com> " ) ;
MODULE_DESCRIPTION ( " VIA CPU temperature monitor " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( via_cputemp_init )
module_exit ( via_cputemp_exit )