2009-06-21 18:53:03 +02:00
/*
* acerhdf - A driver which monitors the temperature
* of the aspire one netbook , turns on / off the fan
* as soon as the upper / lower threshold is reached .
*
* ( C ) 2009 - Peter Feuerer peter ( a ) piie . net
* http : //piie.net
2012-06-07 14:21:12 -07:00
* 2009 Borislav Petkov bp ( a ) alien8 . de
2009-06-21 18:53:03 +02:00
*
* Inspired by and many thanks to :
* o acerfand - Rachel Greenham
* o acer_ec . pl - Michael Kurz michi . kurz ( at ) googlemail . com
* - Petr Tomasek tomasek ( # ) etf , cuni , cz
* - Carlos Corbacho cathectic ( at ) gmail . com
* o lkml - Matthew Garrett
* - Borislav Petkov
* - Andreas Mohr
*
* 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 of the License , 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 .
*
* 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 . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# define pr_fmt(fmt) "acerhdf: " fmt
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/dmi.h>
2011-04-24 18:07:55 +02:00
# include <linux/acpi.h>
2009-06-21 18:53:03 +02:00
# include <linux/thermal.h>
# include <linux/platform_device.h>
/*
* The driver is started with " kernel mode off " by default . That means , the BIOS
* is still in control of the fan . In this mode the driver allows to read the
* temperature of the cpu and a userspace tool may take over control of the fan .
* If the driver is switched to " kernel mode " ( e . g . via module parameter ) the
* driver is in full control of the fan . If you want the module to be started in
* kernel mode by default , define the following :
*/
# undef START_IN_KERNEL_MODE
2014-11-28 15:20:50 +01:00
# define DRV_VER "0.7.0"
2009-06-21 18:53:03 +02:00
/*
* According to the Atom N270 datasheet ,
* ( http : //download.intel.com/design/processor/datashts/320032.pdf) the
* CPU ' s optimal operating limits denoted in junction temperature as
* measured by the on - die thermal monitor are within 0 < = Tj < = 90. So ,
* assume 89 ° C is critical temperature .
*/
2009-11-17 14:07:21 -08:00
# define ACERHDF_TEMP_CRIT 89000
2009-06-21 18:53:03 +02:00
# define ACERHDF_FAN_OFF 0
# define ACERHDF_FAN_AUTO 1
/*
* No matter what value the user puts into the fanon variable , turn on the fan
* at 80 degree Celsius to prevent hardware damage
*/
2009-11-17 14:07:21 -08:00
# define ACERHDF_MAX_FANON 80000
2009-06-21 18:53:03 +02:00
/*
* Maximum interval between two temperature checks is 15 seconds , as the die
* can get hot really fast under heavy load ( plus we shouldn ' t forget about
* possible impact of _external_ aggressive sources such as heaters , sun etc . )
*/
# define ACERHDF_MAX_INTERVAL 15
# ifdef START_IN_KERNEL_MODE
static int kernelmode = 1 ;
# else
static int kernelmode ;
# endif
static unsigned int interval = 10 ;
2012-04-25 16:01:49 -07:00
static unsigned int fanon = 60000 ;
static unsigned int fanoff = 53000 ;
2009-06-21 18:53:03 +02:00
static unsigned int verbose ;
static unsigned int fanstate = ACERHDF_FAN_AUTO ;
static char force_bios [ 16 ] ;
2009-08-06 15:57:52 -07:00
static char force_product [ 16 ] ;
2009-06-21 18:53:03 +02:00
static unsigned int prev_interval ;
2010-07-20 15:19:43 -07:00
static struct thermal_zone_device * thz_dev ;
static struct thermal_cooling_device * cl_dev ;
static struct platform_device * acerhdf_dev ;
2009-06-21 18:53:03 +02:00
module_param ( kernelmode , uint , 0 ) ;
MODULE_PARM_DESC ( kernelmode , " Kernel mode fan control on / off " ) ;
module_param ( interval , uint , 0600 ) ;
MODULE_PARM_DESC ( interval , " Polling interval of temperature check " ) ;
module_param ( fanon , uint , 0600 ) ;
MODULE_PARM_DESC ( fanon , " Turn the fan on above this temperature " ) ;
module_param ( fanoff , uint , 0600 ) ;
MODULE_PARM_DESC ( fanoff , " Turn the fan off below this temperature " ) ;
module_param ( verbose , uint , 0600 ) ;
MODULE_PARM_DESC ( verbose , " Enable verbose dmesg output " ) ;
module_param_string ( force_bios , force_bios , 16 , 0 ) ;
MODULE_PARM_DESC ( force_bios , " Force BIOS version and omit BIOS check " ) ;
2009-08-06 15:57:52 -07:00
module_param_string ( force_product , force_product , 16 , 0 ) ;
MODULE_PARM_DESC ( force_product , " Force BIOS product and omit BIOS check " ) ;
/*
2010-07-20 15:19:32 -07:00
* cmd_off : to switch the fan completely off and check if the fan is off
2009-08-06 15:57:52 -07:00
* cmd_auto : to set the BIOS in control of the fan . The BIOS regulates then
* the fan speed depending on the temperature
*/
struct fancmd {
u8 cmd_off ;
u8 cmd_auto ;
} ;
2009-06-21 18:53:03 +02:00
2014-11-28 15:20:48 +01:00
struct manualcmd {
u8 mreg ;
u8 moff ;
} ;
/* default register and command to disable fan in manual mode */
static const struct manualcmd mcmd = {
. mreg = 0x94 ,
. moff = 0xff ,
} ;
2009-06-21 18:53:03 +02:00
/* BIOS settings */
2014-11-28 15:20:52 +01:00
struct bios_settings {
2009-06-21 18:53:03 +02:00
const char * vendor ;
2009-08-06 15:57:52 -07:00
const char * product ;
2009-06-21 18:53:03 +02:00
const char * version ;
2014-11-28 15:20:52 +01:00
u8 fanreg ;
u8 tempreg ;
2009-08-06 15:57:52 -07:00
struct fancmd cmd ;
2014-11-28 15:20:48 +01:00
int mcmd_enable ;
2009-06-21 18:53:03 +02:00
} ;
/* Register addresses and values for different BIOS versions */
2014-11-28 15:20:52 +01:00
static const struct bios_settings bios_tbl [ ] = {
2009-08-06 15:57:52 -07:00
/* AOA110 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " AOA110 " , " v0.3109 " , 0x55 , 0x58 , { 0x1f , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3114 " , 0x55 , 0x58 , { 0x1f , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3301 " , 0x55 , 0x58 , { 0xaf , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3304 " , 0x55 , 0x58 , { 0xaf , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3305 " , 0x55 , 0x58 , { 0xaf , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3307 " , 0x55 , 0x58 , { 0xaf , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3308 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3309 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
{ " Acer " , " AOA110 " , " v0.3310 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
2009-08-06 15:57:52 -07:00
/* AOA150 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " AOA150 " , " v0.3114 " , 0x55 , 0x58 , { 0x1f , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3301 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3304 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3305 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3307 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3308 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3309 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AOA150 " , " v0.3310 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
2012-04-25 16:01:49 -07:00
/* LT1005u */
2014-11-28 15:20:48 +01:00
{ " Acer " , " LT-10Q " , " v0.3310 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
2009-11-17 14:27:37 -08:00
/* Acer 1410 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " Aspire 1410 " , " v0.3108 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v0.3113 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v0.3115 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v0.3117 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v0.3119 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v0.3120 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v1.3204 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v1.3303 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v1.3308 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v1.3310 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1410 " , " v1.3314 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
2009-12-21 16:20:05 -08:00
/* Acer 1810xx */
2014-11-28 15:20:48 +01:00
{ " Acer " , " Aspire 1810TZ " , " v0.3108 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3108 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v0.3113 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3113 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v0.3115 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3115 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v0.3117 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3117 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v0.3119 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3119 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v0.3120 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v0.3120 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v1.3204 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v1.3204 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v1.3303 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v1.3303 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v1.3308 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v1.3308 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v1.3310 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v1.3310 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810TZ " , " v1.3314 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1810T " , " v1.3314 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
2014-11-28 15:20:49 +01:00
/* Acer 5755G */
{ " Acer " , " Aspire 5755G " , " V1.20 " , 0xab , 0xb4 , { 0x00 , 0x08 } , 0 } ,
{ " Acer " , " Aspire 5755G " , " V1.21 " , 0xab , 0xb3 , { 0x00 , 0x08 } , 0 } ,
/* Acer 521 */
{ " Acer " , " AO521 " , " V1.11 " , 0x55 , 0x58 , { 0x1f , 0x00 } , 0 } ,
2010-07-20 15:19:33 -07:00
/* Acer 531 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " AO531h " , " v0.3104 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AO531h " , " v0.3201 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Acer " , " AO531h " , " v0.3304 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
2012-04-25 16:01:49 -07:00
/* Acer 751 */
2014-11-28 15:20:49 +01:00
{ " Acer " , " AO751h " , " V0.3206 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
2014-11-28 15:20:48 +01:00
{ " Acer " , " AO751h " , " V0.3212 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
2014-11-28 15:20:49 +01:00
/* Acer 753 */
{ " Acer " , " Aspire One 753 " , " V1.24 " , 0x93 , 0xac , { 0x14 , 0x04 } , 1 } ,
2012-04-25 16:01:49 -07:00
/* Acer 1825 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " Aspire 1825PTZ " , " V1.3118 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Acer " , " Aspire 1825PTZ " , " V1.3127 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
2014-11-28 15:20:49 +01:00
/* Acer Extensa 5420 */
{ " Acer " , " Extensa 5420 " , " V1.17 " , 0x93 , 0xac , { 0x14 , 0x04 } , 1 } ,
/* Acer Aspire 5315 */
{ " Acer " , " Aspire 5315 " , " V1.19 " , 0x93 , 0xac , { 0x14 , 0x04 } , 1 } ,
/* Acer Aspire 5739 */
{ " Acer " , " Aspire 5739G " , " V1.3311 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
2012-04-25 16:01:49 -07:00
/* Acer TravelMate 7730 */
2014-11-28 15:20:48 +01:00
{ " Acer " , " TravelMate 7730G " , " v0.3509 " , 0x55 , 0x58 , { 0xaf , 0x00 } , 0 } ,
2014-11-28 15:20:49 +01:00
/* Acer TravelMate TM8573T */
{ " Acer " , " TM8573T " , " V1.13 " , 0x93 , 0xa8 , { 0x14 , 0x04 } , 1 } ,
2009-12-21 16:20:05 -08:00
/* Gateway */
2014-11-28 15:20:48 +01:00
{ " Gateway " , " AOA110 " , " v0.3103 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
{ " Gateway " , " AOA150 " , " v0.3103 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Gateway " , " LT31 " , " v1.3103 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Gateway " , " LT31 " , " v1.3201 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Gateway " , " LT31 " , " v1.3302 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Gateway " , " LT31 " , " v1.3303t " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
2009-12-21 16:20:05 -08:00
/* Packard Bell */
2014-11-28 15:20:48 +01:00
{ " Packard Bell " , " DOA150 " , " v0.3104 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
{ " Packard Bell " , " DOA150 " , " v0.3105 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Packard Bell " , " AOA110 " , " v0.3105 " , 0x55 , 0x58 , { 0x21 , 0x00 } , 0 } ,
{ " Packard Bell " , " AOA150 " , " v0.3105 " , 0x55 , 0x58 , { 0x20 , 0x00 } , 0 } ,
{ " Packard Bell " , " ENBFT " , " V1.3118 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " ENBFT " , " V1.3127 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v1.3303 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3120 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3108 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3113 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3115 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3117 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v0.3119 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMU " , " v1.3204 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMA " , " v1.3201 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMA " , " v1.3302 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTMA " , " v1.3303t " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
{ " Packard Bell " , " DOTVR46 " , " v1.3308 " , 0x55 , 0x58 , { 0x9e , 0x00 } , 0 } ,
2009-08-06 15:57:52 -07:00
/* pewpew-terminator */
2014-11-28 15:20:48 +01:00
{ " " , " " , " " , 0 , 0 , { 0 , 0 } , 0 }
2009-06-21 18:53:03 +02:00
} ;
2014-11-28 15:20:52 +01:00
static const struct bios_settings * bios_cfg __read_mostly ;
2009-06-21 18:53:03 +02:00
2014-11-28 15:20:50 +01:00
/*
* this struct is used to instruct thermal layer to use bang_bang instead of
* default governor for acerhdf
*/
static struct thermal_zone_params acerhdf_zone_params = {
. governor_name = " bang_bang " ,
} ;
2009-06-21 18:53:03 +02:00
static int acerhdf_get_temp ( int * temp )
{
u8 read_temp ;
if ( ec_read ( bios_cfg - > tempreg , & read_temp ) )
return - EINVAL ;
2009-11-17 14:07:21 -08:00
* temp = read_temp * 1000 ;
2009-06-21 18:53:03 +02:00
return 0 ;
}
static int acerhdf_get_fanstate ( int * state )
{
u8 fan ;
if ( ec_read ( bios_cfg - > fanreg , & fan ) )
return - EINVAL ;
2010-07-20 15:19:32 -07:00
if ( fan ! = bios_cfg - > cmd . cmd_off )
2009-08-06 15:57:52 -07:00
* state = ACERHDF_FAN_AUTO ;
else
* state = ACERHDF_FAN_OFF ;
2009-06-21 18:53:03 +02:00
return 0 ;
}
static void acerhdf_change_fanstate ( int state )
{
unsigned char cmd ;
if ( verbose )
2011-11-29 11:04:06 -08:00
pr_notice ( " fan %s \n " , state = = ACERHDF_FAN_OFF ? " OFF " : " ON " ) ;
2009-06-21 18:53:03 +02:00
if ( ( state ! = ACERHDF_FAN_OFF ) & & ( state ! = ACERHDF_FAN_AUTO ) ) {
pr_err ( " invalid fan state %d requested, setting to auto! \n " ,
2011-11-29 11:04:06 -08:00
state ) ;
2009-06-21 18:53:03 +02:00
state = ACERHDF_FAN_AUTO ;
}
2009-08-06 15:57:52 -07:00
cmd = ( state = = ACERHDF_FAN_OFF ) ? bios_cfg - > cmd . cmd_off
: bios_cfg - > cmd . cmd_auto ;
2009-06-21 18:53:03 +02:00
fanstate = state ;
ec_write ( bios_cfg - > fanreg , cmd ) ;
2014-11-28 15:20:48 +01:00
if ( bios_cfg - > mcmd_enable & & state = = ACERHDF_FAN_OFF ) {
if ( verbose )
pr_notice ( " turning off fan manually \n " ) ;
ec_write ( mcmd . mreg , mcmd . moff ) ;
}
2009-06-21 18:53:03 +02:00
}
static void acerhdf_check_param ( struct thermal_zone_device * thermal )
{
if ( fanon > ACERHDF_MAX_FANON ) {
pr_err ( " fanon temperature too high, set to %d \n " ,
2011-11-29 11:04:06 -08:00
ACERHDF_MAX_FANON ) ;
2009-06-21 18:53:03 +02:00
fanon = ACERHDF_MAX_FANON ;
}
if ( kernelmode & & prev_interval ! = interval ) {
if ( interval > ACERHDF_MAX_INTERVAL ) {
pr_err ( " interval too high, set to %d \n " ,
2011-11-29 11:04:06 -08:00
ACERHDF_MAX_INTERVAL ) ;
2009-06-21 18:53:03 +02:00
interval = ACERHDF_MAX_INTERVAL ;
}
if ( verbose )
2011-11-29 11:04:06 -08:00
pr_notice ( " interval changed to: %d \n " , interval ) ;
2009-06-21 18:53:03 +02:00
thermal - > polling_delay = interval * 1000 ;
prev_interval = interval ;
}
}
/*
* This is the thermal zone callback which does the delayed polling of the fan
* state . We do check / sysfs - originating settings here in acerhdf_check_param ( )
* as late as the polling interval is since we can ' t do that in the respective
* accessors of the module parameters .
*/
2015-07-24 08:12:54 +02:00
static int acerhdf_get_ec_temp ( struct thermal_zone_device * thermal , int * t )
2009-06-21 18:53:03 +02:00
{
int temp , err = 0 ;
acerhdf_check_param ( thermal ) ;
err = acerhdf_get_temp ( & temp ) ;
if ( err )
return err ;
if ( verbose )
pr_notice ( " temp %d \n " , temp ) ;
* t = temp ;
return 0 ;
}
static int acerhdf_bind ( struct thermal_zone_device * thermal ,
struct thermal_cooling_device * cdev )
{
/* if the cooling device is the one from acerhdf bind it */
if ( cdev ! = cl_dev )
return 0 ;
2012-06-26 16:35:57 +08:00
if ( thermal_zone_bind_cooling_device ( thermal , 0 , cdev ,
2015-02-18 16:04:21 +00:00
THERMAL_NO_LIMIT , THERMAL_NO_LIMIT ,
THERMAL_WEIGHT_DEFAULT ) ) {
2009-06-21 18:53:03 +02:00
pr_err ( " error binding cooling dev \n " ) ;
return - EINVAL ;
}
return 0 ;
}
static int acerhdf_unbind ( struct thermal_zone_device * thermal ,
struct thermal_cooling_device * cdev )
{
if ( cdev ! = cl_dev )
return 0 ;
if ( thermal_zone_unbind_cooling_device ( thermal , 0 , cdev ) ) {
pr_err ( " error unbinding cooling dev \n " ) ;
return - EINVAL ;
}
return 0 ;
}
static inline void acerhdf_revert_to_bios_mode ( void )
{
acerhdf_change_fanstate ( ACERHDF_FAN_AUTO ) ;
kernelmode = 0 ;
if ( thz_dev )
thz_dev - > polling_delay = 0 ;
pr_notice ( " kernel mode fan control OFF \n " ) ;
}
static inline void acerhdf_enable_kernelmode ( void )
{
kernelmode = 1 ;
thz_dev - > polling_delay = interval * 1000 ;
thermal_zone_device_update ( thz_dev ) ;
pr_notice ( " kernel mode fan control ON \n " ) ;
}
static int acerhdf_get_mode ( struct thermal_zone_device * thermal ,
enum thermal_device_mode * mode )
{
if ( verbose )
pr_notice ( " kernel mode fan control %d \n " , kernelmode ) ;
* mode = ( kernelmode ) ? THERMAL_DEVICE_ENABLED
: THERMAL_DEVICE_DISABLED ;
return 0 ;
}
/*
* set operation mode ;
* enabled : the thermal layer of the kernel takes care about
* the temperature and the fan .
* disabled : the BIOS takes control of the fan .
*/
static int acerhdf_set_mode ( struct thermal_zone_device * thermal ,
enum thermal_device_mode mode )
{
if ( mode = = THERMAL_DEVICE_DISABLED & & kernelmode )
acerhdf_revert_to_bios_mode ( ) ;
else if ( mode = = THERMAL_DEVICE_ENABLED & & ! kernelmode )
acerhdf_enable_kernelmode ( ) ;
return 0 ;
}
static int acerhdf_get_trip_type ( struct thermal_zone_device * thermal , int trip ,
enum thermal_trip_type * type )
{
if ( trip = = 0 )
* type = THERMAL_TRIP_ACTIVE ;
2014-11-28 15:20:51 +01:00
else if ( trip = = 1 )
* type = THERMAL_TRIP_CRITICAL ;
else
return - EINVAL ;
2009-06-21 18:53:03 +02:00
return 0 ;
}
2014-11-28 15:20:50 +01:00
static int acerhdf_get_trip_hyst ( struct thermal_zone_device * thermal , int trip ,
2015-07-24 08:12:54 +02:00
int * temp )
2014-11-28 15:20:50 +01:00
{
if ( trip ! = 0 )
return - EINVAL ;
* temp = fanon - fanoff ;
2009-06-21 18:53:03 +02:00
return 0 ;
}
static int acerhdf_get_trip_temp ( struct thermal_zone_device * thermal , int trip ,
2015-07-24 08:12:54 +02:00
int * temp )
2009-06-21 18:53:03 +02:00
{
if ( trip = = 0 )
* temp = fanon ;
2014-11-28 15:20:51 +01:00
else if ( trip = = 1 )
* temp = ACERHDF_TEMP_CRIT ;
else
return - EINVAL ;
2009-06-21 18:53:03 +02:00
return 0 ;
}
static int acerhdf_get_crit_temp ( struct thermal_zone_device * thermal ,
2015-07-24 08:12:54 +02:00
int * temperature )
2009-06-21 18:53:03 +02:00
{
* temperature = ACERHDF_TEMP_CRIT ;
return 0 ;
}
/* bind callback functions to thermalzone */
2010-07-20 15:19:43 -07:00
static struct thermal_zone_device_ops acerhdf_dev_ops = {
2009-06-21 18:53:03 +02:00
. bind = acerhdf_bind ,
. unbind = acerhdf_unbind ,
. get_temp = acerhdf_get_ec_temp ,
. get_mode = acerhdf_get_mode ,
. set_mode = acerhdf_set_mode ,
. get_trip_type = acerhdf_get_trip_type ,
2014-11-28 15:20:50 +01:00
. get_trip_hyst = acerhdf_get_trip_hyst ,
2009-06-21 18:53:03 +02:00
. get_trip_temp = acerhdf_get_trip_temp ,
. get_crit_temp = acerhdf_get_crit_temp ,
} ;
/*
* cooling device callback functions
* get maximal fan cooling state
*/
static int acerhdf_get_max_state ( struct thermal_cooling_device * cdev ,
unsigned long * state )
{
* state = 1 ;
return 0 ;
}
static int acerhdf_get_cur_state ( struct thermal_cooling_device * cdev ,
unsigned long * state )
{
int err = 0 , tmp ;
err = acerhdf_get_fanstate ( & tmp ) ;
if ( err )
return err ;
* state = ( tmp = = ACERHDF_FAN_AUTO ) ? 1 : 0 ;
return 0 ;
}
/* change current fan state - is overwritten when running in kernel mode */
static int acerhdf_set_cur_state ( struct thermal_cooling_device * cdev ,
unsigned long state )
{
int cur_temp , cur_state , err = 0 ;
if ( ! kernelmode )
return 0 ;
err = acerhdf_get_temp ( & cur_temp ) ;
if ( err ) {
pr_err ( " error reading temperature, hand off control to BIOS \n " ) ;
goto err_out ;
}
err = acerhdf_get_fanstate ( & cur_state ) ;
if ( err ) {
pr_err ( " error reading fan state, hand off control to BIOS \n " ) ;
goto err_out ;
}
if ( state = = 0 ) {
2014-11-28 15:20:50 +01:00
if ( cur_state = = ACERHDF_FAN_AUTO )
2009-06-21 18:53:03 +02:00
acerhdf_change_fanstate ( ACERHDF_FAN_OFF ) ;
} else {
if ( cur_state = = ACERHDF_FAN_OFF )
acerhdf_change_fanstate ( ACERHDF_FAN_AUTO ) ;
}
return 0 ;
err_out :
acerhdf_revert_to_bios_mode ( ) ;
return - EINVAL ;
}
/* bind fan callbacks to fan device */
2010-07-20 15:19:43 -07:00
static struct thermal_cooling_device_ops acerhdf_cooling_ops = {
2009-06-21 18:53:03 +02:00
. get_max_state = acerhdf_get_max_state ,
. get_cur_state = acerhdf_get_cur_state ,
. set_cur_state = acerhdf_set_cur_state ,
} ;
/* suspend / resume functionality */
2009-09-18 12:41:05 -07:00
static int acerhdf_suspend ( struct device * dev )
2009-06-21 18:53:03 +02:00
{
if ( kernelmode )
acerhdf_change_fanstate ( ACERHDF_FAN_AUTO ) ;
if ( verbose )
pr_notice ( " going suspend \n " ) ;
return 0 ;
}
2012-12-21 13:18:33 -08:00
static int acerhdf_probe ( struct platform_device * device )
2009-06-21 18:53:03 +02:00
{
return 0 ;
}
static int acerhdf_remove ( struct platform_device * device )
{
return 0 ;
}
2009-12-14 18:00:08 -08:00
static const struct dev_pm_ops acerhdf_pm_ops = {
2009-09-18 12:41:05 -07:00
. suspend = acerhdf_suspend ,
. freeze = acerhdf_suspend ,
} ;
2009-08-06 15:57:52 -07:00
static struct platform_driver acerhdf_driver = {
2009-06-21 18:53:03 +02:00
. driver = {
2009-09-18 12:41:05 -07:00
. name = " acerhdf " ,
. pm = & acerhdf_pm_ops ,
2009-06-21 18:53:03 +02:00
} ,
. probe = acerhdf_probe ,
. remove = acerhdf_remove ,
} ;
2009-12-21 16:20:06 -08:00
/* checks if str begins with start */
static int str_starts_with ( const char * str , const char * start )
{
unsigned long str_len = 0 , start_len = 0 ;
str_len = strlen ( str ) ;
start_len = strlen ( start ) ;
if ( str_len > = start_len & &
! strncmp ( str , start , start_len ) )
return 1 ;
return 0 ;
}
2009-06-21 18:53:03 +02:00
/* check hardware */
static int acerhdf_check_hardware ( void )
{
char const * vendor , * version , * product ;
2014-11-28 15:20:52 +01:00
const struct bios_settings * bt = NULL ;
2009-06-21 18:53:03 +02:00
/* get BIOS data */
vendor = dmi_get_system_info ( DMI_SYS_VENDOR ) ;
version = dmi_get_system_info ( DMI_BIOS_VERSION ) ;
product = dmi_get_system_info ( DMI_PRODUCT_NAME ) ;
2010-07-20 15:19:33 -07:00
if ( ! vendor | | ! version | | ! product ) {
pr_err ( " error getting hardware information \n " ) ;
return - EINVAL ;
}
2009-08-06 15:57:52 -07:00
2009-06-21 18:53:03 +02:00
pr_info ( " Acer Aspire One Fan driver, v.%s \n " , DRV_VER ) ;
2009-08-06 15:57:52 -07:00
if ( force_bios [ 0 ] ) {
2009-06-21 18:53:03 +02:00
version = force_bios ;
2009-08-06 15:57:52 -07:00
pr_info ( " forcing BIOS version: %s \n " , version ) ;
2009-06-21 18:53:03 +02:00
kernelmode = 0 ;
}
2009-08-06 15:57:52 -07:00
if ( force_product [ 0 ] ) {
product = force_product ;
pr_info ( " forcing BIOS product: %s \n " , product ) ;
kernelmode = 0 ;
}
2009-06-21 18:53:03 +02:00
if ( verbose )
pr_info ( " BIOS info: %s %s, product: %s \n " ,
vendor , version , product ) ;
/* search BIOS version and vendor in BIOS settings table */
2009-12-21 16:20:06 -08:00
for ( bt = bios_tbl ; bt - > vendor [ 0 ] ; bt + + ) {
/*
* check if actual hardware BIOS vendor , product and version
* IDs start with the strings of BIOS table entry
*/
if ( str_starts_with ( vendor , bt - > vendor ) & &
str_starts_with ( product , bt - > product ) & &
str_starts_with ( version , bt - > version ) ) {
bios_cfg = bt ;
2009-06-21 18:53:03 +02:00
break ;
}
}
if ( ! bios_cfg ) {
2011-11-29 11:04:06 -08:00
pr_err ( " unknown (unsupported) BIOS version %s/%s/%s, please report, aborting! \n " ,
vendor , product , version ) ;
2009-06-21 18:53:03 +02:00
return - EINVAL ;
}
/*
* if started with kernel mode off , prevent the kernel from switching
* off the fan
*/
if ( ! kernelmode ) {
pr_notice ( " Fan control off, to enable do: \n " ) ;
2011-11-29 11:04:06 -08:00
pr_notice ( " echo -n \" enabled \" > /sys/class/thermal/thermal_zone0/mode \n " ) ;
2009-06-21 18:53:03 +02:00
}
return 0 ;
}
static int acerhdf_register_platform ( void )
{
int err = 0 ;
2009-08-06 15:57:52 -07:00
err = platform_driver_register ( & acerhdf_driver ) ;
2009-06-21 18:53:03 +02:00
if ( err )
return err ;
acerhdf_dev = platform_device_alloc ( " acerhdf " , - 1 ) ;
2010-07-20 15:19:49 -07:00
if ( ! acerhdf_dev ) {
err = - ENOMEM ;
goto err_device_alloc ;
}
err = platform_device_add ( acerhdf_dev ) ;
if ( err )
goto err_device_add ;
2009-06-21 18:53:03 +02:00
return 0 ;
2010-07-20 15:19:49 -07:00
err_device_add :
platform_device_put ( acerhdf_dev ) ;
err_device_alloc :
platform_driver_unregister ( & acerhdf_driver ) ;
return err ;
2009-06-21 18:53:03 +02:00
}
static void acerhdf_unregister_platform ( void )
{
2010-07-20 15:19:49 -07:00
platform_device_unregister ( acerhdf_dev ) ;
2009-08-06 15:57:52 -07:00
platform_driver_unregister ( & acerhdf_driver ) ;
2009-06-21 18:53:03 +02:00
}
static int acerhdf_register_thermal ( void )
{
cl_dev = thermal_cooling_device_register ( " acerhdf-fan " , NULL ,
& acerhdf_cooling_ops ) ;
if ( IS_ERR ( cl_dev ) )
return - EINVAL ;
2014-11-28 15:20:51 +01:00
thz_dev = thermal_zone_device_register ( " acerhdf " , 2 , 0 , NULL ,
2014-11-28 15:20:50 +01:00
& acerhdf_dev_ops ,
& acerhdf_zone_params , 0 ,
2009-06-21 18:53:03 +02:00
( kernelmode ) ? interval * 1000 : 0 ) ;
if ( IS_ERR ( thz_dev ) )
return - EINVAL ;
2014-11-28 15:20:50 +01:00
if ( strcmp ( thz_dev - > governor - > name ,
acerhdf_zone_params . governor_name ) ) {
pr_err ( " Didn't get thermal governor %s, perhaps not compiled into thermal subsystem. \n " ,
acerhdf_zone_params . governor_name ) ;
return - EINVAL ;
}
2009-06-21 18:53:03 +02:00
return 0 ;
}
static void acerhdf_unregister_thermal ( void )
{
if ( cl_dev ) {
thermal_cooling_device_unregister ( cl_dev ) ;
cl_dev = NULL ;
}
if ( thz_dev ) {
thermal_zone_device_unregister ( thz_dev ) ;
thz_dev = NULL ;
}
}
static int __init acerhdf_init ( void )
{
int err = 0 ;
err = acerhdf_check_hardware ( ) ;
if ( err )
goto out_err ;
err = acerhdf_register_platform ( ) ;
if ( err )
2010-07-20 15:19:49 -07:00
goto out_err ;
2009-06-21 18:53:03 +02:00
err = acerhdf_register_thermal ( ) ;
if ( err )
goto err_unreg ;
return 0 ;
err_unreg :
acerhdf_unregister_thermal ( ) ;
acerhdf_unregister_platform ( ) ;
out_err :
2010-07-20 15:19:49 -07:00
return err ;
2009-06-21 18:53:03 +02:00
}
static void __exit acerhdf_exit ( void )
{
acerhdf_change_fanstate ( ACERHDF_FAN_AUTO ) ;
acerhdf_unregister_thermal ( ) ;
acerhdf_unregister_platform ( ) ;
}
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Peter Feuerer " ) ;
MODULE_DESCRIPTION ( " Aspire One temperature and fan driver " ) ;
2009-12-21 16:20:04 -08:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAOA*: " ) ;
2012-04-25 16:01:49 -07:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAO751h*: " ) ;
2011-07-28 18:05:35 +04:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*1410*: " ) ;
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*1810*: " ) ;
2014-11-28 15:20:49 +01:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*5755G: " ) ;
2012-04-25 16:01:49 -07:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*1825PTZ: " ) ;
2014-11-28 15:20:49 +01:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAO521*: " ) ;
2010-07-20 15:19:33 -07:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAO531*: " ) ;
2014-11-28 15:20:49 +01:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*5739G: " ) ;
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*One*753: " ) ;
MODULE_ALIAS ( " dmi:*:*Acer*:pnAspire*5315: " ) ;
2012-04-25 16:01:49 -07:00
MODULE_ALIAS ( " dmi:*:*Acer*:TravelMate*7730G: " ) ;
2014-11-28 15:20:49 +01:00
MODULE_ALIAS ( " dmi:*:*Acer*:TM8573T: " ) ;
2009-12-21 16:20:04 -08:00
MODULE_ALIAS ( " dmi:*:*Gateway*:pnAOA*: " ) ;
2009-12-21 16:20:05 -08:00
MODULE_ALIAS ( " dmi:*:*Gateway*:pnLT31*: " ) ;
2011-07-28 18:05:35 +04:00
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnAOA*: " ) ;
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnDOA*: " ) ;
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnDOTMU*: " ) ;
2012-04-25 16:01:49 -07:00
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnENBFT*: " ) ;
2011-07-28 18:05:35 +04:00
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnDOTMA*: " ) ;
2012-04-25 16:01:49 -07:00
MODULE_ALIAS ( " dmi:*:*Packard*Bell*:pnDOTVR46*: " ) ;
2014-11-28 15:20:49 +01:00
MODULE_ALIAS ( " dmi:*:*Acer*:pnExtensa 5420*: " ) ;
2009-06-21 18:53:03 +02:00
module_init ( acerhdf_init ) ;
module_exit ( acerhdf_exit ) ;