2005-04-16 15:20:36 -07:00
/*
* i8k . c - - Linux driver for accessing the SMM BIOS on Dell laptops .
*
* Copyright ( C ) 2001 Massimo Dal Zotto < dz @ debian . org >
*
2011-05-25 20:43:33 +02:00
* Hwmon integration :
2014-01-29 20:40:08 +01:00
* Copyright ( C ) 2011 Jean Delvare < jdelvare @ suse . de >
2013-12-14 09:30:16 -08:00
* Copyright ( C ) 2013 Guenter Roeck < linux @ roeck - us . net >
2011-05-25 20:43:33 +02:00
*
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 as published by the
* Free Software Foundation ; either version 2 , or ( at your option ) any
* later version .
*
* 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 .
*/
2013-12-14 09:30:08 -08:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
# include <linux/types.h>
# include <linux/init.h>
# include <linux/proc_fs.h>
2005-06-25 14:54:26 -07:00
# include <linux/seq_file.h>
2005-06-25 14:54:25 -07:00
# include <linux/dmi.h>
2007-05-08 00:28:59 -07:00
# include <linux/capability.h>
2010-06-02 14:28:52 +02:00
# include <linux/mutex.h>
2011-05-25 20:43:33 +02:00
# include <linux/hwmon.h>
# include <linux/hwmon-sysfs.h>
2013-12-14 09:30:09 -08:00
# include <linux/uaccess.h>
# include <linux/io.h>
2013-12-14 09:30:15 -08:00
# include <linux/sched.h>
2005-04-16 15:20:36 -07:00
# include <linux/i8k.h>
# define I8K_SMM_FN_STATUS 0x0025
# define I8K_SMM_POWER_STATUS 0x0069
# define I8K_SMM_SET_FAN 0x01a3
# define I8K_SMM_GET_FAN 0x00a3
# define I8K_SMM_GET_SPEED 0x02a3
# define I8K_SMM_GET_TEMP 0x10a3
2005-06-25 14:54:28 -07:00
# define I8K_SMM_GET_DELL_SIG1 0xfea3
# define I8K_SMM_GET_DELL_SIG2 0xffa3
2005-04-16 15:20:36 -07:00
# define I8K_FAN_MULT 30
# define I8K_MAX_TEMP 127
# define I8K_FN_NONE 0x00
# define I8K_FN_UP 0x01
# define I8K_FN_DOWN 0x02
# define I8K_FN_MUTE 0x04
# define I8K_FN_MASK 0x07
# define I8K_FN_SHIFT 8
# define I8K_POWER_AC 0x05
# define I8K_POWER_BATTERY 0x01
# define I8K_TEMPERATURE_BUG 1
2010-06-02 14:28:52 +02:00
static DEFINE_MUTEX ( i8k_mutex ) ;
2005-06-25 14:54:25 -07:00
static char bios_version [ 4 ] ;
2011-05-25 20:43:33 +02:00
static struct device * i8k_hwmon_dev ;
2013-12-14 09:30:10 -08:00
static u32 i8k_hwmon_flags ;
2013-12-14 09:30:19 -08:00
static int i8k_fan_mult ;
2014-06-21 08:08:10 -07:00
static int i8k_pwm_mult ;
static int i8k_fan_max = I8K_FAN_HIGH ;
2013-12-14 09:30:10 -08:00
# define I8K_HWMON_HAVE_TEMP1 (1 << 0)
2013-12-14 09:30:11 -08:00
# define I8K_HWMON_HAVE_TEMP2 (1 << 1)
# define I8K_HWMON_HAVE_TEMP3 (1 << 2)
# define I8K_HWMON_HAVE_TEMP4 (1 << 3)
# define I8K_HWMON_HAVE_FAN1 (1 << 4)
# define I8K_HWMON_HAVE_FAN2 (1 << 5)
2005-04-16 15:20:36 -07:00
MODULE_AUTHOR ( " Massimo Dal Zotto (dz@debian.org) " ) ;
MODULE_DESCRIPTION ( " Driver for accessing SMM BIOS on Dell laptops " ) ;
MODULE_LICENSE ( " GPL " ) ;
2012-01-13 09:32:20 +10:30
static bool force ;
2005-04-16 15:20:36 -07:00
module_param ( force , bool , 0 ) ;
MODULE_PARM_DESC ( force , " Force loading without checking for supported models " ) ;
2012-01-13 09:32:20 +10:30
static bool ignore_dmi ;
2005-06-25 14:54:25 -07:00
module_param ( ignore_dmi , bool , 0 ) ;
MODULE_PARM_DESC ( ignore_dmi , " Continue probing hardware even if DMI data does not match " ) ;
2012-01-13 09:32:20 +10:30
static bool restricted ;
2005-04-16 15:20:36 -07:00
module_param ( restricted , bool , 0 ) ;
MODULE_PARM_DESC ( restricted , " Allow fan control if SYS_ADMIN capability set " ) ;
2012-01-13 09:32:20 +10:30
static bool power_status ;
2005-04-16 15:20:36 -07:00
module_param ( power_status , bool , 0600 ) ;
MODULE_PARM_DESC ( power_status , " Report power status in /proc/i8k " ) ;
2008-05-01 04:34:58 -07:00
static int fan_mult = I8K_FAN_MULT ;
module_param ( fan_mult , int , 0 ) ;
MODULE_PARM_DESC ( fan_mult , " Factor to multiply fan speed with " ) ;
2014-06-21 08:08:10 -07:00
static int fan_max = I8K_FAN_HIGH ;
module_param ( fan_max , int , 0 ) ;
MODULE_PARM_DESC ( fan_max , " Maximum configurable fan speed " ) ;
2005-06-25 14:54:26 -07:00
static int i8k_open_fs ( struct inode * inode , struct file * file ) ;
2010-03-30 07:27:50 +02:00
static long i8k_ioctl ( struct file * , unsigned int , unsigned long ) ;
2005-04-16 15:20:36 -07:00
2006-07-03 00:24:21 -07:00
static const struct file_operations i8k_fops = {
2008-04-29 01:02:34 -07:00
. owner = THIS_MODULE ,
2005-06-25 14:54:26 -07:00
. open = i8k_open_fs ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
2010-03-30 07:27:50 +02:00
. unlocked_ioctl = i8k_ioctl ,
2005-04-16 15:20:36 -07:00
} ;
2005-06-25 14:54:27 -07:00
struct smm_regs {
2005-06-25 14:54:23 -07:00
unsigned int eax ;
2013-12-14 09:30:09 -08:00
unsigned int ebx __packed ;
unsigned int ecx __packed ;
unsigned int edx __packed ;
unsigned int esi __packed ;
unsigned int edi __packed ;
2005-06-25 14:54:27 -07:00
} ;
2005-04-16 15:20:36 -07:00
2007-10-03 15:15:40 -04:00
static inline const char * i8k_get_dmi_data ( int field )
2005-06-25 14:54:25 -07:00
{
2007-10-03 15:15:40 -04:00
const char * dmi_data = dmi_get_system_info ( field ) ;
2005-11-12 00:55:15 -05:00
return dmi_data & & * dmi_data ? dmi_data : " ? " ;
2005-06-25 14:54:25 -07:00
}
2005-04-16 15:20:36 -07:00
/*
* Call the System Management Mode BIOS . Code provided by Jonathan Buzzard .
*/
2005-06-25 14:54:27 -07:00
static int i8k_smm ( struct smm_regs * regs )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:23 -07:00
int rc ;
int eax = regs - > eax ;
2013-12-14 09:30:15 -08:00
cpumask_var_t old_mask ;
/* SMM requires CPU 0 */
if ( ! alloc_cpumask_var ( & old_mask , GFP_KERNEL ) )
return - ENOMEM ;
cpumask_copy ( old_mask , & current - > cpus_allowed ) ;
2014-06-21 08:08:08 -07:00
rc = set_cpus_allowed_ptr ( current , cpumask_of ( 0 ) ) ;
if ( rc )
goto out ;
2013-12-14 09:30:15 -08:00
if ( smp_processor_id ( ) ! = 0 ) {
rc = - EBUSY ;
goto out ;
}
2005-06-25 14:54:23 -07:00
2008-02-07 00:16:27 -08:00
# if defined(CONFIG_X86_64)
2010-11-15 21:22:37 +01:00
asm volatile ( " pushq %%rax \n \t "
2008-02-07 00:16:27 -08:00
" movl 0(%%rax),%%edx \n \t "
" pushq %%rdx \n \t "
" movl 4(%%rax),%%ebx \n \t "
" movl 8(%%rax),%%ecx \n \t "
" movl 12(%%rax),%%edx \n \t "
" movl 16(%%rax),%%esi \n \t "
" movl 20(%%rax),%%edi \n \t "
" popq %%rax \n \t "
" out %%al,$0xb2 \n \t "
" out %%al,$0x84 \n \t "
" xchgq %%rax,(%%rsp) \n \t "
" movl %%ebx,4(%%rax) \n \t "
" movl %%ecx,8(%%rax) \n \t "
" movl %%edx,12(%%rax) \n \t "
" movl %%esi,16(%%rax) \n \t "
" movl %%edi,20(%%rax) \n \t "
" popq %%rdx \n \t "
" movl %%edx,0(%%rax) \n \t "
2011-05-25 20:43:31 +02:00
" pushfq \n \t "
" popq %%rax \n \t "
2008-02-07 00:16:27 -08:00
" andl $1,%%eax \n "
2013-12-14 09:30:09 -08:00
: " =a " ( rc )
2008-02-07 00:16:27 -08:00
: " a " ( regs )
: " %ebx " , " %ecx " , " %edx " , " %esi " , " %edi " , " memory " ) ;
# else
2010-11-15 21:22:37 +01:00
asm volatile ( " pushl %%eax \n \t "
2005-06-25 14:54:23 -07:00
" movl 0(%%eax),%%edx \n \t "
" push %%edx \n \t "
" movl 4(%%eax),%%ebx \n \t "
" movl 8(%%eax),%%ecx \n \t "
" movl 12(%%eax),%%edx \n \t "
" movl 16(%%eax),%%esi \n \t "
" movl 20(%%eax),%%edi \n \t "
" popl %%eax \n \t "
" out %%al,$0xb2 \n \t "
" out %%al,$0x84 \n \t "
" xchgl %%eax,(%%esp) \n \t "
" movl %%ebx,4(%%eax) \n \t "
" movl %%ecx,8(%%eax) \n \t "
" movl %%edx,12(%%eax) \n \t "
" movl %%esi,16(%%eax) \n \t "
" movl %%edi,20(%%eax) \n \t "
" popl %%edx \n \t "
" movl %%edx,0(%%eax) \n \t "
" lahf \n \t "
" shrl $8,%%eax \n \t "
2010-11-13 12:13:53 +01:00
" andl $1,%%eax \n "
2013-12-14 09:30:09 -08:00
: " =a " ( rc )
2005-06-25 14:54:23 -07:00
: " a " ( regs )
: " %ebx " , " %ecx " , " %edx " , " %esi " , " %edi " , " memory " ) ;
2008-02-07 00:16:27 -08:00
# endif
2005-06-25 14:54:27 -07:00
if ( rc ! = 0 | | ( regs - > eax & 0xffff ) = = 0xffff | | regs - > eax = = eax )
2013-12-14 09:30:15 -08:00
rc = - EINVAL ;
2005-06-25 14:54:23 -07:00
2013-12-14 09:30:15 -08:00
out :
set_cpus_allowed_ptr ( current , old_mask ) ;
free_cpumask_var ( old_mask ) ;
return rc ;
2005-04-16 15:20:36 -07:00
}
/*
* Read the Fn key status .
*/
static int i8k_get_fn_status ( void )
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_FN_STATUS , } ;
2005-06-25 14:54:23 -07:00
int rc ;
2013-12-14 09:30:09 -08:00
rc = i8k_smm ( & regs ) ;
if ( rc < 0 )
2005-06-25 14:54:23 -07:00
return rc ;
switch ( ( regs . eax > > I8K_FN_SHIFT ) & I8K_FN_MASK ) {
case I8K_FN_UP :
return I8K_VOL_UP ;
case I8K_FN_DOWN :
return I8K_VOL_DOWN ;
case I8K_FN_MUTE :
return I8K_VOL_MUTE ;
default :
return 0 ;
}
2005-04-16 15:20:36 -07:00
}
/*
* Read the power status .
*/
static int i8k_get_power_status ( void )
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_POWER_STATUS , } ;
2005-06-25 14:54:23 -07:00
int rc ;
2013-12-14 09:30:09 -08:00
rc = i8k_smm ( & regs ) ;
if ( rc < 0 )
2005-06-25 14:54:23 -07:00
return rc ;
2005-06-25 14:54:27 -07:00
return ( regs . eax & 0xff ) = = I8K_POWER_AC ? I8K_AC : I8K_BATTERY ;
2005-04-16 15:20:36 -07:00
}
/*
* Read the fan status .
*/
static int i8k_get_fan_status ( int fan )
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_GET_FAN , } ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
regs . ebx = fan & 0xff ;
2005-06-25 14:54:27 -07:00
return i8k_smm ( & regs ) ? : regs . eax & 0xff ;
2005-04-16 15:20:36 -07:00
}
/*
* Read the fan speed in RPM .
*/
static int i8k_get_fan_speed ( int fan )
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_GET_SPEED , } ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
regs . ebx = fan & 0xff ;
2013-12-14 09:30:19 -08:00
return i8k_smm ( & regs ) ? : ( regs . eax & 0xffff ) * i8k_fan_mult ;
2005-04-16 15:20:36 -07:00
}
/*
* Set the fan speed ( off , low , high ) . Returns the new fan status .
*/
static int i8k_set_fan ( int fan , int speed )
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_SET_FAN , } ;
2005-04-16 15:20:36 -07:00
2014-06-21 08:08:10 -07:00
speed = ( speed < 0 ) ? 0 : ( ( speed > i8k_fan_max ) ? i8k_fan_max : speed ) ;
2005-06-25 14:54:23 -07:00
regs . ebx = ( fan & 0xff ) | ( speed < < 8 ) ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:27 -07:00
return i8k_smm ( & regs ) ? : i8k_get_fan_status ( fan ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Read the cpu temperature .
*/
2005-06-25 14:54:28 -07:00
static int i8k_get_temp ( int sensor )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:27 -07:00
struct smm_regs regs = { . eax = I8K_SMM_GET_TEMP , } ;
2005-06-25 14:54:23 -07:00
int rc ;
int temp ;
2005-04-16 15:20:36 -07:00
# ifdef I8K_TEMPERATURE_BUG
2013-12-14 09:30:11 -08:00
static int prev [ 4 ] ;
2005-04-16 15:20:36 -07:00
# endif
2005-06-25 14:54:28 -07:00
regs . ebx = sensor & 0xff ;
2013-12-14 09:30:09 -08:00
rc = i8k_smm ( & regs ) ;
if ( rc < 0 )
2005-06-25 14:54:23 -07:00
return rc ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
temp = regs . eax & 0xff ;
2005-04-16 15:20:36 -07:00
# ifdef I8K_TEMPERATURE_BUG
2005-06-25 14:54:23 -07:00
/*
* Sometimes the temperature sensor returns 0x99 , which is out of range .
* In this case we return ( once ) the previous cached value . For example :
# 1003655137 00000058 00005a4b
# 1003655138 00000099 00003a80 <--- 0x99 = 153 degrees
# 1003655139 00000054 00005c52
*/
if ( temp > I8K_MAX_TEMP ) {
2013-12-14 09:30:11 -08:00
temp = prev [ sensor ] ;
prev [ sensor ] = I8K_MAX_TEMP ;
2005-06-25 14:54:23 -07:00
} else {
2013-12-14 09:30:11 -08:00
prev [ sensor ] = temp ;
2005-06-25 14:54:23 -07:00
}
2005-04-16 15:20:36 -07:00
# endif
2005-06-25 14:54:23 -07:00
return temp ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:28 -07:00
static int i8k_get_dell_signature ( int req_fn )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:28 -07:00
struct smm_regs regs = { . eax = req_fn , } ;
2005-06-25 14:54:23 -07:00
int rc ;
2005-04-16 15:20:36 -07:00
2013-12-14 09:30:09 -08:00
rc = i8k_smm ( & regs ) ;
if ( rc < 0 )
2005-06-25 14:54:23 -07:00
return rc ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:27 -07:00
return regs . eax = = 1145651527 & & regs . edx = = 1145392204 ? 0 : - 1 ;
2005-04-16 15:20:36 -07:00
}
2010-03-30 07:27:50 +02:00
static int
i8k_ioctl_unlocked ( struct file * fp , unsigned int cmd , unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:25 -07:00
int val = 0 ;
2005-06-25 14:54:23 -07:00
int speed ;
unsigned char buff [ 16 ] ;
int __user * argp = ( int __user * ) arg ;
if ( ! argp )
return - EINVAL ;
switch ( cmd ) {
case I8K_BIOS_VERSION :
2013-12-14 09:30:20 -08:00
val = ( bios_version [ 0 ] < < 16 ) |
( bios_version [ 1 ] < < 8 ) | bios_version [ 2 ] ;
2005-06-25 14:54:23 -07:00
break ;
case I8K_MACHINE_ID :
memset ( buff , 0 , 16 ) ;
2013-12-14 09:30:09 -08:00
strlcpy ( buff , i8k_get_dmi_data ( DMI_PRODUCT_SERIAL ) ,
sizeof ( buff ) ) ;
2005-06-25 14:54:23 -07:00
break ;
case I8K_FN_STATUS :
val = i8k_get_fn_status ( ) ;
break ;
case I8K_POWER_STATUS :
val = i8k_get_power_status ( ) ;
break ;
case I8K_GET_TEMP :
2005-06-25 14:54:28 -07:00
val = i8k_get_temp ( 0 ) ;
2005-06-25 14:54:23 -07:00
break ;
case I8K_GET_SPEED :
2005-06-25 14:54:27 -07:00
if ( copy_from_user ( & val , argp , sizeof ( int ) ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
val = i8k_get_fan_speed ( val ) ;
break ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
case I8K_GET_FAN :
2005-06-25 14:54:27 -07:00
if ( copy_from_user ( & val , argp , sizeof ( int ) ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
val = i8k_get_fan_status ( val ) ;
break ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
case I8K_SET_FAN :
2005-06-25 14:54:27 -07:00
if ( restricted & & ! capable ( CAP_SYS_ADMIN ) )
2005-06-25 14:54:23 -07:00
return - EPERM ;
2005-06-25 14:54:27 -07:00
if ( copy_from_user ( & val , argp , sizeof ( int ) ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
if ( copy_from_user ( & speed , argp + 1 , sizeof ( int ) ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
val = i8k_set_fan ( val , speed ) ;
break ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
default :
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:27 -07:00
if ( val < 0 )
2005-06-25 14:54:23 -07:00
return val ;
2005-04-16 15:20:36 -07:00
2005-06-25 14:54:23 -07:00
switch ( cmd ) {
case I8K_BIOS_VERSION :
2005-06-25 14:54:27 -07:00
if ( copy_to_user ( argp , & val , 4 ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
break ;
case I8K_MACHINE_ID :
2005-06-25 14:54:27 -07:00
if ( copy_to_user ( argp , buff , 16 ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
break ;
default :
2005-06-25 14:54:27 -07:00
if ( copy_to_user ( argp , & val , sizeof ( int ) ) )
2005-06-25 14:54:23 -07:00
return - EFAULT ;
2005-06-25 14:54:27 -07:00
2005-06-25 14:54:23 -07:00
break ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:23 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2010-03-30 07:27:50 +02:00
static long i8k_ioctl ( struct file * fp , unsigned int cmd , unsigned long arg )
{
long ret ;
2010-06-02 14:28:52 +02:00
mutex_lock ( & i8k_mutex ) ;
2010-03-30 07:27:50 +02:00
ret = i8k_ioctl_unlocked ( fp , cmd , arg ) ;
2010-06-02 14:28:52 +02:00
mutex_unlock ( & i8k_mutex ) ;
2010-03-30 07:27:50 +02:00
return ret ;
}
2005-04-16 15:20:36 -07:00
/*
* Print the information for / proc / i8k .
*/
2005-06-25 14:54:26 -07:00
static int i8k_proc_show ( struct seq_file * seq , void * offset )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:26 -07:00
int fn_key , cpu_temp , ac_power ;
2005-06-25 14:54:23 -07:00
int left_fan , right_fan , left_speed , right_speed ;
2007-10-19 23:21:04 +02:00
cpu_temp = i8k_get_temp ( 0 ) ; /* 11100 µs */
left_fan = i8k_get_fan_status ( I8K_FAN_LEFT ) ; /* 580 µs */
right_fan = i8k_get_fan_status ( I8K_FAN_RIGHT ) ; /* 580 µs */
left_speed = i8k_get_fan_speed ( I8K_FAN_LEFT ) ; /* 580 µs */
right_speed = i8k_get_fan_speed ( I8K_FAN_RIGHT ) ; /* 580 µs */
fn_key = i8k_get_fn_status ( ) ; /* 750 µs */
2005-06-25 14:54:27 -07:00
if ( power_status )
2007-10-19 23:21:04 +02:00
ac_power = i8k_get_power_status ( ) ; /* 14700 µs */
2005-06-25 14:54:27 -07:00
else
2005-06-25 14:54:23 -07:00
ac_power = - 1 ;
/*
* Info :
*
* 1 ) Format version ( this will change if format changes )
* 2 ) BIOS version
* 3 ) BIOS machine ID
* 4 ) Cpu temperature
* 5 ) Left fan status
* 6 ) Right fan status
* 7 ) Left fan speed
* 8 ) Right fan speed
* 9 ) AC power
* 10 ) Fn Key status
*/
2005-06-25 14:54:26 -07:00
return seq_printf ( seq , " %s %s %s %d %d %d %d %d %d %d \n " ,
I8K_PROC_FMT ,
bios_version ,
2005-11-12 00:55:15 -05:00
i8k_get_dmi_data ( DMI_PRODUCT_SERIAL ) ,
2005-06-25 14:54:26 -07:00
cpu_temp ,
left_fan , right_fan , left_speed , right_speed ,
ac_power , fn_key ) ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:26 -07:00
static int i8k_open_fs ( struct inode * inode , struct file * file )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:26 -07:00
return single_open ( file , i8k_proc_show , NULL ) ;
2005-04-16 15:20:36 -07:00
}
2011-05-25 20:43:33 +02:00
/*
* Hwmon interface
*/
static ssize_t i8k_hwmon_show_temp ( struct device * dev ,
struct device_attribute * devattr ,
char * buf )
{
2013-12-14 09:30:11 -08:00
int index = to_sensor_dev_attr ( devattr ) - > index ;
int temp ;
2011-05-25 20:43:33 +02:00
2013-12-14 09:30:11 -08:00
temp = i8k_get_temp ( index ) ;
if ( temp < 0 )
return temp ;
return sprintf ( buf , " %d \n " , temp * 1000 ) ;
2011-05-25 20:43:33 +02:00
}
static ssize_t i8k_hwmon_show_fan ( struct device * dev ,
struct device_attribute * devattr ,
char * buf )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
int fan_speed ;
fan_speed = i8k_get_fan_speed ( index ) ;
if ( fan_speed < 0 )
return fan_speed ;
return sprintf ( buf , " %d \n " , fan_speed ) ;
}
2013-12-14 09:30:21 -08:00
static ssize_t i8k_hwmon_show_pwm ( struct device * dev ,
struct device_attribute * devattr ,
char * buf )
{
int index = to_sensor_dev_attr ( devattr ) - > index ;
int status ;
status = i8k_get_fan_status ( index ) ;
if ( status < 0 )
return - EIO ;
2014-06-21 08:08:10 -07:00
return sprintf ( buf , " %d \n " , clamp_val ( status * i8k_pwm_mult , 0 , 255 ) ) ;
2013-12-14 09:30:21 -08:00
}
static ssize_t i8k_hwmon_set_pwm ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
{
int index = to_sensor_dev_attr ( attr ) - > index ;
unsigned long val ;
int err ;
err = kstrtoul ( buf , 10 , & val ) ;
if ( err )
return err ;
2014-06-21 08:08:10 -07:00
val = clamp_val ( DIV_ROUND_CLOSEST ( val , i8k_pwm_mult ) , 0 , i8k_fan_max ) ;
2013-12-14 09:30:21 -08:00
mutex_lock ( & i8k_mutex ) ;
err = i8k_set_fan ( index , val ) ;
mutex_unlock ( & i8k_mutex ) ;
return err < 0 ? - EIO : count ;
}
2013-12-14 09:30:11 -08:00
static SENSOR_DEVICE_ATTR ( temp1_input , S_IRUGO , i8k_hwmon_show_temp , NULL , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp2_input , S_IRUGO , i8k_hwmon_show_temp , NULL , 1 ) ;
static SENSOR_DEVICE_ATTR ( temp3_input , S_IRUGO , i8k_hwmon_show_temp , NULL , 2 ) ;
static SENSOR_DEVICE_ATTR ( temp4_input , S_IRUGO , i8k_hwmon_show_temp , NULL , 3 ) ;
2011-05-25 20:43:33 +02:00
static SENSOR_DEVICE_ATTR ( fan1_input , S_IRUGO , i8k_hwmon_show_fan , NULL ,
I8K_FAN_LEFT ) ;
2013-12-14 09:30:21 -08:00
static SENSOR_DEVICE_ATTR ( pwm1 , S_IRUGO | S_IWUSR , i8k_hwmon_show_pwm ,
i8k_hwmon_set_pwm , I8K_FAN_LEFT ) ;
2011-05-25 20:43:33 +02:00
static SENSOR_DEVICE_ATTR ( fan2_input , S_IRUGO , i8k_hwmon_show_fan , NULL ,
I8K_FAN_RIGHT ) ;
2013-12-14 09:30:21 -08:00
static SENSOR_DEVICE_ATTR ( pwm2 , S_IRUGO | S_IWUSR , i8k_hwmon_show_pwm ,
i8k_hwmon_set_pwm , I8K_FAN_RIGHT ) ;
2013-12-14 09:30:10 -08:00
static struct attribute * i8k_attrs [ ] = {
2013-12-14 09:30:11 -08:00
& sensor_dev_attr_temp1_input . dev_attr . attr , /* 0 */
2014-06-21 08:08:09 -07:00
& sensor_dev_attr_temp2_input . dev_attr . attr , /* 1 */
& sensor_dev_attr_temp3_input . dev_attr . attr , /* 2 */
& sensor_dev_attr_temp4_input . dev_attr . attr , /* 3 */
& sensor_dev_attr_fan1_input . dev_attr . attr , /* 4 */
& sensor_dev_attr_pwm1 . dev_attr . attr , /* 5 */
& sensor_dev_attr_fan2_input . dev_attr . attr , /* 6 */
& sensor_dev_attr_pwm2 . dev_attr . attr , /* 7 */
2013-12-14 09:30:10 -08:00
NULL
} ;
2011-05-25 20:43:33 +02:00
2013-12-14 09:30:10 -08:00
static umode_t i8k_is_visible ( struct kobject * kobj , struct attribute * attr ,
int index )
2011-05-25 20:43:33 +02:00
{
2014-06-21 08:08:09 -07:00
if ( index = = 0 & & ! ( i8k_hwmon_flags & I8K_HWMON_HAVE_TEMP1 ) )
2013-12-14 09:30:10 -08:00
return 0 ;
2014-06-21 08:08:09 -07:00
if ( index = = 1 & & ! ( i8k_hwmon_flags & I8K_HWMON_HAVE_TEMP2 ) )
2013-12-14 09:30:11 -08:00
return 0 ;
2014-06-21 08:08:09 -07:00
if ( index = = 2 & & ! ( i8k_hwmon_flags & I8K_HWMON_HAVE_TEMP3 ) )
2013-12-14 09:30:11 -08:00
return 0 ;
2014-06-21 08:08:09 -07:00
if ( index = = 3 & & ! ( i8k_hwmon_flags & I8K_HWMON_HAVE_TEMP4 ) )
2013-12-14 09:30:11 -08:00
return 0 ;
2014-06-21 08:08:09 -07:00
if ( index > = 4 & & index < = 5 & &
2013-12-14 09:30:10 -08:00
! ( i8k_hwmon_flags & I8K_HWMON_HAVE_FAN1 ) )
return 0 ;
2014-06-21 08:08:09 -07:00
if ( index > = 6 & & index < = 7 & &
2013-12-14 09:30:10 -08:00
! ( i8k_hwmon_flags & I8K_HWMON_HAVE_FAN2 ) )
return 0 ;
return attr - > mode ;
2011-05-25 20:43:33 +02:00
}
2013-12-14 09:30:10 -08:00
static const struct attribute_group i8k_group = {
. attrs = i8k_attrs ,
. is_visible = i8k_is_visible ,
} ;
__ATTRIBUTE_GROUPS ( i8k ) ;
2011-05-25 20:43:33 +02:00
static int __init i8k_init_hwmon ( void )
{
int err ;
2013-12-14 09:30:10 -08:00
i8k_hwmon_flags = 0 ;
2011-05-25 20:43:33 +02:00
/* CPU temperature attributes, if temperature reading is OK */
err = i8k_get_temp ( 0 ) ;
2013-12-14 09:30:10 -08:00
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_TEMP1 ;
2013-12-14 09:30:11 -08:00
/* check for additional temperature sensors */
err = i8k_get_temp ( 1 ) ;
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_TEMP2 ;
err = i8k_get_temp ( 2 ) ;
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_TEMP3 ;
err = i8k_get_temp ( 3 ) ;
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_TEMP4 ;
2011-05-25 20:43:33 +02:00
/* Left fan attributes, if left fan is present */
err = i8k_get_fan_status ( I8K_FAN_LEFT ) ;
2013-12-14 09:30:10 -08:00
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_FAN1 ;
2011-05-25 20:43:33 +02:00
/* Right fan attributes, if right fan is present */
err = i8k_get_fan_status ( I8K_FAN_RIGHT ) ;
2013-12-14 09:30:10 -08:00
if ( err > = 0 )
i8k_hwmon_flags | = I8K_HWMON_HAVE_FAN2 ;
2011-05-25 20:43:33 +02:00
2013-12-14 09:30:10 -08:00
i8k_hwmon_dev = hwmon_device_register_with_groups ( NULL , " i8k " , NULL ,
i8k_groups ) ;
if ( IS_ERR ( i8k_hwmon_dev ) ) {
err = PTR_ERR ( i8k_hwmon_dev ) ;
i8k_hwmon_dev = NULL ;
pr_err ( " hwmon registration failed (%d) \n " , err ) ;
return err ;
}
2011-05-25 20:43:33 +02:00
return 0 ;
}
2014-06-21 08:08:10 -07:00
struct i8k_config_data {
int fan_mult ;
int fan_max ;
} ;
enum i8k_configs {
2014-06-21 08:08:12 -07:00
DELL_LATITUDE_D520 ,
2014-09-05 18:03:42 +01:00
DELL_LATITUDE_E6540 ,
2014-06-21 08:08:12 -07:00
DELL_PRECISION_490 ,
2014-06-21 08:08:10 -07:00
DELL_STUDIO ,
DELL_XPS_M140 ,
} ;
static const struct i8k_config_data i8k_config_data [ ] = {
2014-06-21 08:08:12 -07:00
[ DELL_LATITUDE_D520 ] = {
. fan_mult = 1 ,
. fan_max = I8K_FAN_TURBO ,
} ,
2014-09-05 18:03:42 +01:00
[ DELL_LATITUDE_E6540 ] = {
. fan_mult = 1 ,
. fan_max = I8K_FAN_HIGH ,
} ,
2014-06-21 08:08:12 -07:00
[ DELL_PRECISION_490 ] = {
. fan_mult = 1 ,
. fan_max = I8K_FAN_TURBO ,
} ,
2014-06-21 08:08:10 -07:00
[ DELL_STUDIO ] = {
. fan_mult = 1 ,
. fan_max = I8K_FAN_HIGH ,
} ,
[ DELL_XPS_M140 ] = {
. fan_mult = 1 ,
. fan_max = I8K_FAN_HIGH ,
} ,
} ;
2013-12-14 09:30:09 -08:00
static struct dmi_system_id i8k_dmi_table [ ] __initdata = {
2005-06-25 14:54:25 -07:00
{
. ident = " Dell Inspiron " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Computer " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Inspiron " ) ,
} ,
} ,
{
. ident = " Dell Latitude " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Computer " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Latitude " ) ,
} ,
} ,
2005-06-25 14:54:28 -07:00
{
. ident = " Dell Inspiron 2 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Inspiron " ) ,
} ,
} ,
2014-06-21 08:08:12 -07:00
{
. ident = " Dell Latitude D520 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Latitude D520 " ) ,
} ,
. driver_data = ( void * ) & i8k_config_data [ DELL_LATITUDE_D520 ] ,
} ,
2014-09-05 18:03:42 +01:00
{
. ident = " Dell Latitude E6540 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Latitude E6540 " ) ,
} ,
. driver_data = ( void * ) & i8k_config_data [ DELL_LATITUDE_E6540 ] ,
} ,
2005-06-25 14:54:28 -07:00
{
. ident = " Dell Latitude 2 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Latitude " ) ,
} ,
} ,
2008-02-06 01:37:47 -08:00
{ /* UK Inspiron 6400 */
. ident = " Dell Inspiron 3 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MM061 " ) ,
} ,
} ,
2008-02-07 00:16:31 -08:00
{
. ident = " Dell Inspiron 3 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MP061 " ) ,
} ,
} ,
2014-06-21 08:08:12 -07:00
{
. ident = " Dell Precision 490 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME ,
" Precision WorkStation 490 " ) ,
} ,
. driver_data = ( void * ) & i8k_config_data [ DELL_PRECISION_490 ] ,
} ,
2009-01-02 16:19:13 +00:00
{
. ident = " Dell Precision " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Precision " ) ,
} ,
} ,
2009-01-02 16:19:23 +00:00
{
. ident = " Dell Vostro " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Vostro " ) ,
} ,
} ,
2013-12-03 13:56:56 -08:00
{
. ident = " Dell XPS421 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " XPS L421X " ) ,
} ,
} ,
2013-12-14 09:30:17 -08:00
{
. ident = " Dell Studio " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Studio " ) ,
} ,
2014-06-21 08:08:10 -07:00
. driver_data = ( void * ) & i8k_config_data [ DELL_STUDIO ] ,
2013-12-14 09:30:17 -08:00
} ,
2013-12-14 09:30:18 -08:00
{
. ident = " Dell XPS M140 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " MXC051 " ) ,
} ,
2014-06-21 08:08:10 -07:00
. driver_data = ( void * ) & i8k_config_data [ DELL_XPS_M140 ] ,
2013-12-14 09:30:18 -08:00
} ,
2013-12-14 09:30:09 -08:00
{ }
2005-06-25 14:54:25 -07:00
} ;
2005-04-16 15:20:36 -07:00
/*
* Probe for the presence of a supported laptop .
*/
static int __init i8k_probe ( void )
{
2013-12-14 09:30:19 -08:00
const struct dmi_system_id * id ;
2005-06-25 14:54:23 -07:00
2005-04-16 15:20:36 -07:00
/*
2005-06-25 14:54:23 -07:00
* Get DMI information
2005-04-16 15:20:36 -07:00
*/
2005-06-25 14:54:25 -07:00
if ( ! dmi_check_system ( i8k_dmi_table ) ) {
if ( ! ignore_dmi & & ! force )
return - ENODEV ;
2013-12-14 09:30:08 -08:00
pr_info ( " not running on a supported Dell system. \n " ) ;
pr_info ( " vendor=%s, model=%s, version=%s \n " ,
2005-06-25 14:54:25 -07:00
i8k_get_dmi_data ( DMI_SYS_VENDOR ) ,
i8k_get_dmi_data ( DMI_PRODUCT_NAME ) ,
i8k_get_dmi_data ( DMI_BIOS_VERSION ) ) ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:23 -07:00
2013-12-14 09:30:09 -08:00
strlcpy ( bios_version , i8k_get_dmi_data ( DMI_BIOS_VERSION ) ,
sizeof ( bios_version ) ) ;
2005-06-25 14:54:25 -07:00
2005-04-16 15:20:36 -07:00
/*
2005-06-25 14:54:23 -07:00
* Get SMM Dell signature
2005-04-16 15:20:36 -07:00
*/
2005-06-25 14:54:28 -07:00
if ( i8k_get_dell_signature ( I8K_SMM_GET_DELL_SIG1 ) & &
i8k_get_dell_signature ( I8K_SMM_GET_DELL_SIG2 ) ) {
2013-12-14 09:30:08 -08:00
pr_err ( " unable to get SMM Dell signature \n " ) ;
2005-06-25 14:54:25 -07:00
if ( ! force )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2013-12-14 09:30:19 -08:00
i8k_fan_mult = fan_mult ;
2014-06-21 08:08:10 -07:00
i8k_fan_max = fan_max ? : I8K_FAN_HIGH ; /* Must not be 0 */
2013-12-14 09:30:19 -08:00
id = dmi_first_match ( i8k_dmi_table ) ;
2014-06-21 08:08:10 -07:00
if ( id & & id - > driver_data ) {
const struct i8k_config_data * conf = id - > driver_data ;
if ( fan_mult = = I8K_FAN_MULT & & conf - > fan_mult )
i8k_fan_mult = conf - > fan_mult ;
if ( fan_max = = I8K_FAN_HIGH & & conf - > fan_max )
i8k_fan_max = conf - > fan_max ;
}
i8k_pwm_mult = DIV_ROUND_UP ( 255 , i8k_fan_max ) ;
2013-12-14 09:30:19 -08:00
2005-06-25 14:54:23 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:27 -07:00
static int __init i8k_init ( void )
2005-04-16 15:20:36 -07:00
{
2005-06-25 14:54:23 -07:00
struct proc_dir_entry * proc_i8k ;
2011-05-25 20:43:33 +02:00
int err ;
2005-06-25 14:54:23 -07:00
/* Are we running on an supported laptop? */
2005-06-25 14:54:25 -07:00
if ( i8k_probe ( ) )
2005-06-25 14:54:23 -07:00
return - ENODEV ;
/* Register the proc entry */
2008-04-29 01:02:34 -07:00
proc_i8k = proc_create ( " i8k " , 0 , NULL , & i8k_fops ) ;
2005-06-25 14:54:26 -07:00
if ( ! proc_i8k )
2005-06-25 14:54:23 -07:00
return - ENOENT ;
2005-06-25 14:54:26 -07:00
2011-05-25 20:43:33 +02:00
err = i8k_init_hwmon ( ) ;
if ( err )
goto exit_remove_proc ;
2005-06-25 14:54:23 -07:00
return 0 ;
2011-05-25 20:43:33 +02:00
exit_remove_proc :
remove_proc_entry ( " i8k " , NULL ) ;
return err ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:27 -07:00
static void __exit i8k_exit ( void )
2005-04-16 15:20:36 -07:00
{
2013-12-14 09:30:10 -08:00
hwmon_device_unregister ( i8k_hwmon_dev ) ;
2005-06-25 14:54:23 -07:00
remove_proc_entry ( " i8k " , NULL ) ;
2005-04-16 15:20:36 -07:00
}
2005-06-25 14:54:27 -07:00
module_init ( i8k_init ) ;
module_exit ( i8k_exit ) ;