windfarm_core.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507
  1. /*
  2. * Windfarm PowerMac thermal control. Core
  3. *
  4. * (c) Copyright 2005 Benjamin Herrenschmidt, IBM Corp.
  5. * <benh@kernel.crashing.org>
  6. *
  7. * Released under the term of the GNU GPL v2.
  8. *
  9. * This core code tracks the list of sensors & controls, register
  10. * clients, and holds the kernel thread used for control.
  11. *
  12. * TODO:
  13. *
  14. * Add some information about sensor/control type and data format to
  15. * sensors/controls, and have the sysfs attribute stuff be moved
  16. * generically here instead of hard coded in the platform specific
  17. * driver as it us currently
  18. *
  19. * This however requires solving some annoying lifetime issues with
  20. * sysfs which doesn't seem to have lifetime rules for struct attribute,
  21. * I may have to create full features kobjects for every sensor/control
  22. * instead which is a bit of an overkill imho
  23. */
  24. #include <linux/types.h>
  25. #include <linux/errno.h>
  26. #include <linux/kernel.h>
  27. #include <linux/slab.h>
  28. #include <linux/init.h>
  29. #include <linux/spinlock.h>
  30. #include <linux/kthread.h>
  31. #include <linux/jiffies.h>
  32. #include <linux/reboot.h>
  33. #include <linux/device.h>
  34. #include <linux/platform_device.h>
  35. #include <linux/mutex.h>
  36. #include <linux/freezer.h>
  37. #include <asm/prom.h>
  38. #include "windfarm.h"
  39. #define VERSION "0.2"
  40. #undef DEBUG
  41. #ifdef DEBUG
  42. #define DBG(args...) printk(args)
  43. #else
  44. #define DBG(args...) do { } while(0)
  45. #endif
  46. static LIST_HEAD(wf_controls);
  47. static LIST_HEAD(wf_sensors);
  48. static DEFINE_MUTEX(wf_lock);
  49. static BLOCKING_NOTIFIER_HEAD(wf_client_list);
  50. static int wf_client_count;
  51. static unsigned int wf_overtemp;
  52. static unsigned int wf_overtemp_counter;
  53. struct task_struct *wf_thread;
  54. static struct platform_device wf_platform_device = {
  55. .name = "windfarm",
  56. };
  57. /*
  58. * Utilities & tick thread
  59. */
  60. static inline void wf_notify(int event, void *param)
  61. {
  62. blocking_notifier_call_chain(&wf_client_list, event, param);
  63. }
  64. int wf_critical_overtemp(void)
  65. {
  66. static char * critical_overtemp_path = "/sbin/critical_overtemp";
  67. char *argv[] = { critical_overtemp_path, NULL };
  68. static char *envp[] = { "HOME=/",
  69. "TERM=linux",
  70. "PATH=/sbin:/usr/sbin:/bin:/usr/bin",
  71. NULL };
  72. return call_usermodehelper(critical_overtemp_path,
  73. argv, envp, UMH_WAIT_EXEC);
  74. }
  75. EXPORT_SYMBOL_GPL(wf_critical_overtemp);
  76. static int wf_thread_func(void *data)
  77. {
  78. unsigned long next, delay;
  79. next = jiffies;
  80. DBG("wf: thread started\n");
  81. set_freezable();
  82. while (!kthread_should_stop()) {
  83. try_to_freeze();
  84. if (time_after_eq(jiffies, next)) {
  85. wf_notify(WF_EVENT_TICK, NULL);
  86. if (wf_overtemp) {
  87. wf_overtemp_counter++;
  88. /* 10 seconds overtemp, notify userland */
  89. if (wf_overtemp_counter > 10)
  90. wf_critical_overtemp();
  91. /* 30 seconds, shutdown */
  92. if (wf_overtemp_counter > 30) {
  93. printk(KERN_ERR "windfarm: Overtemp "
  94. "for more than 30"
  95. " seconds, shutting down\n");
  96. machine_power_off();
  97. }
  98. }
  99. next += HZ;
  100. }
  101. delay = next - jiffies;
  102. if (delay <= HZ)
  103. schedule_timeout_interruptible(delay);
  104. }
  105. DBG("wf: thread stopped\n");
  106. return 0;
  107. }
  108. static void wf_start_thread(void)
  109. {
  110. wf_thread = kthread_run(wf_thread_func, NULL, "kwindfarm");
  111. if (IS_ERR(wf_thread)) {
  112. printk(KERN_ERR "windfarm: failed to create thread,err %ld\n",
  113. PTR_ERR(wf_thread));
  114. wf_thread = NULL;
  115. }
  116. }
  117. static void wf_stop_thread(void)
  118. {
  119. if (wf_thread)
  120. kthread_stop(wf_thread);
  121. wf_thread = NULL;
  122. }
  123. /*
  124. * Controls
  125. */
  126. static void wf_control_release(struct kref *kref)
  127. {
  128. struct wf_control *ct = container_of(kref, struct wf_control, ref);
  129. DBG("wf: Deleting control %s\n", ct->name);
  130. if (ct->ops && ct->ops->release)
  131. ct->ops->release(ct);
  132. else
  133. kfree(ct);
  134. }
  135. static ssize_t wf_show_control(struct device *dev,
  136. struct device_attribute *attr, char *buf)
  137. {
  138. struct wf_control *ctrl = container_of(attr, struct wf_control, attr);
  139. const char *typestr;
  140. s32 val = 0;
  141. int err;
  142. err = ctrl->ops->get_value(ctrl, &val);
  143. if (err < 0) {
  144. if (err == -EFAULT)
  145. return sprintf(buf, "<HW FAULT>\n");
  146. return err;
  147. }
  148. switch(ctrl->type) {
  149. case WF_CONTROL_RPM_FAN:
  150. typestr = " RPM";
  151. break;
  152. case WF_CONTROL_PWM_FAN:
  153. typestr = " %";
  154. break;
  155. default:
  156. typestr = "";
  157. }
  158. return sprintf(buf, "%d%s\n", val, typestr);
  159. }
  160. /* This is really only for debugging... */
  161. static ssize_t wf_store_control(struct device *dev,
  162. struct device_attribute *attr,
  163. const char *buf, size_t count)
  164. {
  165. struct wf_control *ctrl = container_of(attr, struct wf_control, attr);
  166. int val;
  167. int err;
  168. char *endp;
  169. val = simple_strtoul(buf, &endp, 0);
  170. while (endp < buf + count && (*endp == ' ' || *endp == '\n'))
  171. ++endp;
  172. if (endp - buf < count)
  173. return -EINVAL;
  174. err = ctrl->ops->set_value(ctrl, val);
  175. if (err < 0)
  176. return err;
  177. return count;
  178. }
  179. int wf_register_control(struct wf_control *new_ct)
  180. {
  181. struct wf_control *ct;
  182. mutex_lock(&wf_lock);
  183. list_for_each_entry(ct, &wf_controls, link) {
  184. if (!strcmp(ct->name, new_ct->name)) {
  185. printk(KERN_WARNING "windfarm: trying to register"
  186. " duplicate control %s\n", ct->name);
  187. mutex_unlock(&wf_lock);
  188. return -EEXIST;
  189. }
  190. }
  191. kref_init(&new_ct->ref);
  192. list_add(&new_ct->link, &wf_controls);
  193. sysfs_attr_init(&new_ct->attr.attr);
  194. new_ct->attr.attr.name = new_ct->name;
  195. new_ct->attr.attr.mode = 0644;
  196. new_ct->attr.show = wf_show_control;
  197. new_ct->attr.store = wf_store_control;
  198. if (device_create_file(&wf_platform_device.dev, &new_ct->attr))
  199. printk(KERN_WARNING "windfarm: device_create_file failed"
  200. " for %s\n", new_ct->name);
  201. /* the subsystem still does useful work without the file */
  202. DBG("wf: Registered control %s\n", new_ct->name);
  203. wf_notify(WF_EVENT_NEW_CONTROL, new_ct);
  204. mutex_unlock(&wf_lock);
  205. return 0;
  206. }
  207. EXPORT_SYMBOL_GPL(wf_register_control);
  208. void wf_unregister_control(struct wf_control *ct)
  209. {
  210. mutex_lock(&wf_lock);
  211. list_del(&ct->link);
  212. mutex_unlock(&wf_lock);
  213. DBG("wf: Unregistered control %s\n", ct->name);
  214. kref_put(&ct->ref, wf_control_release);
  215. }
  216. EXPORT_SYMBOL_GPL(wf_unregister_control);
  217. struct wf_control * wf_find_control(const char *name)
  218. {
  219. struct wf_control *ct;
  220. mutex_lock(&wf_lock);
  221. list_for_each_entry(ct, &wf_controls, link) {
  222. if (!strcmp(ct->name, name)) {
  223. if (wf_get_control(ct))
  224. ct = NULL;
  225. mutex_unlock(&wf_lock);
  226. return ct;
  227. }
  228. }
  229. mutex_unlock(&wf_lock);
  230. return NULL;
  231. }
  232. EXPORT_SYMBOL_GPL(wf_find_control);
  233. int wf_get_control(struct wf_control *ct)
  234. {
  235. if (!try_module_get(ct->ops->owner))
  236. return -ENODEV;
  237. kref_get(&ct->ref);
  238. return 0;
  239. }
  240. EXPORT_SYMBOL_GPL(wf_get_control);
  241. void wf_put_control(struct wf_control *ct)
  242. {
  243. struct module *mod = ct->ops->owner;
  244. kref_put(&ct->ref, wf_control_release);
  245. module_put(mod);
  246. }
  247. EXPORT_SYMBOL_GPL(wf_put_control);
  248. /*
  249. * Sensors
  250. */
  251. static void wf_sensor_release(struct kref *kref)
  252. {
  253. struct wf_sensor *sr = container_of(kref, struct wf_sensor, ref);
  254. DBG("wf: Deleting sensor %s\n", sr->name);
  255. if (sr->ops && sr->ops->release)
  256. sr->ops->release(sr);
  257. else
  258. kfree(sr);
  259. }
  260. static ssize_t wf_show_sensor(struct device *dev,
  261. struct device_attribute *attr, char *buf)
  262. {
  263. struct wf_sensor *sens = container_of(attr, struct wf_sensor, attr);
  264. s32 val = 0;
  265. int err;
  266. err = sens->ops->get_value(sens, &val);
  267. if (err < 0)
  268. return err;
  269. return sprintf(buf, "%d.%03d\n", FIX32TOPRINT(val));
  270. }
  271. int wf_register_sensor(struct wf_sensor *new_sr)
  272. {
  273. struct wf_sensor *sr;
  274. mutex_lock(&wf_lock);
  275. list_for_each_entry(sr, &wf_sensors, link) {
  276. if (!strcmp(sr->name, new_sr->name)) {
  277. printk(KERN_WARNING "windfarm: trying to register"
  278. " duplicate sensor %s\n", sr->name);
  279. mutex_unlock(&wf_lock);
  280. return -EEXIST;
  281. }
  282. }
  283. kref_init(&new_sr->ref);
  284. list_add(&new_sr->link, &wf_sensors);
  285. sysfs_attr_init(&new_sr->attr.attr);
  286. new_sr->attr.attr.name = new_sr->name;
  287. new_sr->attr.attr.mode = 0444;
  288. new_sr->attr.show = wf_show_sensor;
  289. new_sr->attr.store = NULL;
  290. if (device_create_file(&wf_platform_device.dev, &new_sr->attr))
  291. printk(KERN_WARNING "windfarm: device_create_file failed"
  292. " for %s\n", new_sr->name);
  293. /* the subsystem still does useful work without the file */
  294. DBG("wf: Registered sensor %s\n", new_sr->name);
  295. wf_notify(WF_EVENT_NEW_SENSOR, new_sr);
  296. mutex_unlock(&wf_lock);
  297. return 0;
  298. }
  299. EXPORT_SYMBOL_GPL(wf_register_sensor);
  300. void wf_unregister_sensor(struct wf_sensor *sr)
  301. {
  302. mutex_lock(&wf_lock);
  303. list_del(&sr->link);
  304. mutex_unlock(&wf_lock);
  305. DBG("wf: Unregistered sensor %s\n", sr->name);
  306. wf_put_sensor(sr);
  307. }
  308. EXPORT_SYMBOL_GPL(wf_unregister_sensor);
  309. struct wf_sensor * wf_find_sensor(const char *name)
  310. {
  311. struct wf_sensor *sr;
  312. mutex_lock(&wf_lock);
  313. list_for_each_entry(sr, &wf_sensors, link) {
  314. if (!strcmp(sr->name, name)) {
  315. if (wf_get_sensor(sr))
  316. sr = NULL;
  317. mutex_unlock(&wf_lock);
  318. return sr;
  319. }
  320. }
  321. mutex_unlock(&wf_lock);
  322. return NULL;
  323. }
  324. EXPORT_SYMBOL_GPL(wf_find_sensor);
  325. int wf_get_sensor(struct wf_sensor *sr)
  326. {
  327. if (!try_module_get(sr->ops->owner))
  328. return -ENODEV;
  329. kref_get(&sr->ref);
  330. return 0;
  331. }
  332. EXPORT_SYMBOL_GPL(wf_get_sensor);
  333. void wf_put_sensor(struct wf_sensor *sr)
  334. {
  335. struct module *mod = sr->ops->owner;
  336. kref_put(&sr->ref, wf_sensor_release);
  337. module_put(mod);
  338. }
  339. EXPORT_SYMBOL_GPL(wf_put_sensor);
  340. /*
  341. * Client & notification
  342. */
  343. int wf_register_client(struct notifier_block *nb)
  344. {
  345. int rc;
  346. struct wf_control *ct;
  347. struct wf_sensor *sr;
  348. mutex_lock(&wf_lock);
  349. rc = blocking_notifier_chain_register(&wf_client_list, nb);
  350. if (rc != 0)
  351. goto bail;
  352. wf_client_count++;
  353. list_for_each_entry(ct, &wf_controls, link)
  354. wf_notify(WF_EVENT_NEW_CONTROL, ct);
  355. list_for_each_entry(sr, &wf_sensors, link)
  356. wf_notify(WF_EVENT_NEW_SENSOR, sr);
  357. if (wf_client_count == 1)
  358. wf_start_thread();
  359. bail:
  360. mutex_unlock(&wf_lock);
  361. return rc;
  362. }
  363. EXPORT_SYMBOL_GPL(wf_register_client);
  364. int wf_unregister_client(struct notifier_block *nb)
  365. {
  366. mutex_lock(&wf_lock);
  367. blocking_notifier_chain_unregister(&wf_client_list, nb);
  368. wf_client_count++;
  369. if (wf_client_count == 0)
  370. wf_stop_thread();
  371. mutex_unlock(&wf_lock);
  372. return 0;
  373. }
  374. EXPORT_SYMBOL_GPL(wf_unregister_client);
  375. void wf_set_overtemp(void)
  376. {
  377. mutex_lock(&wf_lock);
  378. wf_overtemp++;
  379. if (wf_overtemp == 1) {
  380. printk(KERN_WARNING "windfarm: Overtemp condition detected !\n");
  381. wf_overtemp_counter = 0;
  382. wf_notify(WF_EVENT_OVERTEMP, NULL);
  383. }
  384. mutex_unlock(&wf_lock);
  385. }
  386. EXPORT_SYMBOL_GPL(wf_set_overtemp);
  387. void wf_clear_overtemp(void)
  388. {
  389. mutex_lock(&wf_lock);
  390. WARN_ON(wf_overtemp == 0);
  391. if (wf_overtemp == 0) {
  392. mutex_unlock(&wf_lock);
  393. return;
  394. }
  395. wf_overtemp--;
  396. if (wf_overtemp == 0) {
  397. printk(KERN_WARNING "windfarm: Overtemp condition cleared !\n");
  398. wf_notify(WF_EVENT_NORMALTEMP, NULL);
  399. }
  400. mutex_unlock(&wf_lock);
  401. }
  402. EXPORT_SYMBOL_GPL(wf_clear_overtemp);
  403. int wf_is_overtemp(void)
  404. {
  405. return (wf_overtemp != 0);
  406. }
  407. EXPORT_SYMBOL_GPL(wf_is_overtemp);
  408. static int __init windfarm_core_init(void)
  409. {
  410. DBG("wf: core loaded\n");
  411. platform_device_register(&wf_platform_device);
  412. return 0;
  413. }
  414. static void __exit windfarm_core_exit(void)
  415. {
  416. BUG_ON(wf_client_count != 0);
  417. DBG("wf: core unloaded\n");
  418. platform_device_unregister(&wf_platform_device);
  419. }
  420. module_init(windfarm_core_init);
  421. module_exit(windfarm_core_exit);
  422. MODULE_AUTHOR("Benjamin Herrenschmidt <benh@kernel.crashing.org>");
  423. MODULE_DESCRIPTION("Core component of PowerMac thermal control");
  424. MODULE_LICENSE("GPL");