wd.c 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  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. struct mei_cl *cl;
  231. int ret;
  232. dev = watchdog_get_drvdata(wd_dev);
  233. if (!dev)
  234. return -ENODEV;
  235. cl = &dev->wd_cl;
  236. mutex_lock(&dev->device_lock);
  237. if (cl->state != MEI_FILE_CONNECTED) {
  238. dev_err(dev->dev, "wd: not connected.\n");
  239. ret = -ENODEV;
  240. goto end;
  241. }
  242. dev->wd_state = MEI_WD_RUNNING;
  243. ret = mei_cl_flow_ctrl_creds(cl);
  244. if (ret < 0)
  245. goto end;
  246. /* Check if we can send the ping to HW*/
  247. if (ret && mei_hbuf_acquire(dev)) {
  248. dev_dbg(dev->dev, "wd: sending ping\n");
  249. ret = mei_wd_send(dev);
  250. if (ret)
  251. goto end;
  252. dev->wd_pending = false;
  253. } else {
  254. dev->wd_pending = true;
  255. }
  256. end:
  257. mutex_unlock(&dev->device_lock);
  258. return ret;
  259. }
  260. /*
  261. * mei_wd_ops_set_timeout - wd set timeout command from the watchdog core.
  262. *
  263. * @wd_dev - watchdog device struct
  264. * @timeout - timeout value to set
  265. *
  266. * Return: 0 if success, negative errno code for failure
  267. */
  268. static int mei_wd_ops_set_timeout(struct watchdog_device *wd_dev,
  269. unsigned int timeout)
  270. {
  271. struct mei_device *dev;
  272. dev = watchdog_get_drvdata(wd_dev);
  273. if (!dev)
  274. return -ENODEV;
  275. /* Check Timeout value */
  276. if (timeout < MEI_WD_MIN_TIMEOUT || timeout > MEI_WD_MAX_TIMEOUT)
  277. return -EINVAL;
  278. mutex_lock(&dev->device_lock);
  279. dev->wd_timeout = timeout;
  280. wd_dev->timeout = timeout;
  281. mei_wd_set_start_timeout(dev, dev->wd_timeout);
  282. mutex_unlock(&dev->device_lock);
  283. return 0;
  284. }
  285. /*
  286. * Watchdog Device structs
  287. */
  288. static const struct watchdog_ops wd_ops = {
  289. .owner = THIS_MODULE,
  290. .start = mei_wd_ops_start,
  291. .stop = mei_wd_ops_stop,
  292. .ping = mei_wd_ops_ping,
  293. .set_timeout = mei_wd_ops_set_timeout,
  294. };
  295. static const struct watchdog_info wd_info = {
  296. .identity = INTEL_AMT_WATCHDOG_ID,
  297. .options = WDIOF_KEEPALIVEPING |
  298. WDIOF_SETTIMEOUT |
  299. WDIOF_ALARMONLY,
  300. };
  301. static struct watchdog_device amt_wd_dev = {
  302. .info = &wd_info,
  303. .ops = &wd_ops,
  304. .timeout = MEI_WD_DEFAULT_TIMEOUT,
  305. .min_timeout = MEI_WD_MIN_TIMEOUT,
  306. .max_timeout = MEI_WD_MAX_TIMEOUT,
  307. };
  308. int mei_watchdog_register(struct mei_device *dev)
  309. {
  310. int ret;
  311. /* unlock to perserve correct locking order */
  312. mutex_unlock(&dev->device_lock);
  313. ret = watchdog_register_device(&amt_wd_dev);
  314. mutex_lock(&dev->device_lock);
  315. if (ret) {
  316. dev_err(dev->dev, "wd: unable to register watchdog device = %d.\n",
  317. ret);
  318. return ret;
  319. }
  320. dev_dbg(dev->dev, "wd: successfully register watchdog interface.\n");
  321. watchdog_set_drvdata(&amt_wd_dev, dev);
  322. return 0;
  323. }
  324. void mei_watchdog_unregister(struct mei_device *dev)
  325. {
  326. if (watchdog_get_drvdata(&amt_wd_dev) == NULL)
  327. return;
  328. watchdog_set_drvdata(&amt_wd_dev, NULL);
  329. watchdog_unregister_device(&amt_wd_dev);
  330. }