acerhdf.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * acerhdf - A driver which monitors the temperature
  4. * of the aspire one netbook, turns on/off the fan
  5. * as soon as the upper/lower threshold is reached.
  6. *
  7. * (C) 2009 - Peter Kaestle peter (a) piie.net
  8. * https://piie.net
  9. * 2009 Borislav Petkov bp (a) alien8.de
  10. *
  11. * Inspired by and many thanks to:
  12. * o acerfand - Rachel Greenham
  13. * o acer_ec.pl - Michael Kurz michi.kurz (at) googlemail.com
  14. * - Petr Tomasek tomasek (#) etf,cuni,cz
  15. * - Carlos Corbacho cathectic (at) gmail.com
  16. * o lkml - Matthew Garrett
  17. * - Borislav Petkov
  18. * - Andreas Mohr
  19. */
  20. #define pr_fmt(fmt) "acerhdf: " fmt
  21. #include <linux/kernel.h>
  22. #include <linux/module.h>
  23. #include <linux/dmi.h>
  24. #include <linux/acpi.h>
  25. #include <linux/thermal.h>
  26. #include <linux/platform_device.h>
  27. /*
  28. * The driver is started with "kernel mode off" by default. That means, the BIOS
  29. * is still in control of the fan. In this mode the driver allows to read the
  30. * temperature of the cpu and a userspace tool may take over control of the fan.
  31. * If the driver is switched to "kernel mode" (e.g. via module parameter) the
  32. * driver is in full control of the fan. If you want the module to be started in
  33. * kernel mode by default, define the following:
  34. */
  35. #undef START_IN_KERNEL_MODE
  36. #define DRV_VER "0.7.0"
  37. /*
  38. * According to the Atom N270 datasheet,
  39. * (http://download.intel.com/design/processor/datashts/320032.pdf) the
  40. * CPU's optimal operating limits denoted in junction temperature as
  41. * measured by the on-die thermal monitor are within 0 <= Tj <= 90. So,
  42. * assume 89°C is critical temperature.
  43. */
  44. #define ACERHDF_TEMP_CRIT 89000
  45. #define ACERHDF_FAN_OFF 0
  46. #define ACERHDF_FAN_AUTO 1
  47. /*
  48. * No matter what value the user puts into the fanon variable, turn on the fan
  49. * at 80 degree Celsius to prevent hardware damage
  50. */
  51. #define ACERHDF_MAX_FANON 80000
  52. /*
  53. * Maximum interval between two temperature checks is 15 seconds, as the die
  54. * can get hot really fast under heavy load (plus we shouldn't forget about
  55. * possible impact of _external_ aggressive sources such as heaters, sun etc.)
  56. */
  57. #define ACERHDF_MAX_INTERVAL 15
  58. #ifdef START_IN_KERNEL_MODE
  59. static int kernelmode = 1;
  60. #else
  61. static int kernelmode;
  62. #endif
  63. static unsigned int interval = 10;
  64. static unsigned int fanon = 60000;
  65. static unsigned int fanoff = 53000;
  66. static unsigned int verbose;
  67. static unsigned int list_supported;
  68. static unsigned int fanstate = ACERHDF_FAN_AUTO;
  69. static char force_bios[16];
  70. static char force_product[16];
  71. static unsigned int prev_interval;
  72. static struct thermal_zone_device *thz_dev;
  73. static struct thermal_cooling_device *cl_dev;
  74. static struct platform_device *acerhdf_dev;
  75. module_param(kernelmode, uint, 0);
  76. MODULE_PARM_DESC(kernelmode, "Kernel mode fan control on / off");
  77. module_param(fanon, uint, 0600);
  78. MODULE_PARM_DESC(fanon, "Turn the fan on above this temperature");
  79. module_param(fanoff, uint, 0600);
  80. MODULE_PARM_DESC(fanoff, "Turn the fan off below this temperature");
  81. module_param(verbose, uint, 0600);
  82. MODULE_PARM_DESC(verbose, "Enable verbose dmesg output");
  83. module_param(list_supported, uint, 0600);
  84. MODULE_PARM_DESC(list_supported, "List supported models and BIOS versions");
  85. module_param_string(force_bios, force_bios, 16, 0);
  86. MODULE_PARM_DESC(force_bios, "Pretend system has this known supported BIOS version");
  87. module_param_string(force_product, force_product, 16, 0);
  88. MODULE_PARM_DESC(force_product, "Pretend system is this known supported model");
  89. /*
  90. * cmd_off: to switch the fan completely off and check if the fan is off
  91. * cmd_auto: to set the BIOS in control of the fan. The BIOS regulates then
  92. * the fan speed depending on the temperature
  93. */
  94. struct fancmd {
  95. u8 cmd_off;
  96. u8 cmd_auto;
  97. };
  98. struct manualcmd {
  99. u8 mreg;
  100. u8 moff;
  101. };
  102. /* default register and command to disable fan in manual mode */
  103. static const struct manualcmd mcmd = {
  104. .mreg = 0x94,
  105. .moff = 0xff,
  106. };
  107. /* BIOS settings - only used during probe */
  108. struct bios_settings {
  109. const char *vendor;
  110. const char *product;
  111. const char *version;
  112. u8 fanreg;
  113. u8 tempreg;
  114. struct fancmd cmd;
  115. int mcmd_enable;
  116. };
  117. /* This could be a daughter struct in the above, but not worth the redirect */
  118. struct ctrl_settings {
  119. u8 fanreg;
  120. u8 tempreg;
  121. struct fancmd cmd;
  122. int mcmd_enable;
  123. };
  124. static struct ctrl_settings ctrl_cfg __read_mostly;
  125. /* Register addresses and values for different BIOS versions */
  126. static const struct bios_settings bios_tbl[] __initconst = {
  127. /* AOA110 */
  128. {"Acer", "AOA110", "v0.3109", 0x55, 0x58, {0x1f, 0x00}, 0},
  129. {"Acer", "AOA110", "v0.3114", 0x55, 0x58, {0x1f, 0x00}, 0},
  130. {"Acer", "AOA110", "v0.3301", 0x55, 0x58, {0xaf, 0x00}, 0},
  131. {"Acer", "AOA110", "v0.3304", 0x55, 0x58, {0xaf, 0x00}, 0},
  132. {"Acer", "AOA110", "v0.3305", 0x55, 0x58, {0xaf, 0x00}, 0},
  133. {"Acer", "AOA110", "v0.3307", 0x55, 0x58, {0xaf, 0x00}, 0},
  134. {"Acer", "AOA110", "v0.3308", 0x55, 0x58, {0x21, 0x00}, 0},
  135. {"Acer", "AOA110", "v0.3309", 0x55, 0x58, {0x21, 0x00}, 0},
  136. {"Acer", "AOA110", "v0.3310", 0x55, 0x58, {0x21, 0x00}, 0},
  137. /* AOA150 */
  138. {"Acer", "AOA150", "v0.3114", 0x55, 0x58, {0x1f, 0x00}, 0},
  139. {"Acer", "AOA150", "v0.3301", 0x55, 0x58, {0x20, 0x00}, 0},
  140. {"Acer", "AOA150", "v0.3304", 0x55, 0x58, {0x20, 0x00}, 0},
  141. {"Acer", "AOA150", "v0.3305", 0x55, 0x58, {0x20, 0x00}, 0},
  142. {"Acer", "AOA150", "v0.3307", 0x55, 0x58, {0x20, 0x00}, 0},
  143. {"Acer", "AOA150", "v0.3308", 0x55, 0x58, {0x20, 0x00}, 0},
  144. {"Acer", "AOA150", "v0.3309", 0x55, 0x58, {0x20, 0x00}, 0},
  145. {"Acer", "AOA150", "v0.3310", 0x55, 0x58, {0x20, 0x00}, 0},
  146. /* LT1005u */
  147. {"Acer", "LT-10Q", "v0.3310", 0x55, 0x58, {0x20, 0x00}, 0},
  148. /* Acer 1410 */
  149. {"Acer", "Aspire 1410", "v0.3108", 0x55, 0x58, {0x9e, 0x00}, 0},
  150. {"Acer", "Aspire 1410", "v0.3113", 0x55, 0x58, {0x9e, 0x00}, 0},
  151. {"Acer", "Aspire 1410", "v0.3115", 0x55, 0x58, {0x9e, 0x00}, 0},
  152. {"Acer", "Aspire 1410", "v0.3117", 0x55, 0x58, {0x9e, 0x00}, 0},
  153. {"Acer", "Aspire 1410", "v0.3119", 0x55, 0x58, {0x9e, 0x00}, 0},
  154. {"Acer", "Aspire 1410", "v0.3120", 0x55, 0x58, {0x9e, 0x00}, 0},
  155. {"Acer", "Aspire 1410", "v1.3204", 0x55, 0x58, {0x9e, 0x00}, 0},
  156. {"Acer", "Aspire 1410", "v1.3303", 0x55, 0x58, {0x9e, 0x00}, 0},
  157. {"Acer", "Aspire 1410", "v1.3308", 0x55, 0x58, {0x9e, 0x00}, 0},
  158. {"Acer", "Aspire 1410", "v1.3310", 0x55, 0x58, {0x9e, 0x00}, 0},
  159. {"Acer", "Aspire 1410", "v1.3314", 0x55, 0x58, {0x9e, 0x00}, 0},
  160. /* Acer 1810xx */
  161. {"Acer", "Aspire 1810TZ", "v0.3108", 0x55, 0x58, {0x9e, 0x00}, 0},
  162. {"Acer", "Aspire 1810T", "v0.3108", 0x55, 0x58, {0x9e, 0x00}, 0},
  163. {"Acer", "Aspire 1810TZ", "v0.3113", 0x55, 0x58, {0x9e, 0x00}, 0},
  164. {"Acer", "Aspire 1810T", "v0.3113", 0x55, 0x58, {0x9e, 0x00}, 0},
  165. {"Acer", "Aspire 1810TZ", "v0.3115", 0x55, 0x58, {0x9e, 0x00}, 0},
  166. {"Acer", "Aspire 1810T", "v0.3115", 0x55, 0x58, {0x9e, 0x00}, 0},
  167. {"Acer", "Aspire 1810TZ", "v0.3117", 0x55, 0x58, {0x9e, 0x00}, 0},
  168. {"Acer", "Aspire 1810T", "v0.3117", 0x55, 0x58, {0x9e, 0x00}, 0},
  169. {"Acer", "Aspire 1810TZ", "v0.3119", 0x55, 0x58, {0x9e, 0x00}, 0},
  170. {"Acer", "Aspire 1810T", "v0.3119", 0x55, 0x58, {0x9e, 0x00}, 0},
  171. {"Acer", "Aspire 1810TZ", "v0.3120", 0x55, 0x58, {0x9e, 0x00}, 0},
  172. {"Acer", "Aspire 1810T", "v0.3120", 0x55, 0x58, {0x9e, 0x00}, 0},
  173. {"Acer", "Aspire 1810TZ", "v1.3204", 0x55, 0x58, {0x9e, 0x00}, 0},
  174. {"Acer", "Aspire 1810T", "v1.3204", 0x55, 0x58, {0x9e, 0x00}, 0},
  175. {"Acer", "Aspire 1810TZ", "v1.3303", 0x55, 0x58, {0x9e, 0x00}, 0},
  176. {"Acer", "Aspire 1810T", "v1.3303", 0x55, 0x58, {0x9e, 0x00}, 0},
  177. {"Acer", "Aspire 1810TZ", "v1.3308", 0x55, 0x58, {0x9e, 0x00}, 0},
  178. {"Acer", "Aspire 1810T", "v1.3308", 0x55, 0x58, {0x9e, 0x00}, 0},
  179. {"Acer", "Aspire 1810TZ", "v1.3310", 0x55, 0x58, {0x9e, 0x00}, 0},
  180. {"Acer", "Aspire 1810T", "v1.3310", 0x55, 0x58, {0x9e, 0x00}, 0},
  181. {"Acer", "Aspire 1810TZ", "v1.3314", 0x55, 0x58, {0x9e, 0x00}, 0},
  182. {"Acer", "Aspire 1810T", "v1.3314", 0x55, 0x58, {0x9e, 0x00}, 0},
  183. /* Acer 5755G */
  184. {"Acer", "Aspire 5755G", "V1.20", 0xab, 0xb4, {0x00, 0x08}, 0},
  185. {"Acer", "Aspire 5755G", "V1.21", 0xab, 0xb3, {0x00, 0x08}, 0},
  186. /* Acer 521 */
  187. {"Acer", "AO521", "V1.11", 0x55, 0x58, {0x1f, 0x00}, 0},
  188. /* Acer 531 */
  189. {"Acer", "AO531h", "v0.3104", 0x55, 0x58, {0x20, 0x00}, 0},
  190. {"Acer", "AO531h", "v0.3201", 0x55, 0x58, {0x20, 0x00}, 0},
  191. {"Acer", "AO531h", "v0.3304", 0x55, 0x58, {0x20, 0x00}, 0},
  192. /* Acer 751 */
  193. {"Acer", "AO751h", "V0.3206", 0x55, 0x58, {0x21, 0x00}, 0},
  194. {"Acer", "AO751h", "V0.3212", 0x55, 0x58, {0x21, 0x00}, 0},
  195. /* Acer 753 */
  196. {"Acer", "Aspire One 753", "V1.24", 0x93, 0xac, {0x14, 0x04}, 1},
  197. /* Acer 1825 */
  198. {"Acer", "Aspire 1825PTZ", "V1.3118", 0x55, 0x58, {0x9e, 0x00}, 0},
  199. {"Acer", "Aspire 1825PTZ", "V1.3127", 0x55, 0x58, {0x9e, 0x00}, 0},
  200. /* Acer Extensa 5420 */
  201. {"Acer", "Extensa 5420", "V1.17", 0x93, 0xac, {0x14, 0x04}, 1},
  202. /* Acer Aspire 5315 */
  203. {"Acer", "Aspire 5315", "V1.19", 0x93, 0xac, {0x14, 0x04}, 1},
  204. /* Acer Aspire 5739 */
  205. {"Acer", "Aspire 5739G", "V1.3311", 0x55, 0x58, {0x20, 0x00}, 0},
  206. /* Acer TravelMate 7730 */
  207. {"Acer", "TravelMate 7730G", "v0.3509", 0x55, 0x58, {0xaf, 0x00}, 0},
  208. /* Acer Aspire 7551 */
  209. {"Acer", "Aspire 7551", "V1.18", 0x93, 0xa8, {0x14, 0x04}, 1},
  210. /* Acer TravelMate TM8573T */
  211. {"Acer", "TM8573T", "V1.13", 0x93, 0xa8, {0x14, 0x04}, 1},
  212. /* Gateway */
  213. {"Gateway", "AOA110", "v0.3103", 0x55, 0x58, {0x21, 0x00}, 0},
  214. {"Gateway", "AOA150", "v0.3103", 0x55, 0x58, {0x20, 0x00}, 0},
  215. {"Gateway", "LT31", "v1.3103", 0x55, 0x58, {0x9e, 0x00}, 0},
  216. {"Gateway", "LT31", "v1.3201", 0x55, 0x58, {0x9e, 0x00}, 0},
  217. {"Gateway", "LT31", "v1.3302", 0x55, 0x58, {0x9e, 0x00}, 0},
  218. {"Gateway", "LT31", "v1.3303t", 0x55, 0x58, {0x9e, 0x00}, 0},
  219. {"Gateway", "LT31", "v1.3307", 0x55, 0x58, {0x9e, 0x00}, 0},
  220. /* Packard Bell */
  221. {"Packard Bell", "DOA150", "v0.3104", 0x55, 0x58, {0x21, 0x00}, 0},
  222. {"Packard Bell", "DOA150", "v0.3105", 0x55, 0x58, {0x20, 0x00}, 0},
  223. {"Packard Bell", "AOA110", "v0.3105", 0x55, 0x58, {0x21, 0x00}, 0},
  224. {"Packard Bell", "AOA150", "v0.3105", 0x55, 0x58, {0x20, 0x00}, 0},
  225. {"Packard Bell", "ENBFT", "V1.3118", 0x55, 0x58, {0x9e, 0x00}, 0},
  226. {"Packard Bell", "ENBFT", "V1.3127", 0x55, 0x58, {0x9e, 0x00}, 0},
  227. {"Packard Bell", "DOTMU", "v1.3303", 0x55, 0x58, {0x9e, 0x00}, 0},
  228. {"Packard Bell", "DOTMU", "v0.3120", 0x55, 0x58, {0x9e, 0x00}, 0},
  229. {"Packard Bell", "DOTMU", "v0.3108", 0x55, 0x58, {0x9e, 0x00}, 0},
  230. {"Packard Bell", "DOTMU", "v0.3113", 0x55, 0x58, {0x9e, 0x00}, 0},
  231. {"Packard Bell", "DOTMU", "v0.3115", 0x55, 0x58, {0x9e, 0x00}, 0},
  232. {"Packard Bell", "DOTMU", "v0.3117", 0x55, 0x58, {0x9e, 0x00}, 0},
  233. {"Packard Bell", "DOTMU", "v0.3119", 0x55, 0x58, {0x9e, 0x00}, 0},
  234. {"Packard Bell", "DOTMU", "v1.3204", 0x55, 0x58, {0x9e, 0x00}, 0},
  235. {"Packard Bell", "DOTMA", "v1.3201", 0x55, 0x58, {0x9e, 0x00}, 0},
  236. {"Packard Bell", "DOTMA", "v1.3302", 0x55, 0x58, {0x9e, 0x00}, 0},
  237. {"Packard Bell", "DOTMA", "v1.3303t", 0x55, 0x58, {0x9e, 0x00}, 0},
  238. {"Packard Bell", "DOTVR46", "v1.3308", 0x55, 0x58, {0x9e, 0x00}, 0},
  239. /* pewpew-terminator */
  240. {"", "", "", 0, 0, {0, 0}, 0}
  241. };
  242. /*
  243. * this struct is used to instruct thermal layer to use bang_bang instead of
  244. * default governor for acerhdf
  245. */
  246. static struct thermal_zone_params acerhdf_zone_params = {
  247. .governor_name = "bang_bang",
  248. };
  249. static int acerhdf_get_temp(int *temp)
  250. {
  251. u8 read_temp;
  252. if (ec_read(ctrl_cfg.tempreg, &read_temp))
  253. return -EINVAL;
  254. *temp = read_temp * 1000;
  255. return 0;
  256. }
  257. static int acerhdf_get_fanstate(int *state)
  258. {
  259. u8 fan;
  260. if (ec_read(ctrl_cfg.fanreg, &fan))
  261. return -EINVAL;
  262. if (fan != ctrl_cfg.cmd.cmd_off)
  263. *state = ACERHDF_FAN_AUTO;
  264. else
  265. *state = ACERHDF_FAN_OFF;
  266. return 0;
  267. }
  268. static void acerhdf_change_fanstate(int state)
  269. {
  270. unsigned char cmd;
  271. if (verbose)
  272. pr_notice("fan %s\n", state == ACERHDF_FAN_OFF ? "OFF" : "ON");
  273. if ((state != ACERHDF_FAN_OFF) && (state != ACERHDF_FAN_AUTO)) {
  274. pr_err("invalid fan state %d requested, setting to auto!\n",
  275. state);
  276. state = ACERHDF_FAN_AUTO;
  277. }
  278. cmd = (state == ACERHDF_FAN_OFF) ? ctrl_cfg.cmd.cmd_off
  279. : ctrl_cfg.cmd.cmd_auto;
  280. fanstate = state;
  281. ec_write(ctrl_cfg.fanreg, cmd);
  282. if (ctrl_cfg.mcmd_enable && state == ACERHDF_FAN_OFF) {
  283. if (verbose)
  284. pr_notice("turning off fan manually\n");
  285. ec_write(mcmd.mreg, mcmd.moff);
  286. }
  287. }
  288. static void acerhdf_check_param(struct thermal_zone_device *thermal)
  289. {
  290. if (fanon > ACERHDF_MAX_FANON) {
  291. pr_err("fanon temperature too high, set to %d\n",
  292. ACERHDF_MAX_FANON);
  293. fanon = ACERHDF_MAX_FANON;
  294. }
  295. if (kernelmode && prev_interval != interval) {
  296. if (interval > ACERHDF_MAX_INTERVAL) {
  297. pr_err("interval too high, set to %d\n",
  298. ACERHDF_MAX_INTERVAL);
  299. interval = ACERHDF_MAX_INTERVAL;
  300. }
  301. if (verbose)
  302. pr_notice("interval changed to: %d\n", interval);
  303. if (thermal)
  304. thermal->polling_delay_jiffies =
  305. round_jiffies(msecs_to_jiffies(interval * 1000));
  306. prev_interval = interval;
  307. }
  308. }
  309. /*
  310. * This is the thermal zone callback which does the delayed polling of the fan
  311. * state. We do check /sysfs-originating settings here in acerhdf_check_param()
  312. * as late as the polling interval is since we can't do that in the respective
  313. * accessors of the module parameters.
  314. */
  315. static int acerhdf_get_ec_temp(struct thermal_zone_device *thermal, int *t)
  316. {
  317. int temp, err = 0;
  318. err = acerhdf_get_temp(&temp);
  319. if (err)
  320. return err;
  321. if (verbose)
  322. pr_notice("temp %d\n", temp);
  323. *t = temp;
  324. return 0;
  325. }
  326. static int acerhdf_bind(struct thermal_zone_device *thermal,
  327. struct thermal_cooling_device *cdev)
  328. {
  329. /* if the cooling device is the one from acerhdf bind it */
  330. if (cdev != cl_dev)
  331. return 0;
  332. if (thermal_zone_bind_cooling_device(thermal, 0, cdev,
  333. THERMAL_NO_LIMIT, THERMAL_NO_LIMIT,
  334. THERMAL_WEIGHT_DEFAULT)) {
  335. pr_err("error binding cooling dev\n");
  336. return -EINVAL;
  337. }
  338. return 0;
  339. }
  340. static int acerhdf_unbind(struct thermal_zone_device *thermal,
  341. struct thermal_cooling_device *cdev)
  342. {
  343. if (cdev != cl_dev)
  344. return 0;
  345. if (thermal_zone_unbind_cooling_device(thermal, 0, cdev)) {
  346. pr_err("error unbinding cooling dev\n");
  347. return -EINVAL;
  348. }
  349. return 0;
  350. }
  351. static inline void acerhdf_revert_to_bios_mode(void)
  352. {
  353. acerhdf_change_fanstate(ACERHDF_FAN_AUTO);
  354. kernelmode = 0;
  355. pr_notice("kernel mode fan control OFF\n");
  356. }
  357. static inline void acerhdf_enable_kernelmode(void)
  358. {
  359. kernelmode = 1;
  360. pr_notice("kernel mode fan control ON\n");
  361. }
  362. /*
  363. * set operation mode;
  364. * enabled: the thermal layer of the kernel takes care about
  365. * the temperature and the fan.
  366. * disabled: the BIOS takes control of the fan.
  367. */
  368. static int acerhdf_change_mode(struct thermal_zone_device *thermal,
  369. enum thermal_device_mode mode)
  370. {
  371. if (mode == THERMAL_DEVICE_DISABLED && kernelmode)
  372. acerhdf_revert_to_bios_mode();
  373. else if (mode == THERMAL_DEVICE_ENABLED && !kernelmode)
  374. acerhdf_enable_kernelmode();
  375. return 0;
  376. }
  377. static int acerhdf_get_trip_type(struct thermal_zone_device *thermal, int trip,
  378. enum thermal_trip_type *type)
  379. {
  380. if (trip == 0)
  381. *type = THERMAL_TRIP_ACTIVE;
  382. else if (trip == 1)
  383. *type = THERMAL_TRIP_CRITICAL;
  384. else
  385. return -EINVAL;
  386. return 0;
  387. }
  388. static int acerhdf_get_trip_hyst(struct thermal_zone_device *thermal, int trip,
  389. int *temp)
  390. {
  391. if (trip != 0)
  392. return -EINVAL;
  393. *temp = fanon - fanoff;
  394. return 0;
  395. }
  396. static int acerhdf_get_trip_temp(struct thermal_zone_device *thermal, int trip,
  397. int *temp)
  398. {
  399. if (trip == 0)
  400. *temp = fanon;
  401. else if (trip == 1)
  402. *temp = ACERHDF_TEMP_CRIT;
  403. else
  404. return -EINVAL;
  405. return 0;
  406. }
  407. static int acerhdf_get_crit_temp(struct thermal_zone_device *thermal,
  408. int *temperature)
  409. {
  410. *temperature = ACERHDF_TEMP_CRIT;
  411. return 0;
  412. }
  413. /* bind callback functions to thermalzone */
  414. static struct thermal_zone_device_ops acerhdf_dev_ops = {
  415. .bind = acerhdf_bind,
  416. .unbind = acerhdf_unbind,
  417. .get_temp = acerhdf_get_ec_temp,
  418. .change_mode = acerhdf_change_mode,
  419. .get_trip_type = acerhdf_get_trip_type,
  420. .get_trip_hyst = acerhdf_get_trip_hyst,
  421. .get_trip_temp = acerhdf_get_trip_temp,
  422. .get_crit_temp = acerhdf_get_crit_temp,
  423. };
  424. /*
  425. * cooling device callback functions
  426. * get maximal fan cooling state
  427. */
  428. static int acerhdf_get_max_state(struct thermal_cooling_device *cdev,
  429. unsigned long *state)
  430. {
  431. *state = 1;
  432. return 0;
  433. }
  434. static int acerhdf_get_cur_state(struct thermal_cooling_device *cdev,
  435. unsigned long *state)
  436. {
  437. int err = 0, tmp;
  438. err = acerhdf_get_fanstate(&tmp);
  439. if (err)
  440. return err;
  441. *state = (tmp == ACERHDF_FAN_AUTO) ? 1 : 0;
  442. return 0;
  443. }
  444. /* change current fan state - is overwritten when running in kernel mode */
  445. static int acerhdf_set_cur_state(struct thermal_cooling_device *cdev,
  446. unsigned long state)
  447. {
  448. int cur_temp, cur_state, err = 0;
  449. if (!kernelmode)
  450. return 0;
  451. err = acerhdf_get_temp(&cur_temp);
  452. if (err) {
  453. pr_err("error reading temperature, hand off control to BIOS\n");
  454. goto err_out;
  455. }
  456. err = acerhdf_get_fanstate(&cur_state);
  457. if (err) {
  458. pr_err("error reading fan state, hand off control to BIOS\n");
  459. goto err_out;
  460. }
  461. if (state == 0) {
  462. if (cur_state == ACERHDF_FAN_AUTO)
  463. acerhdf_change_fanstate(ACERHDF_FAN_OFF);
  464. } else {
  465. if (cur_state == ACERHDF_FAN_OFF)
  466. acerhdf_change_fanstate(ACERHDF_FAN_AUTO);
  467. }
  468. return 0;
  469. err_out:
  470. acerhdf_revert_to_bios_mode();
  471. return -EINVAL;
  472. }
  473. /* bind fan callbacks to fan device */
  474. static const struct thermal_cooling_device_ops acerhdf_cooling_ops = {
  475. .get_max_state = acerhdf_get_max_state,
  476. .get_cur_state = acerhdf_get_cur_state,
  477. .set_cur_state = acerhdf_set_cur_state,
  478. };
  479. /* suspend / resume functionality */
  480. static int acerhdf_suspend(struct device *dev)
  481. {
  482. if (kernelmode)
  483. acerhdf_change_fanstate(ACERHDF_FAN_AUTO);
  484. if (verbose)
  485. pr_notice("going suspend\n");
  486. return 0;
  487. }
  488. static int acerhdf_probe(struct platform_device *device)
  489. {
  490. return 0;
  491. }
  492. static int acerhdf_remove(struct platform_device *device)
  493. {
  494. return 0;
  495. }
  496. static const struct dev_pm_ops acerhdf_pm_ops = {
  497. .suspend = acerhdf_suspend,
  498. .freeze = acerhdf_suspend,
  499. };
  500. static struct platform_driver acerhdf_driver = {
  501. .driver = {
  502. .name = "acerhdf",
  503. .pm = &acerhdf_pm_ops,
  504. },
  505. .probe = acerhdf_probe,
  506. .remove = acerhdf_remove,
  507. };
  508. /* check hardware */
  509. static int __init acerhdf_check_hardware(void)
  510. {
  511. char const *vendor, *version, *product;
  512. const struct bios_settings *bt = NULL;
  513. int found = 0;
  514. /* get BIOS data */
  515. vendor = dmi_get_system_info(DMI_SYS_VENDOR);
  516. version = dmi_get_system_info(DMI_BIOS_VERSION);
  517. product = dmi_get_system_info(DMI_PRODUCT_NAME);
  518. if (!vendor || !version || !product) {
  519. pr_err("error getting hardware information\n");
  520. return -EINVAL;
  521. }
  522. pr_info("Acer Aspire One Fan driver, v.%s\n", DRV_VER);
  523. if (list_supported) {
  524. pr_info("List of supported Manufacturer/Model/BIOS:\n");
  525. pr_info("---------------------------------------------------\n");
  526. for (bt = bios_tbl; bt->vendor[0]; bt++) {
  527. pr_info("%-13s | %-17s | %-10s\n", bt->vendor,
  528. bt->product, bt->version);
  529. }
  530. pr_info("---------------------------------------------------\n");
  531. return -ECANCELED;
  532. }
  533. if (force_bios[0]) {
  534. version = force_bios;
  535. pr_info("forcing BIOS version: %s\n", version);
  536. kernelmode = 0;
  537. }
  538. if (force_product[0]) {
  539. product = force_product;
  540. pr_info("forcing BIOS product: %s\n", product);
  541. kernelmode = 0;
  542. }
  543. if (verbose)
  544. pr_info("BIOS info: %s %s, product: %s\n",
  545. vendor, version, product);
  546. /* search BIOS version and vendor in BIOS settings table */
  547. for (bt = bios_tbl; bt->vendor[0]; bt++) {
  548. /*
  549. * check if actual hardware BIOS vendor, product and version
  550. * IDs start with the strings of BIOS table entry
  551. */
  552. if (strstarts(vendor, bt->vendor) &&
  553. strstarts(product, bt->product) &&
  554. strstarts(version, bt->version)) {
  555. found = 1;
  556. break;
  557. }
  558. }
  559. if (!found) {
  560. pr_err("unknown (unsupported) BIOS version %s/%s/%s, please report, aborting!\n",
  561. vendor, product, version);
  562. return -EINVAL;
  563. }
  564. /* Copy control settings from BIOS table before we free it. */
  565. ctrl_cfg.fanreg = bt->fanreg;
  566. ctrl_cfg.tempreg = bt->tempreg;
  567. memcpy(&ctrl_cfg.cmd, &bt->cmd, sizeof(struct fancmd));
  568. ctrl_cfg.mcmd_enable = bt->mcmd_enable;
  569. /*
  570. * if started with kernel mode off, prevent the kernel from switching
  571. * off the fan
  572. */
  573. if (!kernelmode) {
  574. pr_notice("Fan control off, to enable do:\n");
  575. pr_notice("echo -n \"enabled\" > /sys/class/thermal/thermal_zoneN/mode # N=0,1,2...\n");
  576. }
  577. return 0;
  578. }
  579. static int __init acerhdf_register_platform(void)
  580. {
  581. int err = 0;
  582. err = platform_driver_register(&acerhdf_driver);
  583. if (err)
  584. return err;
  585. acerhdf_dev = platform_device_alloc("acerhdf", PLATFORM_DEVID_NONE);
  586. if (!acerhdf_dev) {
  587. err = -ENOMEM;
  588. goto err_device_alloc;
  589. }
  590. err = platform_device_add(acerhdf_dev);
  591. if (err)
  592. goto err_device_add;
  593. return 0;
  594. err_device_add:
  595. platform_device_put(acerhdf_dev);
  596. err_device_alloc:
  597. platform_driver_unregister(&acerhdf_driver);
  598. return err;
  599. }
  600. static void acerhdf_unregister_platform(void)
  601. {
  602. platform_device_unregister(acerhdf_dev);
  603. platform_driver_unregister(&acerhdf_driver);
  604. }
  605. static int __init acerhdf_register_thermal(void)
  606. {
  607. int ret;
  608. cl_dev = thermal_cooling_device_register("acerhdf-fan", NULL,
  609. &acerhdf_cooling_ops);
  610. if (IS_ERR(cl_dev))
  611. return -EINVAL;
  612. thz_dev = thermal_zone_device_register("acerhdf", 2, 0, NULL,
  613. &acerhdf_dev_ops,
  614. &acerhdf_zone_params, 0,
  615. (kernelmode) ? interval*1000 : 0);
  616. if (IS_ERR(thz_dev))
  617. return -EINVAL;
  618. if (kernelmode)
  619. ret = thermal_zone_device_enable(thz_dev);
  620. else
  621. ret = thermal_zone_device_disable(thz_dev);
  622. if (ret)
  623. return ret;
  624. if (strcmp(thz_dev->governor->name,
  625. acerhdf_zone_params.governor_name)) {
  626. pr_err("Didn't get thermal governor %s, perhaps not compiled into thermal subsystem.\n",
  627. acerhdf_zone_params.governor_name);
  628. return -EINVAL;
  629. }
  630. return 0;
  631. }
  632. static void acerhdf_unregister_thermal(void)
  633. {
  634. if (cl_dev) {
  635. thermal_cooling_device_unregister(cl_dev);
  636. cl_dev = NULL;
  637. }
  638. if (thz_dev) {
  639. thermal_zone_device_unregister(thz_dev);
  640. thz_dev = NULL;
  641. }
  642. }
  643. static int __init acerhdf_init(void)
  644. {
  645. int err = 0;
  646. err = acerhdf_check_hardware();
  647. if (err)
  648. goto out_err;
  649. err = acerhdf_register_platform();
  650. if (err)
  651. goto out_err;
  652. err = acerhdf_register_thermal();
  653. if (err)
  654. goto err_unreg;
  655. return 0;
  656. err_unreg:
  657. acerhdf_unregister_thermal();
  658. acerhdf_unregister_platform();
  659. out_err:
  660. return err;
  661. }
  662. static void __exit acerhdf_exit(void)
  663. {
  664. acerhdf_change_fanstate(ACERHDF_FAN_AUTO);
  665. acerhdf_unregister_thermal();
  666. acerhdf_unregister_platform();
  667. }
  668. MODULE_LICENSE("GPL");
  669. MODULE_AUTHOR("Peter Kaestle");
  670. MODULE_DESCRIPTION("Aspire One temperature and fan driver");
  671. MODULE_ALIAS("dmi:*:*Acer*:pnAOA*:");
  672. MODULE_ALIAS("dmi:*:*Acer*:pnAO751h*:");
  673. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*1410*:");
  674. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*1810*:");
  675. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*5755G:");
  676. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*1825PTZ:");
  677. MODULE_ALIAS("dmi:*:*Acer*:pnAO521*:");
  678. MODULE_ALIAS("dmi:*:*Acer*:pnAO531*:");
  679. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*5739G:");
  680. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*One*753:");
  681. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*5315:");
  682. MODULE_ALIAS("dmi:*:*Acer*:TravelMate*7730G:");
  683. MODULE_ALIAS("dmi:*:*Acer*:pnAspire*7551:");
  684. MODULE_ALIAS("dmi:*:*Acer*:TM8573T:");
  685. MODULE_ALIAS("dmi:*:*Gateway*:pnAOA*:");
  686. MODULE_ALIAS("dmi:*:*Gateway*:pnLT31*:");
  687. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnAOA*:");
  688. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnDOA*:");
  689. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnDOTMU*:");
  690. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnENBFT*:");
  691. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnDOTMA*:");
  692. MODULE_ALIAS("dmi:*:*Packard*Bell*:pnDOTVR46*:");
  693. MODULE_ALIAS("dmi:*:*Acer*:pnExtensa*5420*:");
  694. module_init(acerhdf_init);
  695. module_exit(acerhdf_exit);
  696. static int interval_set_uint(const char *val, const struct kernel_param *kp)
  697. {
  698. int ret;
  699. ret = param_set_uint(val, kp);
  700. if (ret)
  701. return ret;
  702. acerhdf_check_param(thz_dev);
  703. return 0;
  704. }
  705. static const struct kernel_param_ops interval_ops = {
  706. .set = interval_set_uint,
  707. .get = param_get_uint,
  708. };
  709. module_param_cb(interval, &interval_ops, &interval, 0600);
  710. MODULE_PARM_DESC(interval, "Polling interval of temperature check");