pc87360.c 53 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643
  1. /*
  2. * pc87360.c - Part of lm_sensors, Linux kernel modules
  3. * for hardware monitoring
  4. * Copyright (C) 2004, 2007 Jean Delvare <khali@linux-fr.org>
  5. *
  6. * Copied from smsc47m1.c:
  7. * Copyright (C) 2002 Mark D. Studebaker <mdsxyz123@yahoo.com>
  8. *
  9. * This program is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License as published by
  11. * the Free Software Foundation; either version 2 of the License, or
  12. * (at your option) any later version.
  13. *
  14. * This program is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. * GNU General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU General Public License
  20. * along with this program; if not, write to the Free Software
  21. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  22. *
  23. * Supports the following chips:
  24. *
  25. * Chip #vin #fan #pwm #temp devid
  26. * PC87360 - 2 2 - 0xE1
  27. * PC87363 - 2 2 - 0xE8
  28. * PC87364 - 3 3 - 0xE4
  29. * PC87365 11 3 3 2 0xE5
  30. * PC87366 11 3 3 3-4 0xE9
  31. *
  32. * This driver assumes that no more than one chip is present, and one of
  33. * the standard Super-I/O addresses is used (0x2E/0x2F or 0x4E/0x4F).
  34. */
  35. #include <linux/module.h>
  36. #include <linux/init.h>
  37. #include <linux/slab.h>
  38. #include <linux/jiffies.h>
  39. #include <linux/platform_device.h>
  40. #include <linux/hwmon.h>
  41. #include <linux/hwmon-sysfs.h>
  42. #include <linux/hwmon-vid.h>
  43. #include <linux/err.h>
  44. #include <linux/mutex.h>
  45. #include <asm/io.h>
  46. static u8 devid;
  47. static struct platform_device *pdev;
  48. static unsigned short extra_isa[3];
  49. static u8 confreg[4];
  50. static int init = 1;
  51. module_param(init, int, 0);
  52. MODULE_PARM_DESC(init,
  53. "Chip initialization level:\n"
  54. " 0: None\n"
  55. "*1: Forcibly enable internal voltage and temperature channels, except in9\n"
  56. " 2: Forcibly enable all voltage and temperature channels, except in9\n"
  57. " 3: Forcibly enable all voltage and temperature channels, including in9");
  58. static unsigned short force_id;
  59. module_param(force_id, ushort, 0);
  60. MODULE_PARM_DESC(force_id, "Override the detected device ID");
  61. /*
  62. * Super-I/O registers and operations
  63. */
  64. #define DEV 0x07 /* Register: Logical device select */
  65. #define DEVID 0x20 /* Register: Device ID */
  66. #define ACT 0x30 /* Register: Device activation */
  67. #define BASE 0x60 /* Register: Base address */
  68. #define FSCM 0x09 /* Logical device: fans */
  69. #define VLM 0x0d /* Logical device: voltages */
  70. #define TMS 0x0e /* Logical device: temperatures */
  71. static const u8 logdev[3] = { FSCM, VLM, TMS };
  72. #define LD_FAN 0
  73. #define LD_IN 1
  74. #define LD_TEMP 2
  75. static inline void superio_outb(int sioaddr, int reg, int val)
  76. {
  77. outb(reg, sioaddr);
  78. outb(val, sioaddr+1);
  79. }
  80. static inline int superio_inb(int sioaddr, int reg)
  81. {
  82. outb(reg, sioaddr);
  83. return inb(sioaddr+1);
  84. }
  85. static inline void superio_exit(int sioaddr)
  86. {
  87. outb(0x02, sioaddr);
  88. outb(0x02, sioaddr+1);
  89. }
  90. /*
  91. * Logical devices
  92. */
  93. #define PC87360_EXTENT 0x10
  94. #define PC87365_REG_BANK 0x09
  95. #define NO_BANK 0xff
  96. /*
  97. * Fan registers and conversions
  98. */
  99. /* nr has to be 0 or 1 (PC87360/87363) or 2 (PC87364/87365/87366) */
  100. #define PC87360_REG_PRESCALE(nr) (0x00 + 2 * (nr))
  101. #define PC87360_REG_PWM(nr) (0x01 + 2 * (nr))
  102. #define PC87360_REG_FAN_MIN(nr) (0x06 + 3 * (nr))
  103. #define PC87360_REG_FAN(nr) (0x07 + 3 * (nr))
  104. #define PC87360_REG_FAN_STATUS(nr) (0x08 + 3 * (nr))
  105. #define FAN_FROM_REG(val,div) ((val) == 0 ? 0: \
  106. 480000 / ((val)*(div)))
  107. #define FAN_TO_REG(val,div) ((val) <= 100 ? 0 : \
  108. 480000 / ((val)*(div)))
  109. #define FAN_DIV_FROM_REG(val) (1 << ((val >> 5) & 0x03))
  110. #define FAN_STATUS_FROM_REG(val) ((val) & 0x07)
  111. #define FAN_CONFIG_MONITOR(val,nr) (((val) >> (2 + nr * 3)) & 1)
  112. #define FAN_CONFIG_CONTROL(val,nr) (((val) >> (3 + nr * 3)) & 1)
  113. #define FAN_CONFIG_INVERT(val,nr) (((val) >> (4 + nr * 3)) & 1)
  114. #define PWM_FROM_REG(val,inv) ((inv) ? 255 - (val) : (val))
  115. static inline u8 PWM_TO_REG(int val, int inv)
  116. {
  117. if (inv)
  118. val = 255 - val;
  119. if (val < 0)
  120. return 0;
  121. if (val > 255)
  122. return 255;
  123. return val;
  124. }
  125. /*
  126. * Voltage registers and conversions
  127. */
  128. #define PC87365_REG_IN_CONVRATE 0x07
  129. #define PC87365_REG_IN_CONFIG 0x08
  130. #define PC87365_REG_IN 0x0B
  131. #define PC87365_REG_IN_MIN 0x0D
  132. #define PC87365_REG_IN_MAX 0x0C
  133. #define PC87365_REG_IN_STATUS 0x0A
  134. #define PC87365_REG_IN_ALARMS1 0x00
  135. #define PC87365_REG_IN_ALARMS2 0x01
  136. #define PC87365_REG_VID 0x06
  137. #define IN_FROM_REG(val,ref) (((val) * (ref) + 128) / 256)
  138. #define IN_TO_REG(val,ref) ((val) < 0 ? 0 : \
  139. (val)*256 >= (ref)*255 ? 255: \
  140. ((val) * 256 + (ref)/2) / (ref))
  141. /*
  142. * Temperature registers and conversions
  143. */
  144. #define PC87365_REG_TEMP_CONFIG 0x08
  145. #define PC87365_REG_TEMP 0x0B
  146. #define PC87365_REG_TEMP_MIN 0x0D
  147. #define PC87365_REG_TEMP_MAX 0x0C
  148. #define PC87365_REG_TEMP_CRIT 0x0E
  149. #define PC87365_REG_TEMP_STATUS 0x0A
  150. #define PC87365_REG_TEMP_ALARMS 0x00
  151. #define TEMP_FROM_REG(val) ((val) * 1000)
  152. #define TEMP_TO_REG(val) ((val) < -55000 ? -55 : \
  153. (val) > 127000 ? 127 : \
  154. (val) < 0 ? ((val) - 500) / 1000 : \
  155. ((val) + 500) / 1000)
  156. /*
  157. * Device data
  158. */
  159. struct pc87360_data {
  160. const char *name;
  161. struct device *hwmon_dev;
  162. struct mutex lock;
  163. struct mutex update_lock;
  164. char valid; /* !=0 if following fields are valid */
  165. unsigned long last_updated; /* In jiffies */
  166. int address[3];
  167. u8 fannr, innr, tempnr;
  168. u8 fan[3]; /* Register value */
  169. u8 fan_min[3]; /* Register value */
  170. u8 fan_status[3]; /* Register value */
  171. u8 pwm[3]; /* Register value */
  172. u16 fan_conf; /* Configuration register values, combined */
  173. u16 in_vref; /* 1 mV/bit */
  174. u8 in[14]; /* Register value */
  175. u8 in_min[14]; /* Register value */
  176. u8 in_max[14]; /* Register value */
  177. u8 in_crit[3]; /* Register value */
  178. u8 in_status[14]; /* Register value */
  179. u16 in_alarms; /* Register values, combined, masked */
  180. u8 vid_conf; /* Configuration register value */
  181. u8 vrm;
  182. u8 vid; /* Register value */
  183. s8 temp[3]; /* Register value */
  184. s8 temp_min[3]; /* Register value */
  185. s8 temp_max[3]; /* Register value */
  186. s8 temp_crit[3]; /* Register value */
  187. u8 temp_status[3]; /* Register value */
  188. u8 temp_alarms; /* Register value, masked */
  189. };
  190. /*
  191. * Functions declaration
  192. */
  193. static int pc87360_probe(struct platform_device *pdev);
  194. static int __devexit pc87360_remove(struct platform_device *pdev);
  195. static int pc87360_read_value(struct pc87360_data *data, u8 ldi, u8 bank,
  196. u8 reg);
  197. static void pc87360_write_value(struct pc87360_data *data, u8 ldi, u8 bank,
  198. u8 reg, u8 value);
  199. static void pc87360_init_device(struct platform_device *pdev,
  200. int use_thermistors);
  201. static struct pc87360_data *pc87360_update_device(struct device *dev);
  202. /*
  203. * Driver data
  204. */
  205. static struct platform_driver pc87360_driver = {
  206. .driver = {
  207. .owner = THIS_MODULE,
  208. .name = "pc87360",
  209. },
  210. .probe = pc87360_probe,
  211. .remove = __devexit_p(pc87360_remove),
  212. };
  213. /*
  214. * Sysfs stuff
  215. */
  216. static ssize_t show_fan_input(struct device *dev, struct device_attribute *devattr, char *buf)
  217. {
  218. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  219. struct pc87360_data *data = pc87360_update_device(dev);
  220. return sprintf(buf, "%u\n", FAN_FROM_REG(data->fan[attr->index],
  221. FAN_DIV_FROM_REG(data->fan_status[attr->index])));
  222. }
  223. static ssize_t show_fan_min(struct device *dev, struct device_attribute *devattr, char *buf)
  224. {
  225. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  226. struct pc87360_data *data = pc87360_update_device(dev);
  227. return sprintf(buf, "%u\n", FAN_FROM_REG(data->fan_min[attr->index],
  228. FAN_DIV_FROM_REG(data->fan_status[attr->index])));
  229. }
  230. static ssize_t show_fan_div(struct device *dev, struct device_attribute *devattr, char *buf)
  231. {
  232. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  233. struct pc87360_data *data = pc87360_update_device(dev);
  234. return sprintf(buf, "%u\n",
  235. FAN_DIV_FROM_REG(data->fan_status[attr->index]));
  236. }
  237. static ssize_t show_fan_status(struct device *dev, struct device_attribute *devattr, char *buf)
  238. {
  239. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  240. struct pc87360_data *data = pc87360_update_device(dev);
  241. return sprintf(buf, "%u\n",
  242. FAN_STATUS_FROM_REG(data->fan_status[attr->index]));
  243. }
  244. static ssize_t set_fan_min(struct device *dev, struct device_attribute *devattr, const char *buf,
  245. size_t count)
  246. {
  247. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  248. struct pc87360_data *data = dev_get_drvdata(dev);
  249. long fan_min = simple_strtol(buf, NULL, 10);
  250. mutex_lock(&data->update_lock);
  251. fan_min = FAN_TO_REG(fan_min, FAN_DIV_FROM_REG(data->fan_status[attr->index]));
  252. /* If it wouldn't fit, change clock divisor */
  253. while (fan_min > 255
  254. && (data->fan_status[attr->index] & 0x60) != 0x60) {
  255. fan_min >>= 1;
  256. data->fan[attr->index] >>= 1;
  257. data->fan_status[attr->index] += 0x20;
  258. }
  259. data->fan_min[attr->index] = fan_min > 255 ? 255 : fan_min;
  260. pc87360_write_value(data, LD_FAN, NO_BANK, PC87360_REG_FAN_MIN(attr->index),
  261. data->fan_min[attr->index]);
  262. /* Write new divider, preserve alarm bits */
  263. pc87360_write_value(data, LD_FAN, NO_BANK, PC87360_REG_FAN_STATUS(attr->index),
  264. data->fan_status[attr->index] & 0xF9);
  265. mutex_unlock(&data->update_lock);
  266. return count;
  267. }
  268. static struct sensor_device_attribute fan_input[] = {
  269. SENSOR_ATTR(fan1_input, S_IRUGO, show_fan_input, NULL, 0),
  270. SENSOR_ATTR(fan2_input, S_IRUGO, show_fan_input, NULL, 1),
  271. SENSOR_ATTR(fan3_input, S_IRUGO, show_fan_input, NULL, 2),
  272. };
  273. static struct sensor_device_attribute fan_status[] = {
  274. SENSOR_ATTR(fan1_status, S_IRUGO, show_fan_status, NULL, 0),
  275. SENSOR_ATTR(fan2_status, S_IRUGO, show_fan_status, NULL, 1),
  276. SENSOR_ATTR(fan3_status, S_IRUGO, show_fan_status, NULL, 2),
  277. };
  278. static struct sensor_device_attribute fan_div[] = {
  279. SENSOR_ATTR(fan1_div, S_IRUGO, show_fan_div, NULL, 0),
  280. SENSOR_ATTR(fan2_div, S_IRUGO, show_fan_div, NULL, 1),
  281. SENSOR_ATTR(fan3_div, S_IRUGO, show_fan_div, NULL, 2),
  282. };
  283. static struct sensor_device_attribute fan_min[] = {
  284. SENSOR_ATTR(fan1_min, S_IWUSR | S_IRUGO, show_fan_min, set_fan_min, 0),
  285. SENSOR_ATTR(fan2_min, S_IWUSR | S_IRUGO, show_fan_min, set_fan_min, 1),
  286. SENSOR_ATTR(fan3_min, S_IWUSR | S_IRUGO, show_fan_min, set_fan_min, 2),
  287. };
  288. #define FAN_UNIT_ATTRS(X) \
  289. &fan_input[X].dev_attr.attr, \
  290. &fan_status[X].dev_attr.attr, \
  291. &fan_div[X].dev_attr.attr, \
  292. &fan_min[X].dev_attr.attr
  293. static ssize_t show_pwm(struct device *dev, struct device_attribute *devattr, char *buf)
  294. {
  295. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  296. struct pc87360_data *data = pc87360_update_device(dev);
  297. return sprintf(buf, "%u\n",
  298. PWM_FROM_REG(data->pwm[attr->index],
  299. FAN_CONFIG_INVERT(data->fan_conf,
  300. attr->index)));
  301. }
  302. static ssize_t set_pwm(struct device *dev, struct device_attribute *devattr, const char *buf,
  303. size_t count)
  304. {
  305. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  306. struct pc87360_data *data = dev_get_drvdata(dev);
  307. long val = simple_strtol(buf, NULL, 10);
  308. mutex_lock(&data->update_lock);
  309. data->pwm[attr->index] = PWM_TO_REG(val,
  310. FAN_CONFIG_INVERT(data->fan_conf, attr->index));
  311. pc87360_write_value(data, LD_FAN, NO_BANK, PC87360_REG_PWM(attr->index),
  312. data->pwm[attr->index]);
  313. mutex_unlock(&data->update_lock);
  314. return count;
  315. }
  316. static struct sensor_device_attribute pwm[] = {
  317. SENSOR_ATTR(pwm1, S_IWUSR | S_IRUGO, show_pwm, set_pwm, 0),
  318. SENSOR_ATTR(pwm2, S_IWUSR | S_IRUGO, show_pwm, set_pwm, 1),
  319. SENSOR_ATTR(pwm3, S_IWUSR | S_IRUGO, show_pwm, set_pwm, 2),
  320. };
  321. static struct attribute * pc8736x_fan_attr_array[] = {
  322. FAN_UNIT_ATTRS(0),
  323. FAN_UNIT_ATTRS(1),
  324. FAN_UNIT_ATTRS(2),
  325. &pwm[0].dev_attr.attr,
  326. &pwm[1].dev_attr.attr,
  327. &pwm[2].dev_attr.attr,
  328. NULL
  329. };
  330. static const struct attribute_group pc8736x_fan_group = {
  331. .attrs = pc8736x_fan_attr_array,
  332. };
  333. static ssize_t show_in_input(struct device *dev, struct device_attribute *devattr, char *buf)
  334. {
  335. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  336. struct pc87360_data *data = pc87360_update_device(dev);
  337. return sprintf(buf, "%u\n", IN_FROM_REG(data->in[attr->index],
  338. data->in_vref));
  339. }
  340. static ssize_t show_in_min(struct device *dev, struct device_attribute *devattr, char *buf)
  341. {
  342. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  343. struct pc87360_data *data = pc87360_update_device(dev);
  344. return sprintf(buf, "%u\n", IN_FROM_REG(data->in_min[attr->index],
  345. data->in_vref));
  346. }
  347. static ssize_t show_in_max(struct device *dev, struct device_attribute *devattr, char *buf)
  348. {
  349. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  350. struct pc87360_data *data = pc87360_update_device(dev);
  351. return sprintf(buf, "%u\n", IN_FROM_REG(data->in_max[attr->index],
  352. data->in_vref));
  353. }
  354. static ssize_t show_in_status(struct device *dev, struct device_attribute *devattr, char *buf)
  355. {
  356. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  357. struct pc87360_data *data = pc87360_update_device(dev);
  358. return sprintf(buf, "%u\n", data->in_status[attr->index]);
  359. }
  360. static ssize_t set_in_min(struct device *dev, struct device_attribute *devattr, const char *buf,
  361. size_t count)
  362. {
  363. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  364. struct pc87360_data *data = dev_get_drvdata(dev);
  365. long val = simple_strtol(buf, NULL, 10);
  366. mutex_lock(&data->update_lock);
  367. data->in_min[attr->index] = IN_TO_REG(val, data->in_vref);
  368. pc87360_write_value(data, LD_IN, attr->index, PC87365_REG_IN_MIN,
  369. data->in_min[attr->index]);
  370. mutex_unlock(&data->update_lock);
  371. return count;
  372. }
  373. static ssize_t set_in_max(struct device *dev, struct device_attribute *devattr, const char *buf,
  374. size_t count)
  375. {
  376. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  377. struct pc87360_data *data = dev_get_drvdata(dev);
  378. long val = simple_strtol(buf, NULL, 10);
  379. mutex_lock(&data->update_lock);
  380. data->in_max[attr->index] = IN_TO_REG(val,
  381. data->in_vref);
  382. pc87360_write_value(data, LD_IN, attr->index, PC87365_REG_IN_MAX,
  383. data->in_max[attr->index]);
  384. mutex_unlock(&data->update_lock);
  385. return count;
  386. }
  387. static struct sensor_device_attribute in_input[] = {
  388. SENSOR_ATTR(in0_input, S_IRUGO, show_in_input, NULL, 0),
  389. SENSOR_ATTR(in1_input, S_IRUGO, show_in_input, NULL, 1),
  390. SENSOR_ATTR(in2_input, S_IRUGO, show_in_input, NULL, 2),
  391. SENSOR_ATTR(in3_input, S_IRUGO, show_in_input, NULL, 3),
  392. SENSOR_ATTR(in4_input, S_IRUGO, show_in_input, NULL, 4),
  393. SENSOR_ATTR(in5_input, S_IRUGO, show_in_input, NULL, 5),
  394. SENSOR_ATTR(in6_input, S_IRUGO, show_in_input, NULL, 6),
  395. SENSOR_ATTR(in7_input, S_IRUGO, show_in_input, NULL, 7),
  396. SENSOR_ATTR(in8_input, S_IRUGO, show_in_input, NULL, 8),
  397. SENSOR_ATTR(in9_input, S_IRUGO, show_in_input, NULL, 9),
  398. SENSOR_ATTR(in10_input, S_IRUGO, show_in_input, NULL, 10),
  399. };
  400. static struct sensor_device_attribute in_status[] = {
  401. SENSOR_ATTR(in0_status, S_IRUGO, show_in_status, NULL, 0),
  402. SENSOR_ATTR(in1_status, S_IRUGO, show_in_status, NULL, 1),
  403. SENSOR_ATTR(in2_status, S_IRUGO, show_in_status, NULL, 2),
  404. SENSOR_ATTR(in3_status, S_IRUGO, show_in_status, NULL, 3),
  405. SENSOR_ATTR(in4_status, S_IRUGO, show_in_status, NULL, 4),
  406. SENSOR_ATTR(in5_status, S_IRUGO, show_in_status, NULL, 5),
  407. SENSOR_ATTR(in6_status, S_IRUGO, show_in_status, NULL, 6),
  408. SENSOR_ATTR(in7_status, S_IRUGO, show_in_status, NULL, 7),
  409. SENSOR_ATTR(in8_status, S_IRUGO, show_in_status, NULL, 8),
  410. SENSOR_ATTR(in9_status, S_IRUGO, show_in_status, NULL, 9),
  411. SENSOR_ATTR(in10_status, S_IRUGO, show_in_status, NULL, 10),
  412. };
  413. static struct sensor_device_attribute in_min[] = {
  414. SENSOR_ATTR(in0_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 0),
  415. SENSOR_ATTR(in1_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 1),
  416. SENSOR_ATTR(in2_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 2),
  417. SENSOR_ATTR(in3_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 3),
  418. SENSOR_ATTR(in4_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 4),
  419. SENSOR_ATTR(in5_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 5),
  420. SENSOR_ATTR(in6_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 6),
  421. SENSOR_ATTR(in7_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 7),
  422. SENSOR_ATTR(in8_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 8),
  423. SENSOR_ATTR(in9_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 9),
  424. SENSOR_ATTR(in10_min, S_IWUSR | S_IRUGO, show_in_min, set_in_min, 10),
  425. };
  426. static struct sensor_device_attribute in_max[] = {
  427. SENSOR_ATTR(in0_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 0),
  428. SENSOR_ATTR(in1_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 1),
  429. SENSOR_ATTR(in2_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 2),
  430. SENSOR_ATTR(in3_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 3),
  431. SENSOR_ATTR(in4_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 4),
  432. SENSOR_ATTR(in5_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 5),
  433. SENSOR_ATTR(in6_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 6),
  434. SENSOR_ATTR(in7_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 7),
  435. SENSOR_ATTR(in8_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 8),
  436. SENSOR_ATTR(in9_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 9),
  437. SENSOR_ATTR(in10_max, S_IWUSR | S_IRUGO, show_in_max, set_in_max, 10),
  438. };
  439. /* (temp & vin) channel status register alarm bits (pdf sec.11.5.12) */
  440. #define CHAN_ALM_MIN 0x02 /* min limit crossed */
  441. #define CHAN_ALM_MAX 0x04 /* max limit exceeded */
  442. #define TEMP_ALM_CRIT 0x08 /* temp crit exceeded (temp only) */
  443. /* show_in_min/max_alarm() reads data from the per-channel status
  444. register (sec 11.5.12), not the vin event status registers (sec
  445. 11.5.2) that (legacy) show_in_alarm() resds (via data->in_alarms) */
  446. static ssize_t show_in_min_alarm(struct device *dev,
  447. struct device_attribute *devattr, char *buf)
  448. {
  449. struct pc87360_data *data = pc87360_update_device(dev);
  450. unsigned nr = to_sensor_dev_attr(devattr)->index;
  451. return sprintf(buf, "%u\n", !!(data->in_status[nr] & CHAN_ALM_MIN));
  452. }
  453. static ssize_t show_in_max_alarm(struct device *dev,
  454. struct device_attribute *devattr, char *buf)
  455. {
  456. struct pc87360_data *data = pc87360_update_device(dev);
  457. unsigned nr = to_sensor_dev_attr(devattr)->index;
  458. return sprintf(buf, "%u\n", !!(data->in_status[nr] & CHAN_ALM_MAX));
  459. }
  460. static struct sensor_device_attribute in_min_alarm[] = {
  461. SENSOR_ATTR(in0_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 0),
  462. SENSOR_ATTR(in1_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 1),
  463. SENSOR_ATTR(in2_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 2),
  464. SENSOR_ATTR(in3_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 3),
  465. SENSOR_ATTR(in4_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 4),
  466. SENSOR_ATTR(in5_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 5),
  467. SENSOR_ATTR(in6_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 6),
  468. SENSOR_ATTR(in7_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 7),
  469. SENSOR_ATTR(in8_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 8),
  470. SENSOR_ATTR(in9_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 9),
  471. SENSOR_ATTR(in10_min_alarm, S_IRUGO, show_in_min_alarm, NULL, 10),
  472. };
  473. static struct sensor_device_attribute in_max_alarm[] = {
  474. SENSOR_ATTR(in0_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 0),
  475. SENSOR_ATTR(in1_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 1),
  476. SENSOR_ATTR(in2_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 2),
  477. SENSOR_ATTR(in3_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 3),
  478. SENSOR_ATTR(in4_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 4),
  479. SENSOR_ATTR(in5_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 5),
  480. SENSOR_ATTR(in6_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 6),
  481. SENSOR_ATTR(in7_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 7),
  482. SENSOR_ATTR(in8_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 8),
  483. SENSOR_ATTR(in9_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 9),
  484. SENSOR_ATTR(in10_max_alarm, S_IRUGO, show_in_max_alarm, NULL, 10),
  485. };
  486. #define VIN_UNIT_ATTRS(X) \
  487. &in_input[X].dev_attr.attr, \
  488. &in_status[X].dev_attr.attr, \
  489. &in_min[X].dev_attr.attr, \
  490. &in_max[X].dev_attr.attr, \
  491. &in_min_alarm[X].dev_attr.attr, \
  492. &in_max_alarm[X].dev_attr.attr
  493. static ssize_t show_vid(struct device *dev, struct device_attribute *attr, char *buf)
  494. {
  495. struct pc87360_data *data = pc87360_update_device(dev);
  496. return sprintf(buf, "%u\n", vid_from_reg(data->vid, data->vrm));
  497. }
  498. static DEVICE_ATTR(cpu0_vid, S_IRUGO, show_vid, NULL);
  499. static ssize_t show_vrm(struct device *dev, struct device_attribute *attr, char *buf)
  500. {
  501. struct pc87360_data *data = dev_get_drvdata(dev);
  502. return sprintf(buf, "%u\n", data->vrm);
  503. }
  504. static ssize_t set_vrm(struct device *dev, struct device_attribute *attr, const char *buf, size_t count)
  505. {
  506. struct pc87360_data *data = dev_get_drvdata(dev);
  507. data->vrm = simple_strtoul(buf, NULL, 10);
  508. return count;
  509. }
  510. static DEVICE_ATTR(vrm, S_IRUGO | S_IWUSR, show_vrm, set_vrm);
  511. static ssize_t show_in_alarms(struct device *dev, struct device_attribute *attr, char *buf)
  512. {
  513. struct pc87360_data *data = pc87360_update_device(dev);
  514. return sprintf(buf, "%u\n", data->in_alarms);
  515. }
  516. static DEVICE_ATTR(alarms_in, S_IRUGO, show_in_alarms, NULL);
  517. static struct attribute *pc8736x_vin_attr_array[] = {
  518. VIN_UNIT_ATTRS(0),
  519. VIN_UNIT_ATTRS(1),
  520. VIN_UNIT_ATTRS(2),
  521. VIN_UNIT_ATTRS(3),
  522. VIN_UNIT_ATTRS(4),
  523. VIN_UNIT_ATTRS(5),
  524. VIN_UNIT_ATTRS(6),
  525. VIN_UNIT_ATTRS(7),
  526. VIN_UNIT_ATTRS(8),
  527. VIN_UNIT_ATTRS(9),
  528. VIN_UNIT_ATTRS(10),
  529. &dev_attr_cpu0_vid.attr,
  530. &dev_attr_vrm.attr,
  531. &dev_attr_alarms_in.attr,
  532. NULL
  533. };
  534. static const struct attribute_group pc8736x_vin_group = {
  535. .attrs = pc8736x_vin_attr_array,
  536. };
  537. static ssize_t show_therm_input(struct device *dev, struct device_attribute *devattr, char *buf)
  538. {
  539. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  540. struct pc87360_data *data = pc87360_update_device(dev);
  541. return sprintf(buf, "%u\n", IN_FROM_REG(data->in[attr->index],
  542. data->in_vref));
  543. }
  544. static ssize_t show_therm_min(struct device *dev, struct device_attribute *devattr, char *buf)
  545. {
  546. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  547. struct pc87360_data *data = pc87360_update_device(dev);
  548. return sprintf(buf, "%u\n", IN_FROM_REG(data->in_min[attr->index],
  549. data->in_vref));
  550. }
  551. static ssize_t show_therm_max(struct device *dev, struct device_attribute *devattr, char *buf)
  552. {
  553. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  554. struct pc87360_data *data = pc87360_update_device(dev);
  555. return sprintf(buf, "%u\n", IN_FROM_REG(data->in_max[attr->index],
  556. data->in_vref));
  557. }
  558. static ssize_t show_therm_crit(struct device *dev, struct device_attribute *devattr, char *buf)
  559. {
  560. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  561. struct pc87360_data *data = pc87360_update_device(dev);
  562. return sprintf(buf, "%u\n", IN_FROM_REG(data->in_crit[attr->index-11],
  563. data->in_vref));
  564. }
  565. static ssize_t show_therm_status(struct device *dev, struct device_attribute *devattr, char *buf)
  566. {
  567. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  568. struct pc87360_data *data = pc87360_update_device(dev);
  569. return sprintf(buf, "%u\n", data->in_status[attr->index]);
  570. }
  571. static ssize_t set_therm_min(struct device *dev, struct device_attribute *devattr, const char *buf,
  572. size_t count)
  573. {
  574. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  575. struct pc87360_data *data = dev_get_drvdata(dev);
  576. long val = simple_strtol(buf, NULL, 10);
  577. mutex_lock(&data->update_lock);
  578. data->in_min[attr->index] = IN_TO_REG(val, data->in_vref);
  579. pc87360_write_value(data, LD_IN, attr->index, PC87365_REG_TEMP_MIN,
  580. data->in_min[attr->index]);
  581. mutex_unlock(&data->update_lock);
  582. return count;
  583. }
  584. static ssize_t set_therm_max(struct device *dev, struct device_attribute *devattr, const char *buf,
  585. size_t count)
  586. {
  587. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  588. struct pc87360_data *data = dev_get_drvdata(dev);
  589. long val = simple_strtol(buf, NULL, 10);
  590. mutex_lock(&data->update_lock);
  591. data->in_max[attr->index] = IN_TO_REG(val, data->in_vref);
  592. pc87360_write_value(data, LD_IN, attr->index, PC87365_REG_TEMP_MAX,
  593. data->in_max[attr->index]);
  594. mutex_unlock(&data->update_lock);
  595. return count;
  596. }
  597. static ssize_t set_therm_crit(struct device *dev, struct device_attribute *devattr, const char *buf,
  598. size_t count)
  599. {
  600. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  601. struct pc87360_data *data = dev_get_drvdata(dev);
  602. long val = simple_strtol(buf, NULL, 10);
  603. mutex_lock(&data->update_lock);
  604. data->in_crit[attr->index-11] = IN_TO_REG(val, data->in_vref);
  605. pc87360_write_value(data, LD_IN, attr->index, PC87365_REG_TEMP_CRIT,
  606. data->in_crit[attr->index-11]);
  607. mutex_unlock(&data->update_lock);
  608. return count;
  609. }
  610. /* the +11 term below reflects the fact that VLM units 11,12,13 are
  611. used in the chip to measure voltage across the thermistors
  612. */
  613. static struct sensor_device_attribute therm_input[] = {
  614. SENSOR_ATTR(temp4_input, S_IRUGO, show_therm_input, NULL, 0+11),
  615. SENSOR_ATTR(temp5_input, S_IRUGO, show_therm_input, NULL, 1+11),
  616. SENSOR_ATTR(temp6_input, S_IRUGO, show_therm_input, NULL, 2+11),
  617. };
  618. static struct sensor_device_attribute therm_status[] = {
  619. SENSOR_ATTR(temp4_status, S_IRUGO, show_therm_status, NULL, 0+11),
  620. SENSOR_ATTR(temp5_status, S_IRUGO, show_therm_status, NULL, 1+11),
  621. SENSOR_ATTR(temp6_status, S_IRUGO, show_therm_status, NULL, 2+11),
  622. };
  623. static struct sensor_device_attribute therm_min[] = {
  624. SENSOR_ATTR(temp4_min, S_IRUGO | S_IWUSR,
  625. show_therm_min, set_therm_min, 0+11),
  626. SENSOR_ATTR(temp5_min, S_IRUGO | S_IWUSR,
  627. show_therm_min, set_therm_min, 1+11),
  628. SENSOR_ATTR(temp6_min, S_IRUGO | S_IWUSR,
  629. show_therm_min, set_therm_min, 2+11),
  630. };
  631. static struct sensor_device_attribute therm_max[] = {
  632. SENSOR_ATTR(temp4_max, S_IRUGO | S_IWUSR,
  633. show_therm_max, set_therm_max, 0+11),
  634. SENSOR_ATTR(temp5_max, S_IRUGO | S_IWUSR,
  635. show_therm_max, set_therm_max, 1+11),
  636. SENSOR_ATTR(temp6_max, S_IRUGO | S_IWUSR,
  637. show_therm_max, set_therm_max, 2+11),
  638. };
  639. static struct sensor_device_attribute therm_crit[] = {
  640. SENSOR_ATTR(temp4_crit, S_IRUGO | S_IWUSR,
  641. show_therm_crit, set_therm_crit, 0+11),
  642. SENSOR_ATTR(temp5_crit, S_IRUGO | S_IWUSR,
  643. show_therm_crit, set_therm_crit, 1+11),
  644. SENSOR_ATTR(temp6_crit, S_IRUGO | S_IWUSR,
  645. show_therm_crit, set_therm_crit, 2+11),
  646. };
  647. #define THERM_UNIT_ATTRS(X) \
  648. &therm_input[X].dev_attr.attr, \
  649. &therm_status[X].dev_attr.attr, \
  650. &therm_min[X].dev_attr.attr, \
  651. &therm_max[X].dev_attr.attr, \
  652. &therm_crit[X].dev_attr.attr
  653. static struct attribute * pc8736x_therm_attr_array[] = {
  654. THERM_UNIT_ATTRS(0),
  655. THERM_UNIT_ATTRS(1),
  656. THERM_UNIT_ATTRS(2),
  657. NULL
  658. };
  659. static const struct attribute_group pc8736x_therm_group = {
  660. .attrs = pc8736x_therm_attr_array,
  661. };
  662. static ssize_t show_temp_input(struct device *dev, struct device_attribute *devattr, char *buf)
  663. {
  664. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  665. struct pc87360_data *data = pc87360_update_device(dev);
  666. return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp[attr->index]));
  667. }
  668. static ssize_t show_temp_min(struct device *dev, struct device_attribute *devattr, char *buf)
  669. {
  670. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  671. struct pc87360_data *data = pc87360_update_device(dev);
  672. return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp_min[attr->index]));
  673. }
  674. static ssize_t show_temp_max(struct device *dev, struct device_attribute *devattr, char *buf)
  675. {
  676. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  677. struct pc87360_data *data = pc87360_update_device(dev);
  678. return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp_max[attr->index]));
  679. }
  680. static ssize_t show_temp_crit(struct device *dev, struct device_attribute *devattr, char *buf)
  681. {
  682. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  683. struct pc87360_data *data = pc87360_update_device(dev);
  684. return sprintf(buf, "%d\n", TEMP_FROM_REG(data->temp_crit[attr->index]));
  685. }
  686. static ssize_t show_temp_status(struct device *dev, struct device_attribute *devattr, char *buf)
  687. {
  688. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  689. struct pc87360_data *data = pc87360_update_device(dev);
  690. return sprintf(buf, "%d\n", data->temp_status[attr->index]);
  691. }
  692. static ssize_t set_temp_min(struct device *dev, struct device_attribute *devattr, const char *buf,
  693. size_t count)
  694. {
  695. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  696. struct pc87360_data *data = dev_get_drvdata(dev);
  697. long val = simple_strtol(buf, NULL, 10);
  698. mutex_lock(&data->update_lock);
  699. data->temp_min[attr->index] = TEMP_TO_REG(val);
  700. pc87360_write_value(data, LD_TEMP, attr->index, PC87365_REG_TEMP_MIN,
  701. data->temp_min[attr->index]);
  702. mutex_unlock(&data->update_lock);
  703. return count;
  704. }
  705. static ssize_t set_temp_max(struct device *dev, struct device_attribute *devattr, const char *buf,
  706. size_t count)
  707. {
  708. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  709. struct pc87360_data *data = dev_get_drvdata(dev);
  710. long val = simple_strtol(buf, NULL, 10);
  711. mutex_lock(&data->update_lock);
  712. data->temp_max[attr->index] = TEMP_TO_REG(val);
  713. pc87360_write_value(data, LD_TEMP, attr->index, PC87365_REG_TEMP_MAX,
  714. data->temp_max[attr->index]);
  715. mutex_unlock(&data->update_lock);
  716. return count;
  717. }
  718. static ssize_t set_temp_crit(struct device *dev, struct device_attribute *devattr, const char *buf,
  719. size_t count)
  720. {
  721. struct sensor_device_attribute *attr = to_sensor_dev_attr(devattr);
  722. struct pc87360_data *data = dev_get_drvdata(dev);
  723. long val = simple_strtol(buf, NULL, 10);
  724. mutex_lock(&data->update_lock);
  725. data->temp_crit[attr->index] = TEMP_TO_REG(val);
  726. pc87360_write_value(data, LD_TEMP, attr->index, PC87365_REG_TEMP_CRIT,
  727. data->temp_crit[attr->index]);
  728. mutex_unlock(&data->update_lock);
  729. return count;
  730. }
  731. static struct sensor_device_attribute temp_input[] = {
  732. SENSOR_ATTR(temp1_input, S_IRUGO, show_temp_input, NULL, 0),
  733. SENSOR_ATTR(temp2_input, S_IRUGO, show_temp_input, NULL, 1),
  734. SENSOR_ATTR(temp3_input, S_IRUGO, show_temp_input, NULL, 2),
  735. };
  736. static struct sensor_device_attribute temp_status[] = {
  737. SENSOR_ATTR(temp1_status, S_IRUGO, show_temp_status, NULL, 0),
  738. SENSOR_ATTR(temp2_status, S_IRUGO, show_temp_status, NULL, 1),
  739. SENSOR_ATTR(temp3_status, S_IRUGO, show_temp_status, NULL, 2),
  740. };
  741. static struct sensor_device_attribute temp_min[] = {
  742. SENSOR_ATTR(temp1_min, S_IRUGO | S_IWUSR,
  743. show_temp_min, set_temp_min, 0),
  744. SENSOR_ATTR(temp2_min, S_IRUGO | S_IWUSR,
  745. show_temp_min, set_temp_min, 1),
  746. SENSOR_ATTR(temp3_min, S_IRUGO | S_IWUSR,
  747. show_temp_min, set_temp_min, 2),
  748. };
  749. static struct sensor_device_attribute temp_max[] = {
  750. SENSOR_ATTR(temp1_max, S_IRUGO | S_IWUSR,
  751. show_temp_max, set_temp_max, 0),
  752. SENSOR_ATTR(temp2_max, S_IRUGO | S_IWUSR,
  753. show_temp_max, set_temp_max, 1),
  754. SENSOR_ATTR(temp3_max, S_IRUGO | S_IWUSR,
  755. show_temp_max, set_temp_max, 2),
  756. };
  757. static struct sensor_device_attribute temp_crit[] = {
  758. SENSOR_ATTR(temp1_crit, S_IRUGO | S_IWUSR,
  759. show_temp_crit, set_temp_crit, 0),
  760. SENSOR_ATTR(temp2_crit, S_IRUGO | S_IWUSR,
  761. show_temp_crit, set_temp_crit, 1),
  762. SENSOR_ATTR(temp3_crit, S_IRUGO | S_IWUSR,
  763. show_temp_crit, set_temp_crit, 2),
  764. };
  765. static ssize_t show_temp_alarms(struct device *dev, struct device_attribute *attr, char *buf)
  766. {
  767. struct pc87360_data *data = pc87360_update_device(dev);
  768. return sprintf(buf, "%u\n", data->temp_alarms);
  769. }
  770. static DEVICE_ATTR(alarms_temp, S_IRUGO, show_temp_alarms, NULL);
  771. /* show_temp_min/max_alarm() reads data from the per-channel status
  772. register (sec 12.3.7), not the temp event status registers (sec
  773. 12.3.2) that show_temp_alarm() reads (via data->temp_alarms) */
  774. static ssize_t show_temp_min_alarm(struct device *dev,
  775. struct device_attribute *devattr, char *buf)
  776. {
  777. struct pc87360_data *data = pc87360_update_device(dev);
  778. unsigned nr = to_sensor_dev_attr(devattr)->index;
  779. return sprintf(buf, "%u\n", !!(data->temp_status[nr] & CHAN_ALM_MIN));
  780. }
  781. static ssize_t show_temp_max_alarm(struct device *dev,
  782. struct device_attribute *devattr, char *buf)
  783. {
  784. struct pc87360_data *data = pc87360_update_device(dev);
  785. unsigned nr = to_sensor_dev_attr(devattr)->index;
  786. return sprintf(buf, "%u\n", !!(data->temp_status[nr] & CHAN_ALM_MAX));
  787. }
  788. static ssize_t show_temp_crit_alarm(struct device *dev,
  789. struct device_attribute *devattr, char *buf)
  790. {
  791. struct pc87360_data *data = pc87360_update_device(dev);
  792. unsigned nr = to_sensor_dev_attr(devattr)->index;
  793. return sprintf(buf, "%u\n", !!(data->temp_status[nr] & TEMP_ALM_CRIT));
  794. }
  795. static struct sensor_device_attribute temp_min_alarm[] = {
  796. SENSOR_ATTR(temp1_min_alarm, S_IRUGO, show_temp_min_alarm, NULL, 0),
  797. SENSOR_ATTR(temp2_min_alarm, S_IRUGO, show_temp_min_alarm, NULL, 1),
  798. SENSOR_ATTR(temp3_min_alarm, S_IRUGO, show_temp_min_alarm, NULL, 2),
  799. };
  800. static struct sensor_device_attribute temp_max_alarm[] = {
  801. SENSOR_ATTR(temp1_max_alarm, S_IRUGO, show_temp_max_alarm, NULL, 0),
  802. SENSOR_ATTR(temp2_max_alarm, S_IRUGO, show_temp_max_alarm, NULL, 1),
  803. SENSOR_ATTR(temp3_max_alarm, S_IRUGO, show_temp_max_alarm, NULL, 2),
  804. };
  805. static struct sensor_device_attribute temp_crit_alarm[] = {
  806. SENSOR_ATTR(temp1_crit_alarm, S_IRUGO, show_temp_crit_alarm, NULL, 0),
  807. SENSOR_ATTR(temp2_crit_alarm, S_IRUGO, show_temp_crit_alarm, NULL, 1),
  808. SENSOR_ATTR(temp3_crit_alarm, S_IRUGO, show_temp_crit_alarm, NULL, 2),
  809. };
  810. #define TEMP_FAULT 0x40 /* open diode */
  811. static ssize_t show_temp_fault(struct device *dev,
  812. struct device_attribute *devattr, char *buf)
  813. {
  814. struct pc87360_data *data = pc87360_update_device(dev);
  815. unsigned nr = to_sensor_dev_attr(devattr)->index;
  816. return sprintf(buf, "%u\n", !!(data->temp_status[nr] & TEMP_FAULT));
  817. }
  818. static struct sensor_device_attribute temp_fault[] = {
  819. SENSOR_ATTR(temp1_fault, S_IRUGO, show_temp_fault, NULL, 0),
  820. SENSOR_ATTR(temp2_fault, S_IRUGO, show_temp_fault, NULL, 1),
  821. SENSOR_ATTR(temp3_fault, S_IRUGO, show_temp_fault, NULL, 2),
  822. };
  823. #define TEMP_UNIT_ATTRS(X) \
  824. &temp_input[X].dev_attr.attr, \
  825. &temp_status[X].dev_attr.attr, \
  826. &temp_min[X].dev_attr.attr, \
  827. &temp_max[X].dev_attr.attr, \
  828. &temp_crit[X].dev_attr.attr, \
  829. &temp_min_alarm[X].dev_attr.attr, \
  830. &temp_max_alarm[X].dev_attr.attr, \
  831. &temp_crit_alarm[X].dev_attr.attr, \
  832. &temp_fault[X].dev_attr.attr
  833. static struct attribute * pc8736x_temp_attr_array[] = {
  834. TEMP_UNIT_ATTRS(0),
  835. TEMP_UNIT_ATTRS(1),
  836. TEMP_UNIT_ATTRS(2),
  837. /* include the few miscellaneous atts here */
  838. &dev_attr_alarms_temp.attr,
  839. NULL
  840. };
  841. static const struct attribute_group pc8736x_temp_group = {
  842. .attrs = pc8736x_temp_attr_array,
  843. };
  844. static ssize_t show_name(struct device *dev,
  845. struct device_attribute *devattr, char *buf)
  846. {
  847. struct pc87360_data *data = dev_get_drvdata(dev);
  848. return sprintf(buf, "%s\n", data->name);
  849. }
  850. static DEVICE_ATTR(name, S_IRUGO, show_name, NULL);
  851. /*
  852. * Device detection, registration and update
  853. */
  854. static int __init pc87360_find(int sioaddr, u8 *devid, unsigned short *addresses)
  855. {
  856. u16 val;
  857. int i;
  858. int nrdev; /* logical device count */
  859. /* No superio_enter */
  860. /* Identify device */
  861. val = force_id ? force_id : superio_inb(sioaddr, DEVID);
  862. switch (val) {
  863. case 0xE1: /* PC87360 */
  864. case 0xE8: /* PC87363 */
  865. case 0xE4: /* PC87364 */
  866. nrdev = 1;
  867. break;
  868. case 0xE5: /* PC87365 */
  869. case 0xE9: /* PC87366 */
  870. nrdev = 3;
  871. break;
  872. default:
  873. superio_exit(sioaddr);
  874. return -ENODEV;
  875. }
  876. /* Remember the device id */
  877. *devid = val;
  878. for (i = 0; i < nrdev; i++) {
  879. /* select logical device */
  880. superio_outb(sioaddr, DEV, logdev[i]);
  881. val = superio_inb(sioaddr, ACT);
  882. if (!(val & 0x01)) {
  883. printk(KERN_INFO "pc87360: Device 0x%02x not "
  884. "activated\n", logdev[i]);
  885. continue;
  886. }
  887. val = (superio_inb(sioaddr, BASE) << 8)
  888. | superio_inb(sioaddr, BASE + 1);
  889. if (!val) {
  890. printk(KERN_INFO "pc87360: Base address not set for "
  891. "device 0x%02x\n", logdev[i]);
  892. continue;
  893. }
  894. addresses[i] = val;
  895. if (i==0) { /* Fans */
  896. confreg[0] = superio_inb(sioaddr, 0xF0);
  897. confreg[1] = superio_inb(sioaddr, 0xF1);
  898. #ifdef DEBUG
  899. printk(KERN_DEBUG "pc87360: Fan 1: mon=%d "
  900. "ctrl=%d inv=%d\n", (confreg[0]>>2)&1,
  901. (confreg[0]>>3)&1, (confreg[0]>>4)&1);
  902. printk(KERN_DEBUG "pc87360: Fan 2: mon=%d "
  903. "ctrl=%d inv=%d\n", (confreg[0]>>5)&1,
  904. (confreg[0]>>6)&1, (confreg[0]>>7)&1);
  905. printk(KERN_DEBUG "pc87360: Fan 3: mon=%d "
  906. "ctrl=%d inv=%d\n", confreg[1]&1,
  907. (confreg[1]>>1)&1, (confreg[1]>>2)&1);
  908. #endif
  909. } else if (i==1) { /* Voltages */
  910. /* Are we using thermistors? */
  911. if (*devid == 0xE9) { /* PC87366 */
  912. /* These registers are not logical-device
  913. specific, just that we won't need them if
  914. we don't use the VLM device */
  915. confreg[2] = superio_inb(sioaddr, 0x2B);
  916. confreg[3] = superio_inb(sioaddr, 0x25);
  917. if (confreg[2] & 0x40) {
  918. printk(KERN_INFO "pc87360: Using "
  919. "thermistors for temperature "
  920. "monitoring\n");
  921. }
  922. if (confreg[3] & 0xE0) {
  923. printk(KERN_INFO "pc87360: VID "
  924. "inputs routed (mode %u)\n",
  925. confreg[3] >> 5);
  926. }
  927. }
  928. }
  929. }
  930. superio_exit(sioaddr);
  931. return 0;
  932. }
  933. static int __devinit pc87360_probe(struct platform_device *pdev)
  934. {
  935. int i;
  936. struct pc87360_data *data;
  937. int err = 0;
  938. const char *name = "pc87360";
  939. int use_thermistors = 0;
  940. struct device *dev = &pdev->dev;
  941. if (!(data = kzalloc(sizeof(struct pc87360_data), GFP_KERNEL)))
  942. return -ENOMEM;
  943. data->fannr = 2;
  944. data->innr = 0;
  945. data->tempnr = 0;
  946. switch (devid) {
  947. case 0xe8:
  948. name = "pc87363";
  949. break;
  950. case 0xe4:
  951. name = "pc87364";
  952. data->fannr = 3;
  953. break;
  954. case 0xe5:
  955. name = "pc87365";
  956. data->fannr = extra_isa[0] ? 3 : 0;
  957. data->innr = extra_isa[1] ? 11 : 0;
  958. data->tempnr = extra_isa[2] ? 2 : 0;
  959. break;
  960. case 0xe9:
  961. name = "pc87366";
  962. data->fannr = extra_isa[0] ? 3 : 0;
  963. data->innr = extra_isa[1] ? 14 : 0;
  964. data->tempnr = extra_isa[2] ? 3 : 0;
  965. break;
  966. }
  967. data->name = name;
  968. data->valid = 0;
  969. mutex_init(&data->lock);
  970. mutex_init(&data->update_lock);
  971. platform_set_drvdata(pdev, data);
  972. for (i = 0; i < 3; i++) {
  973. if (((data->address[i] = extra_isa[i]))
  974. && !request_region(extra_isa[i], PC87360_EXTENT,
  975. pc87360_driver.driver.name)) {
  976. dev_err(dev, "Region 0x%x-0x%x already "
  977. "in use!\n", extra_isa[i],
  978. extra_isa[i]+PC87360_EXTENT-1);
  979. for (i--; i >= 0; i--)
  980. release_region(extra_isa[i], PC87360_EXTENT);
  981. err = -EBUSY;
  982. goto ERROR1;
  983. }
  984. }
  985. /* Retrieve the fans configuration from Super-I/O space */
  986. if (data->fannr)
  987. data->fan_conf = confreg[0] | (confreg[1] << 8);
  988. /* Use the correct reference voltage
  989. Unless both the VLM and the TMS logical devices agree to
  990. use an external Vref, the internal one is used. */
  991. if (data->innr) {
  992. i = pc87360_read_value(data, LD_IN, NO_BANK,
  993. PC87365_REG_IN_CONFIG);
  994. if (data->tempnr) {
  995. i &= pc87360_read_value(data, LD_TEMP, NO_BANK,
  996. PC87365_REG_TEMP_CONFIG);
  997. }
  998. data->in_vref = (i&0x02) ? 3025 : 2966;
  999. dev_dbg(dev, "Using %s reference voltage\n",
  1000. (i&0x02) ? "external" : "internal");
  1001. data->vid_conf = confreg[3];
  1002. data->vrm = vid_which_vrm();
  1003. }
  1004. /* Fan clock dividers may be needed before any data is read */
  1005. for (i = 0; i < data->fannr; i++) {
  1006. if (FAN_CONFIG_MONITOR(data->fan_conf, i))
  1007. data->fan_status[i] = pc87360_read_value(data,
  1008. LD_FAN, NO_BANK,
  1009. PC87360_REG_FAN_STATUS(i));
  1010. }
  1011. if (init > 0) {
  1012. if (devid == 0xe9 && data->address[1]) /* PC87366 */
  1013. use_thermistors = confreg[2] & 0x40;
  1014. pc87360_init_device(pdev, use_thermistors);
  1015. }
  1016. /* Register all-or-nothing sysfs groups */
  1017. if (data->innr &&
  1018. (err = sysfs_create_group(&dev->kobj,
  1019. &pc8736x_vin_group)))
  1020. goto ERROR3;
  1021. if (data->innr == 14 &&
  1022. (err = sysfs_create_group(&dev->kobj,
  1023. &pc8736x_therm_group)))
  1024. goto ERROR3;
  1025. /* create device attr-files for varying sysfs groups */
  1026. if (data->tempnr) {
  1027. for (i = 0; i < data->tempnr; i++) {
  1028. if ((err = device_create_file(dev,
  1029. &temp_input[i].dev_attr))
  1030. || (err = device_create_file(dev,
  1031. &temp_min[i].dev_attr))
  1032. || (err = device_create_file(dev,
  1033. &temp_max[i].dev_attr))
  1034. || (err = device_create_file(dev,
  1035. &temp_crit[i].dev_attr))
  1036. || (err = device_create_file(dev,
  1037. &temp_status[i].dev_attr))
  1038. || (err = device_create_file(dev,
  1039. &temp_min_alarm[i].dev_attr))
  1040. || (err = device_create_file(dev,
  1041. &temp_max_alarm[i].dev_attr))
  1042. || (err = device_create_file(dev,
  1043. &temp_crit_alarm[i].dev_attr))
  1044. || (err = device_create_file(dev,
  1045. &temp_fault[i].dev_attr)))
  1046. goto ERROR3;
  1047. }
  1048. if ((err = device_create_file(dev, &dev_attr_alarms_temp)))
  1049. goto ERROR3;
  1050. }
  1051. for (i = 0; i < data->fannr; i++) {
  1052. if (FAN_CONFIG_MONITOR(data->fan_conf, i)
  1053. && ((err = device_create_file(dev,
  1054. &fan_input[i].dev_attr))
  1055. || (err = device_create_file(dev,
  1056. &fan_min[i].dev_attr))
  1057. || (err = device_create_file(dev,
  1058. &fan_div[i].dev_attr))
  1059. || (err = device_create_file(dev,
  1060. &fan_status[i].dev_attr))))
  1061. goto ERROR3;
  1062. if (FAN_CONFIG_CONTROL(data->fan_conf, i)
  1063. && (err = device_create_file(dev, &pwm[i].dev_attr)))
  1064. goto ERROR3;
  1065. }
  1066. if ((err = device_create_file(dev, &dev_attr_name)))
  1067. goto ERROR3;
  1068. data->hwmon_dev = hwmon_device_register(dev);
  1069. if (IS_ERR(data->hwmon_dev)) {
  1070. err = PTR_ERR(data->hwmon_dev);
  1071. goto ERROR3;
  1072. }
  1073. return 0;
  1074. ERROR3:
  1075. device_remove_file(dev, &dev_attr_name);
  1076. /* can still remove groups whose members were added individually */
  1077. sysfs_remove_group(&dev->kobj, &pc8736x_temp_group);
  1078. sysfs_remove_group(&dev->kobj, &pc8736x_fan_group);
  1079. sysfs_remove_group(&dev->kobj, &pc8736x_therm_group);
  1080. sysfs_remove_group(&dev->kobj, &pc8736x_vin_group);
  1081. for (i = 0; i < 3; i++) {
  1082. if (data->address[i]) {
  1083. release_region(data->address[i], PC87360_EXTENT);
  1084. }
  1085. }
  1086. ERROR1:
  1087. kfree(data);
  1088. return err;
  1089. }
  1090. static int __devexit pc87360_remove(struct platform_device *pdev)
  1091. {
  1092. struct pc87360_data *data = platform_get_drvdata(pdev);
  1093. int i;
  1094. hwmon_device_unregister(data->hwmon_dev);
  1095. device_remove_file(&pdev->dev, &dev_attr_name);
  1096. sysfs_remove_group(&pdev->dev.kobj, &pc8736x_temp_group);
  1097. sysfs_remove_group(&pdev->dev.kobj, &pc8736x_fan_group);
  1098. sysfs_remove_group(&pdev->dev.kobj, &pc8736x_therm_group);
  1099. sysfs_remove_group(&pdev->dev.kobj, &pc8736x_vin_group);
  1100. for (i = 0; i < 3; i++) {
  1101. if (data->address[i]) {
  1102. release_region(data->address[i], PC87360_EXTENT);
  1103. }
  1104. }
  1105. kfree(data);
  1106. return 0;
  1107. }
  1108. /* ldi is the logical device index
  1109. bank is for voltages and temperatures only */
  1110. static int pc87360_read_value(struct pc87360_data *data, u8 ldi, u8 bank,
  1111. u8 reg)
  1112. {
  1113. int res;
  1114. mutex_lock(&(data->lock));
  1115. if (bank != NO_BANK)
  1116. outb_p(bank, data->address[ldi] + PC87365_REG_BANK);
  1117. res = inb_p(data->address[ldi] + reg);
  1118. mutex_unlock(&(data->lock));
  1119. return res;
  1120. }
  1121. static void pc87360_write_value(struct pc87360_data *data, u8 ldi, u8 bank,
  1122. u8 reg, u8 value)
  1123. {
  1124. mutex_lock(&(data->lock));
  1125. if (bank != NO_BANK)
  1126. outb_p(bank, data->address[ldi] + PC87365_REG_BANK);
  1127. outb_p(value, data->address[ldi] + reg);
  1128. mutex_unlock(&(data->lock));
  1129. }
  1130. /* (temp & vin) channel conversion status register flags (pdf sec.11.5.12) */
  1131. #define CHAN_CNVRTD 0x80 /* new data ready */
  1132. #define CHAN_ENA 0x01 /* enabled channel (temp or vin) */
  1133. #define CHAN_ALM_ENA 0x10 /* propagate to alarms-reg ?? (chk val!) */
  1134. #define CHAN_READY (CHAN_ENA|CHAN_CNVRTD) /* sample ready mask */
  1135. #define TEMP_OTS_OE 0x20 /* OTS Output Enable */
  1136. #define VIN_RW1C_MASK (CHAN_READY|CHAN_ALM_MAX|CHAN_ALM_MIN) /* 0x87 */
  1137. #define TEMP_RW1C_MASK (VIN_RW1C_MASK|TEMP_ALM_CRIT|TEMP_FAULT) /* 0xCF */
  1138. static void pc87360_init_device(struct platform_device *pdev,
  1139. int use_thermistors)
  1140. {
  1141. struct pc87360_data *data = platform_get_drvdata(pdev);
  1142. int i, nr;
  1143. const u8 init_in[14] = { 2, 2, 2, 2, 2, 2, 2, 1, 1, 3, 1, 2, 2, 2 };
  1144. const u8 init_temp[3] = { 2, 2, 1 };
  1145. u8 reg;
  1146. if (init >= 2 && data->innr) {
  1147. reg = pc87360_read_value(data, LD_IN, NO_BANK,
  1148. PC87365_REG_IN_CONVRATE);
  1149. dev_info(&pdev->dev, "VLM conversion set to "
  1150. "1s period, 160us delay\n");
  1151. pc87360_write_value(data, LD_IN, NO_BANK,
  1152. PC87365_REG_IN_CONVRATE,
  1153. (reg & 0xC0) | 0x11);
  1154. }
  1155. nr = data->innr < 11 ? data->innr : 11;
  1156. for (i = 0; i < nr; i++) {
  1157. if (init >= init_in[i]) {
  1158. /* Forcibly enable voltage channel */
  1159. reg = pc87360_read_value(data, LD_IN, i,
  1160. PC87365_REG_IN_STATUS);
  1161. if (!(reg & CHAN_ENA)) {
  1162. dev_dbg(&pdev->dev, "Forcibly "
  1163. "enabling in%d\n", i);
  1164. pc87360_write_value(data, LD_IN, i,
  1165. PC87365_REG_IN_STATUS,
  1166. (reg & 0x68) | 0x87);
  1167. }
  1168. }
  1169. }
  1170. /* We can't blindly trust the Super-I/O space configuration bit,
  1171. most BIOS won't set it properly */
  1172. for (i = 11; i < data->innr; i++) {
  1173. reg = pc87360_read_value(data, LD_IN, i,
  1174. PC87365_REG_TEMP_STATUS);
  1175. use_thermistors = use_thermistors || (reg & CHAN_ENA);
  1176. }
  1177. i = use_thermistors ? 2 : 0;
  1178. for (; i < data->tempnr; i++) {
  1179. if (init >= init_temp[i]) {
  1180. /* Forcibly enable temperature channel */
  1181. reg = pc87360_read_value(data, LD_TEMP, i,
  1182. PC87365_REG_TEMP_STATUS);
  1183. if (!(reg & CHAN_ENA)) {
  1184. dev_dbg(&pdev->dev, "Forcibly "
  1185. "enabling temp%d\n", i+1);
  1186. pc87360_write_value(data, LD_TEMP, i,
  1187. PC87365_REG_TEMP_STATUS,
  1188. 0xCF);
  1189. }
  1190. }
  1191. }
  1192. if (use_thermistors) {
  1193. for (i = 11; i < data->innr; i++) {
  1194. if (init >= init_in[i]) {
  1195. /* The pin may already be used by thermal
  1196. diodes */
  1197. reg = pc87360_read_value(data, LD_TEMP,
  1198. (i-11)/2, PC87365_REG_TEMP_STATUS);
  1199. if (reg & CHAN_ENA) {
  1200. dev_dbg(&pdev->dev, "Skipping "
  1201. "temp%d, pin already in use "
  1202. "by temp%d\n", i-7, (i-11)/2);
  1203. continue;
  1204. }
  1205. /* Forcibly enable thermistor channel */
  1206. reg = pc87360_read_value(data, LD_IN, i,
  1207. PC87365_REG_IN_STATUS);
  1208. if (!(reg & CHAN_ENA)) {
  1209. dev_dbg(&pdev->dev, "Forcibly "
  1210. "enabling temp%d\n", i-7);
  1211. pc87360_write_value(data, LD_IN, i,
  1212. PC87365_REG_TEMP_STATUS,
  1213. (reg & 0x60) | 0x8F);
  1214. }
  1215. }
  1216. }
  1217. }
  1218. if (data->innr) {
  1219. reg = pc87360_read_value(data, LD_IN, NO_BANK,
  1220. PC87365_REG_IN_CONFIG);
  1221. if (reg & CHAN_ENA) {
  1222. dev_dbg(&pdev->dev, "Forcibly "
  1223. "enabling monitoring (VLM)\n");
  1224. pc87360_write_value(data, LD_IN, NO_BANK,
  1225. PC87365_REG_IN_CONFIG,
  1226. reg & 0xFE);
  1227. }
  1228. }
  1229. if (data->tempnr) {
  1230. reg = pc87360_read_value(data, LD_TEMP, NO_BANK,
  1231. PC87365_REG_TEMP_CONFIG);
  1232. if (reg & CHAN_ENA) {
  1233. dev_dbg(&pdev->dev, "Forcibly enabling "
  1234. "monitoring (TMS)\n");
  1235. pc87360_write_value(data, LD_TEMP, NO_BANK,
  1236. PC87365_REG_TEMP_CONFIG,
  1237. reg & 0xFE);
  1238. }
  1239. if (init >= 2) {
  1240. /* Chip config as documented by National Semi. */
  1241. pc87360_write_value(data, LD_TEMP, 0xF, 0xA, 0x08);
  1242. /* We voluntarily omit the bank here, in case the
  1243. sequence itself matters. It shouldn't be a problem,
  1244. since nobody else is supposed to access the
  1245. device at that point. */
  1246. pc87360_write_value(data, LD_TEMP, NO_BANK, 0xB, 0x04);
  1247. pc87360_write_value(data, LD_TEMP, NO_BANK, 0xC, 0x35);
  1248. pc87360_write_value(data, LD_TEMP, NO_BANK, 0xD, 0x05);
  1249. pc87360_write_value(data, LD_TEMP, NO_BANK, 0xE, 0x05);
  1250. }
  1251. }
  1252. }
  1253. static void pc87360_autodiv(struct device *dev, int nr)
  1254. {
  1255. struct pc87360_data *data = dev_get_drvdata(dev);
  1256. u8 old_min = data->fan_min[nr];
  1257. /* Increase clock divider if needed and possible */
  1258. if ((data->fan_status[nr] & 0x04) /* overflow flag */
  1259. || (data->fan[nr] >= 224)) { /* next to overflow */
  1260. if ((data->fan_status[nr] & 0x60) != 0x60) {
  1261. data->fan_status[nr] += 0x20;
  1262. data->fan_min[nr] >>= 1;
  1263. data->fan[nr] >>= 1;
  1264. dev_dbg(dev, "Increasing "
  1265. "clock divider to %d for fan %d\n",
  1266. FAN_DIV_FROM_REG(data->fan_status[nr]), nr+1);
  1267. }
  1268. } else {
  1269. /* Decrease clock divider if possible */
  1270. while (!(data->fan_min[nr] & 0x80) /* min "nails" divider */
  1271. && data->fan[nr] < 85 /* bad accuracy */
  1272. && (data->fan_status[nr] & 0x60) != 0x00) {
  1273. data->fan_status[nr] -= 0x20;
  1274. data->fan_min[nr] <<= 1;
  1275. data->fan[nr] <<= 1;
  1276. dev_dbg(dev, "Decreasing "
  1277. "clock divider to %d for fan %d\n",
  1278. FAN_DIV_FROM_REG(data->fan_status[nr]),
  1279. nr+1);
  1280. }
  1281. }
  1282. /* Write new fan min if it changed */
  1283. if (old_min != data->fan_min[nr]) {
  1284. pc87360_write_value(data, LD_FAN, NO_BANK,
  1285. PC87360_REG_FAN_MIN(nr),
  1286. data->fan_min[nr]);
  1287. }
  1288. }
  1289. static struct pc87360_data *pc87360_update_device(struct device *dev)
  1290. {
  1291. struct pc87360_data *data = dev_get_drvdata(dev);
  1292. u8 i;
  1293. mutex_lock(&data->update_lock);
  1294. if (time_after(jiffies, data->last_updated + HZ * 2) || !data->valid) {
  1295. dev_dbg(dev, "Data update\n");
  1296. /* Fans */
  1297. for (i = 0; i < data->fannr; i++) {
  1298. if (FAN_CONFIG_MONITOR(data->fan_conf, i)) {
  1299. data->fan_status[i] =
  1300. pc87360_read_value(data, LD_FAN,
  1301. NO_BANK, PC87360_REG_FAN_STATUS(i));
  1302. data->fan[i] = pc87360_read_value(data, LD_FAN,
  1303. NO_BANK, PC87360_REG_FAN(i));
  1304. data->fan_min[i] = pc87360_read_value(data,
  1305. LD_FAN, NO_BANK,
  1306. PC87360_REG_FAN_MIN(i));
  1307. /* Change clock divider if needed */
  1308. pc87360_autodiv(dev, i);
  1309. /* Clear bits and write new divider */
  1310. pc87360_write_value(data, LD_FAN, NO_BANK,
  1311. PC87360_REG_FAN_STATUS(i),
  1312. data->fan_status[i]);
  1313. }
  1314. if (FAN_CONFIG_CONTROL(data->fan_conf, i))
  1315. data->pwm[i] = pc87360_read_value(data, LD_FAN,
  1316. NO_BANK, PC87360_REG_PWM(i));
  1317. }
  1318. /* Voltages */
  1319. for (i = 0; i < data->innr; i++) {
  1320. data->in_status[i] = pc87360_read_value(data, LD_IN, i,
  1321. PC87365_REG_IN_STATUS);
  1322. /* Clear bits */
  1323. pc87360_write_value(data, LD_IN, i,
  1324. PC87365_REG_IN_STATUS,
  1325. data->in_status[i]);
  1326. if ((data->in_status[i] & CHAN_READY) == CHAN_READY) {
  1327. data->in[i] = pc87360_read_value(data, LD_IN,
  1328. i, PC87365_REG_IN);
  1329. }
  1330. if (data->in_status[i] & CHAN_ENA) {
  1331. data->in_min[i] = pc87360_read_value(data,
  1332. LD_IN, i,
  1333. PC87365_REG_IN_MIN);
  1334. data->in_max[i] = pc87360_read_value(data,
  1335. LD_IN, i,
  1336. PC87365_REG_IN_MAX);
  1337. if (i >= 11)
  1338. data->in_crit[i-11] =
  1339. pc87360_read_value(data, LD_IN,
  1340. i, PC87365_REG_TEMP_CRIT);
  1341. }
  1342. }
  1343. if (data->innr) {
  1344. data->in_alarms = pc87360_read_value(data, LD_IN,
  1345. NO_BANK, PC87365_REG_IN_ALARMS1)
  1346. | ((pc87360_read_value(data, LD_IN,
  1347. NO_BANK, PC87365_REG_IN_ALARMS2)
  1348. & 0x07) << 8);
  1349. data->vid = (data->vid_conf & 0xE0) ?
  1350. pc87360_read_value(data, LD_IN,
  1351. NO_BANK, PC87365_REG_VID) : 0x1F;
  1352. }
  1353. /* Temperatures */
  1354. for (i = 0; i < data->tempnr; i++) {
  1355. data->temp_status[i] = pc87360_read_value(data,
  1356. LD_TEMP, i,
  1357. PC87365_REG_TEMP_STATUS);
  1358. /* Clear bits */
  1359. pc87360_write_value(data, LD_TEMP, i,
  1360. PC87365_REG_TEMP_STATUS,
  1361. data->temp_status[i]);
  1362. if ((data->temp_status[i] & CHAN_READY) == CHAN_READY) {
  1363. data->temp[i] = pc87360_read_value(data,
  1364. LD_TEMP, i,
  1365. PC87365_REG_TEMP);
  1366. }
  1367. if (data->temp_status[i] & CHAN_ENA) {
  1368. data->temp_min[i] = pc87360_read_value(data,
  1369. LD_TEMP, i,
  1370. PC87365_REG_TEMP_MIN);
  1371. data->temp_max[i] = pc87360_read_value(data,
  1372. LD_TEMP, i,
  1373. PC87365_REG_TEMP_MAX);
  1374. data->temp_crit[i] = pc87360_read_value(data,
  1375. LD_TEMP, i,
  1376. PC87365_REG_TEMP_CRIT);
  1377. }
  1378. }
  1379. if (data->tempnr) {
  1380. data->temp_alarms = pc87360_read_value(data, LD_TEMP,
  1381. NO_BANK, PC87365_REG_TEMP_ALARMS)
  1382. & 0x3F;
  1383. }
  1384. data->last_updated = jiffies;
  1385. data->valid = 1;
  1386. }
  1387. mutex_unlock(&data->update_lock);
  1388. return data;
  1389. }
  1390. static int __init pc87360_device_add(unsigned short address)
  1391. {
  1392. struct resource res = {
  1393. .name = "pc87360",
  1394. .flags = IORESOURCE_IO,
  1395. };
  1396. int err, i;
  1397. pdev = platform_device_alloc("pc87360", address);
  1398. if (!pdev) {
  1399. err = -ENOMEM;
  1400. printk(KERN_ERR "pc87360: Device allocation failed\n");
  1401. goto exit;
  1402. }
  1403. for (i = 0; i < 3; i++) {
  1404. if (!extra_isa[i])
  1405. continue;
  1406. res.start = extra_isa[i];
  1407. res.end = extra_isa[i] + PC87360_EXTENT - 1;
  1408. err = platform_device_add_resources(pdev, &res, 1);
  1409. if (err) {
  1410. printk(KERN_ERR "pc87360: Device resource[%d] "
  1411. "addition failed (%d)\n", i, err);
  1412. goto exit_device_put;
  1413. }
  1414. }
  1415. err = platform_device_add(pdev);
  1416. if (err) {
  1417. printk(KERN_ERR "pc87360: Device addition failed (%d)\n",
  1418. err);
  1419. goto exit_device_put;
  1420. }
  1421. return 0;
  1422. exit_device_put:
  1423. platform_device_put(pdev);
  1424. exit:
  1425. return err;
  1426. }
  1427. static int __init pc87360_init(void)
  1428. {
  1429. int err, i;
  1430. unsigned short address = 0;
  1431. if (pc87360_find(0x2e, &devid, extra_isa)
  1432. && pc87360_find(0x4e, &devid, extra_isa)) {
  1433. printk(KERN_WARNING "pc87360: PC8736x not detected, "
  1434. "module not inserted.\n");
  1435. return -ENODEV;
  1436. }
  1437. /* Arbitrarily pick one of the addresses */
  1438. for (i = 0; i < 3; i++) {
  1439. if (extra_isa[i] != 0x0000) {
  1440. address = extra_isa[i];
  1441. break;
  1442. }
  1443. }
  1444. if (address == 0x0000) {
  1445. printk(KERN_WARNING "pc87360: No active logical device, "
  1446. "module not inserted.\n");
  1447. return -ENODEV;
  1448. }
  1449. err = platform_driver_register(&pc87360_driver);
  1450. if (err)
  1451. goto exit;
  1452. /* Sets global pdev as a side effect */
  1453. err = pc87360_device_add(address);
  1454. if (err)
  1455. goto exit_driver;
  1456. return 0;
  1457. exit_driver:
  1458. platform_driver_unregister(&pc87360_driver);
  1459. exit:
  1460. return err;
  1461. }
  1462. static void __exit pc87360_exit(void)
  1463. {
  1464. platform_device_unregister(pdev);
  1465. platform_driver_unregister(&pc87360_driver);
  1466. }
  1467. MODULE_AUTHOR("Jean Delvare <khali@linux-fr.org>");
  1468. MODULE_DESCRIPTION("PC8736x hardware monitor");
  1469. MODULE_LICENSE("GPL");
  1470. module_init(pc87360_init);
  1471. module_exit(pc87360_exit);