wd.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. /*
  2. *
  3. * Intel Management Engine Interface (Intel MEI) Linux driver
  4. * Copyright (c) 2003-2012, Intel Corporation.
  5. *
  6. * This program is free software; you can redistribute it and/or modify it
  7. * under the terms and conditions of the GNU General Public License,
  8. * version 2, as published by the Free Software Foundation.
  9. *
  10. * This program is distributed in the hope it will be useful, but WITHOUT
  11. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  12. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  13. * more details.
  14. *
  15. */
  16. #include <linux/kernel.h>
  17. #include <linux/module.h>
  18. #include <linux/moduleparam.h>
  19. #include <linux/device.h>
  20. #include <linux/sched.h>
  21. #include <linux/watchdog.h>
  22. #include <linux/mei.h>
  23. #include "mei_dev.h"
  24. #include "hbm.h"
  25. #include "client.h"
  26. static const u8 mei_start_wd_params[] = { 0x02, 0x12, 0x13, 0x10 };
  27. static const u8 mei_stop_wd_params[] = { 0x02, 0x02, 0x14, 0x10 };
  28. /*
  29. * AMT Watchdog Device
  30. */
  31. #define INTEL_AMT_WATCHDOG_ID "INTCAMT"
  32. /* UUIDs for AMT F/W clients */
  33. const uuid_le mei_wd_guid = UUID_LE(0x05B79A6F, 0x4628, 0x4D7F, 0x89,
  34. 0x9D, 0xA9, 0x15, 0x14, 0xCB,
  35. 0x32, 0xAB);
  36. static void mei_wd_set_start_timeout(struct mei_device *dev, u16 timeout)
  37. {
  38. dev_dbg(dev->dev, "wd: set timeout=%d.\n", timeout);
  39. memcpy(dev->wd_data, mei_start_wd_params, MEI_WD_HDR_SIZE);
  40. memcpy(dev->wd_data + MEI_WD_HDR_SIZE, &timeout, sizeof(u16));
  41. }
  42. /**
  43. * mei_wd_host_init - connect to the watchdog client
  44. *
  45. * @dev: the device structure
  46. *
  47. * Return: -ENOTTY if wd client cannot be found
  48. * -EIO if write has failed
  49. * 0 on success
  50. */
  51. int mei_wd_host_init(struct mei_device *dev)
  52. {
  53. struct mei_cl *cl = &dev->wd_cl;
  54. struct mei_me_client *me_cl;
  55. int ret;
  56. mei_cl_init(cl, dev);
  57. dev->wd_timeout = MEI_WD_DEFAULT_TIMEOUT;
  58. dev->wd_state = MEI_WD_IDLE;
  59. /* check for valid client id */
  60. me_cl = mei_me_cl_by_uuid(dev, &mei_wd_guid);
  61. if (!me_cl) {
  62. dev_info(dev->dev, "wd: failed to find the client\n");
  63. return -ENOTTY;
  64. }
  65. cl->me_client_id = me_cl->client_id;
  66. cl->cl_uuid = me_cl->props.protocol_name;
  67. ret = mei_cl_link(cl, MEI_WD_HOST_CLIENT_ID);
  68. if (ret < 0) {
  69. dev_info(dev->dev, "wd: failed link client\n");
  70. return ret;
  71. }
  72. ret = mei_cl_connect(cl, NULL);
  73. if (ret) {
  74. dev_err(dev->dev, "wd: failed to connect = %d\n", ret);
  75. mei_cl_unlink(cl);
  76. return ret;
  77. }
  78. ret = mei_watchdog_register(dev);
  79. if (ret) {
  80. mei_cl_disconnect(cl);
  81. mei_cl_unlink(cl);
  82. }
  83. return ret;
  84. }
  85. /**
  86. * mei_wd_send - sends watch dog message to fw.
  87. *
  88. * @dev: the device structure
  89. *
  90. * Return: 0 if success,
  91. * -EIO when message send fails
  92. * -EINVAL when invalid message is to be sent
  93. * -ENODEV on flow control failure
  94. */
  95. int mei_wd_send(struct mei_device *dev)
  96. {
  97. struct mei_cl *cl = &dev->wd_cl;
  98. struct mei_msg_hdr hdr;
  99. int ret;
  100. hdr.host_addr = cl->host_client_id;
  101. hdr.me_addr = cl->me_client_id;
  102. hdr.msg_complete = 1;
  103. hdr.reserved = 0;
  104. hdr.internal = 0;
  105. if (!memcmp(dev->wd_data, mei_start_wd_params, MEI_WD_HDR_SIZE))
  106. hdr.length = MEI_WD_START_MSG_SIZE;
  107. else if (!memcmp(dev->wd_data, mei_stop_wd_params, MEI_WD_HDR_SIZE))
  108. hdr.length = MEI_WD_STOP_MSG_SIZE;
  109. else {
  110. dev_err(dev->dev, "wd: invalid message is to be sent, aborting\n");
  111. return -EINVAL;
  112. }
  113. ret = mei_write_message(dev, &hdr, dev->wd_data);
  114. if (ret) {
  115. dev_err(dev->dev, "wd: write message failed\n");
  116. return ret;
  117. }
  118. ret = mei_cl_flow_ctrl_reduce(cl);
  119. if (ret) {
  120. dev_err(dev->dev, "wd: flow_ctrl_reduce failed.\n");
  121. return ret;
  122. }
  123. return 0;
  124. }
  125. /**
  126. * mei_wd_stop - sends watchdog stop message to fw.
  127. *
  128. * @dev: the device structure
  129. *
  130. * Return: 0 if success
  131. * on error:
  132. * -EIO when message send fails
  133. * -EINVAL when invalid message is to be sent
  134. * -ETIME on message timeout
  135. */
  136. int mei_wd_stop(struct mei_device *dev)
  137. {
  138. int ret;
  139. if (dev->wd_cl.state != MEI_FILE_CONNECTED ||
  140. dev->wd_state != MEI_WD_RUNNING)
  141. return 0;
  142. memcpy(dev->wd_data, mei_stop_wd_params, MEI_WD_STOP_MSG_SIZE);
  143. dev->wd_state = MEI_WD_STOPPING;
  144. ret = mei_cl_flow_ctrl_creds(&dev->wd_cl);
  145. if (ret < 0)
  146. goto err;
  147. if (ret && mei_hbuf_acquire(dev)) {
  148. ret = mei_wd_send(dev);
  149. if (ret)
  150. goto err;
  151. dev->wd_pending = false;
  152. } else {
  153. dev->wd_pending = true;
  154. }
  155. mutex_unlock(&dev->device_lock);
  156. ret = wait_event_timeout(dev->wait_stop_wd,
  157. dev->wd_state == MEI_WD_IDLE,
  158. msecs_to_jiffies(MEI_WD_STOP_TIMEOUT));
  159. mutex_lock(&dev->device_lock);
  160. if (dev->wd_state != MEI_WD_IDLE) {
  161. /* timeout */
  162. ret = -ETIME;
  163. dev_warn(dev->dev, "wd: stop failed to complete ret=%d\n", ret);
  164. goto err;
  165. }
  166. dev_dbg(dev->dev, "wd: stop completed after %u msec\n",
  167. MEI_WD_STOP_TIMEOUT - jiffies_to_msecs(ret));
  168. return 0;
  169. err:
  170. return ret;
  171. }
  172. /*
  173. * mei_wd_ops_start - wd start command from the watchdog core.
  174. *
  175. * @wd_dev - watchdog device struct
  176. *
  177. * Return: 0 if success, negative errno code for failure
  178. */
  179. static int mei_wd_ops_start(struct watchdog_device *wd_dev)
  180. {
  181. int err = -ENODEV;
  182. struct mei_device *dev;
  183. dev = watchdog_get_drvdata(wd_dev);
  184. if (!dev)
  185. return -ENODEV;
  186. mutex_lock(&dev->device_lock);
  187. if (dev->dev_state != MEI_DEV_ENABLED) {
  188. dev_dbg(dev->dev, "wd: dev_state != MEI_DEV_ENABLED dev_state = %s\n",
  189. mei_dev_state_str(dev->dev_state));
  190. goto end_unlock;
  191. }
  192. if (dev->wd_cl.state != MEI_FILE_CONNECTED) {
  193. dev_dbg(dev->dev, "MEI Driver is not connected to Watchdog Client\n");
  194. goto end_unlock;
  195. }
  196. mei_wd_set_start_timeout(dev, dev->wd_timeout);
  197. err = 0;
  198. end_unlock:
  199. mutex_unlock(&dev->device_lock);
  200. return err;
  201. }
  202. /*
  203. * mei_wd_ops_stop - wd stop command from the watchdog core.
  204. *
  205. * @wd_dev - watchdog device struct
  206. *
  207. * Return: 0 if success, negative errno code for failure
  208. */
  209. static int mei_wd_ops_stop(struct watchdog_device *wd_dev)
  210. {
  211. struct mei_device *dev;
  212. dev = watchdog_get_drvdata(wd_dev);
  213. if (!dev)
  214. return -ENODEV;
  215. mutex_lock(&dev->device_lock);
  216. mei_wd_stop(dev);
  217. mutex_unlock(&dev->device_lock);
  218. return 0;
  219. }
  220. /*
  221. * mei_wd_ops_ping - wd ping command from the watchdog core.
  222. *
  223. * @wd_dev - watchdog device struct
  224. *
  225. * Return: 0 if success, negative errno code for failure
  226. */
  227. static int mei_wd_ops_ping(struct watchdog_device *wd_dev)
  228. {
  229. struct mei_device *dev;
  230. int ret;
  231. dev = watchdog_get_drvdata(wd_dev);
  232. if (!dev)
  233. return -ENODEV;
  234. mutex_lock(&dev->device_lock);
  235. if (dev->wd_cl.state != MEI_FILE_CONNECTED) {
  236. dev_err(dev->dev, "wd: not connected.\n");
  237. ret = -ENODEV;
  238. goto end;
  239. }
  240. dev->wd_state = MEI_WD_RUNNING;
  241. ret = mei_cl_flow_ctrl_creds(&dev->wd_cl);
  242. if (ret < 0)
  243. goto end;
  244. /* Check if we can send the ping to HW*/
  245. if (ret && mei_hbuf_acquire(dev)) {
  246. dev_dbg(dev->dev, "wd: sending ping\n");
  247. ret = mei_wd_send(dev);
  248. if (ret)
  249. goto end;
  250. dev->wd_pending = false;
  251. } else {
  252. dev->wd_pending = true;
  253. }
  254. end:
  255. mutex_unlock(&dev->device_lock);
  256. return ret;
  257. }
  258. /*
  259. * mei_wd_ops_set_timeout - wd set timeout command from the watchdog core.
  260. *
  261. * @wd_dev - watchdog device struct
  262. * @timeout - timeout value to set
  263. *
  264. * Return: 0 if success, negative errno code for failure
  265. */
  266. static int mei_wd_ops_set_timeout(struct watchdog_device *wd_dev,
  267. unsigned int timeout)
  268. {
  269. struct mei_device *dev;
  270. dev = watchdog_get_drvdata(wd_dev);
  271. if (!dev)
  272. return -ENODEV;
  273. /* Check Timeout value */
  274. if (timeout < MEI_WD_MIN_TIMEOUT || timeout > MEI_WD_MAX_TIMEOUT)
  275. return -EINVAL;
  276. mutex_lock(&dev->device_lock);
  277. dev->wd_timeout = timeout;
  278. wd_dev->timeout = timeout;
  279. mei_wd_set_start_timeout(dev, dev->wd_timeout);
  280. mutex_unlock(&dev->device_lock);
  281. return 0;
  282. }
  283. /*
  284. * Watchdog Device structs
  285. */
  286. static const struct watchdog_ops wd_ops = {
  287. .owner = THIS_MODULE,
  288. .start = mei_wd_ops_start,
  289. .stop = mei_wd_ops_stop,
  290. .ping = mei_wd_ops_ping,
  291. .set_timeout = mei_wd_ops_set_timeout,
  292. };
  293. static const struct watchdog_info wd_info = {
  294. .identity = INTEL_AMT_WATCHDOG_ID,
  295. .options = WDIOF_KEEPALIVEPING |
  296. WDIOF_SETTIMEOUT |
  297. WDIOF_ALARMONLY,
  298. };
  299. static struct watchdog_device amt_wd_dev = {
  300. .info = &wd_info,
  301. .ops = &wd_ops,
  302. .timeout = MEI_WD_DEFAULT_TIMEOUT,
  303. .min_timeout = MEI_WD_MIN_TIMEOUT,
  304. .max_timeout = MEI_WD_MAX_TIMEOUT,
  305. };
  306. int mei_watchdog_register(struct mei_device *dev)
  307. {
  308. int ret;
  309. /* unlock to perserve correct locking order */
  310. mutex_unlock(&dev->device_lock);
  311. ret = watchdog_register_device(&amt_wd_dev);
  312. mutex_lock(&dev->device_lock);
  313. if (ret) {
  314. dev_err(dev->dev, "wd: unable to register watchdog device = %d.\n",
  315. ret);
  316. return ret;
  317. }
  318. dev_dbg(dev->dev, "wd: successfully register watchdog interface.\n");
  319. watchdog_set_drvdata(&amt_wd_dev, dev);
  320. return 0;
  321. }
  322. void mei_watchdog_unregister(struct mei_device *dev)
  323. {
  324. if (watchdog_get_drvdata(&amt_wd_dev) == NULL)
  325. return;
  326. watchdog_set_drvdata(&amt_wd_dev, NULL);
  327. watchdog_unregister_device(&amt_wd_dev);
  328. }