cros_ec_lpc.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. /*
  2. * Chromium OS cros_ec driver - LPC interface
  3. *
  4. * Copyright (c) 2012 The Chromium OS Authors.
  5. *
  6. * SPDX-License-Identifier: GPL-2.0+
  7. */
  8. /*
  9. * The Matrix Keyboard Protocol driver handles talking to the keyboard
  10. * controller chip. Mostly this is for keyboard functions, but some other
  11. * things have slipped in, so we provide generic services to talk to the
  12. * KBC.
  13. */
  14. #include <common.h>
  15. #include <command.h>
  16. #include <cros_ec.h>
  17. #include <asm/io.h>
  18. #ifdef DEBUG_TRACE
  19. #define debug_trace(fmt, b...) debug(fmt, ##b)
  20. #else
  21. #define debug_trace(fmt, b...)
  22. #endif
  23. static int wait_for_sync(struct cros_ec_dev *dev)
  24. {
  25. unsigned long start;
  26. start = get_timer(0);
  27. while (inb(EC_LPC_ADDR_HOST_CMD) & EC_LPC_STATUS_BUSY_MASK) {
  28. if (get_timer(start) > 1000) {
  29. debug("%s: Timeout waiting for CROS_EC sync\n",
  30. __func__);
  31. return -1;
  32. }
  33. }
  34. return 0;
  35. }
  36. /**
  37. * Send a command to a LPC CROS_EC device and return the reply.
  38. *
  39. * The device's internal input/output buffers are used.
  40. *
  41. * @param dev CROS_EC device
  42. * @param cmd Command to send (EC_CMD_...)
  43. * @param cmd_version Version of command to send (EC_VER_...)
  44. * @param dout Output data (may be NULL If dout_len=0)
  45. * @param dout_len Size of output data in bytes
  46. * @param dinp Place to put pointer to response data
  47. * @param din_len Maximum size of response in bytes
  48. * @return number of bytes in response, or -1 on error
  49. */
  50. static int old_lpc_command(struct cros_ec_dev *dev, uint8_t cmd,
  51. const uint8_t *dout, int dout_len,
  52. uint8_t **dinp, int din_len)
  53. {
  54. int ret, i;
  55. if (dout_len > EC_OLD_PARAM_SIZE) {
  56. debug("%s: Cannot send %d bytes\n", __func__, dout_len);
  57. return -1;
  58. }
  59. if (din_len > EC_OLD_PARAM_SIZE) {
  60. debug("%s: Cannot receive %d bytes\n", __func__, din_len);
  61. return -1;
  62. }
  63. if (wait_for_sync(dev)) {
  64. debug("%s: Timeout waiting ready\n", __func__);
  65. return -1;
  66. }
  67. debug_trace("cmd: %02x, ", cmd);
  68. for (i = 0; i < dout_len; i++) {
  69. debug_trace("%02x ", dout[i]);
  70. outb(dout[i], EC_LPC_ADDR_OLD_PARAM + i);
  71. }
  72. outb(cmd, EC_LPC_ADDR_HOST_CMD);
  73. debug_trace("\n");
  74. if (wait_for_sync(dev)) {
  75. debug("%s: Timeout waiting ready\n", __func__);
  76. return -1;
  77. }
  78. ret = inb(EC_LPC_ADDR_HOST_DATA);
  79. if (ret) {
  80. debug("%s: CROS_EC result code %d\n", __func__, ret);
  81. return -ret;
  82. }
  83. debug_trace("resp: %02x, ", ret);
  84. for (i = 0; i < din_len; i++) {
  85. dev->din[i] = inb(EC_LPC_ADDR_OLD_PARAM + i);
  86. debug_trace("%02x ", dev->din[i]);
  87. }
  88. debug_trace("\n");
  89. *dinp = dev->din;
  90. return din_len;
  91. }
  92. int cros_ec_lpc_command(struct cros_ec_dev *dev, uint8_t cmd, int cmd_version,
  93. const uint8_t *dout, int dout_len,
  94. uint8_t **dinp, int din_len)
  95. {
  96. const int cmd_addr = EC_LPC_ADDR_HOST_CMD;
  97. const int data_addr = EC_LPC_ADDR_HOST_DATA;
  98. const int args_addr = EC_LPC_ADDR_HOST_ARGS;
  99. const int param_addr = EC_LPC_ADDR_HOST_PARAM;
  100. struct ec_lpc_host_args args;
  101. uint8_t *d;
  102. int csum;
  103. int i;
  104. /* Fall back to old-style command interface if args aren't supported */
  105. if (!dev->cmd_version_is_supported)
  106. return old_lpc_command(dev, cmd, dout, dout_len, dinp,
  107. din_len);
  108. if (dout_len > EC_HOST_PARAM_SIZE) {
  109. debug("%s: Cannot send %d bytes\n", __func__, dout_len);
  110. return -1;
  111. }
  112. /* Fill in args */
  113. args.flags = EC_HOST_ARGS_FLAG_FROM_HOST;
  114. args.command_version = cmd_version;
  115. args.data_size = dout_len;
  116. /* Calculate checksum */
  117. csum = cmd + args.flags + args.command_version + args.data_size;
  118. for (i = 0, d = (uint8_t *)dout; i < dout_len; i++, d++)
  119. csum += *d;
  120. args.checksum = (uint8_t)csum;
  121. if (wait_for_sync(dev)) {
  122. debug("%s: Timeout waiting ready\n", __func__);
  123. return -1;
  124. }
  125. /* Write args */
  126. for (i = 0, d = (uint8_t *)&args; i < sizeof(args); i++, d++)
  127. outb(*d, args_addr + i);
  128. /* Write data, if any */
  129. debug_trace("cmd: %02x, ver: %02x", cmd, cmd_version);
  130. for (i = 0, d = (uint8_t *)dout; i < dout_len; i++, d++) {
  131. outb(*d, param_addr + i);
  132. debug_trace("%02x ", *d);
  133. }
  134. outb(cmd, cmd_addr);
  135. debug_trace("\n");
  136. if (wait_for_sync(dev)) {
  137. debug("%s: Timeout waiting for response\n", __func__);
  138. return -1;
  139. }
  140. /* Check result */
  141. i = inb(data_addr);
  142. if (i) {
  143. debug("%s: CROS_EC result code %d\n", __func__, i);
  144. return -i;
  145. }
  146. /* Read back args */
  147. for (i = 0, d = (uint8_t *)&args; i < sizeof(args); i++, d++)
  148. *d = inb(args_addr + i);
  149. /*
  150. * If EC didn't modify args flags, then somehow we sent a new-style
  151. * command to an old EC, which means it would have read its params
  152. * from the wrong place.
  153. */
  154. if (!(args.flags & EC_HOST_ARGS_FLAG_TO_HOST)) {
  155. debug("%s: CROS_EC protocol mismatch\n", __func__);
  156. return -EC_RES_INVALID_RESPONSE;
  157. }
  158. if (args.data_size > din_len) {
  159. debug("%s: CROS_EC returned too much data %d > %d\n",
  160. __func__, args.data_size, din_len);
  161. return -EC_RES_INVALID_RESPONSE;
  162. }
  163. /* Read data, if any */
  164. for (i = 0, d = (uint8_t *)dev->din; i < args.data_size; i++, d++) {
  165. *d = inb(param_addr + i);
  166. debug_trace("%02x ", *d);
  167. }
  168. debug_trace("\n");
  169. /* Verify checksum */
  170. csum = cmd + args.flags + args.command_version + args.data_size;
  171. for (i = 0, d = (uint8_t *)dev->din; i < args.data_size; i++, d++)
  172. csum += *d;
  173. if (args.checksum != (uint8_t)csum) {
  174. debug("%s: CROS_EC response has invalid checksum\n", __func__);
  175. return -EC_RES_INVALID_CHECKSUM;
  176. }
  177. *dinp = dev->din;
  178. /* Return actual amount of data received */
  179. return args.data_size;
  180. }
  181. /**
  182. * Initialize LPC protocol.
  183. *
  184. * @param dev CROS_EC device
  185. * @param blob Device tree blob
  186. * @return 0 if ok, -1 on error
  187. */
  188. int cros_ec_lpc_init(struct cros_ec_dev *dev, const void *blob)
  189. {
  190. int byte, i;
  191. /* See if we can find an EC at the other end */
  192. byte = 0xff;
  193. byte &= inb(EC_LPC_ADDR_HOST_CMD);
  194. byte &= inb(EC_LPC_ADDR_HOST_DATA);
  195. for (i = 0; i < EC_HOST_PARAM_SIZE && (byte == 0xff); i++)
  196. byte &= inb(EC_LPC_ADDR_HOST_PARAM + i);
  197. if (byte == 0xff) {
  198. debug("%s: CROS_EC device not found on LPC bus\n",
  199. __func__);
  200. return -1;
  201. }
  202. return 0;
  203. }
  204. /*
  205. * Test if LPC command args are supported.
  206. *
  207. * The cheapest way to do this is by looking for the memory-mapped
  208. * flag. This is faster than sending a new-style 'hello' command and
  209. * seeing whether the EC sets the EC_HOST_ARGS_FLAG_FROM_HOST flag
  210. * in args when it responds.
  211. */
  212. int cros_ec_lpc_check_version(struct cros_ec_dev *dev)
  213. {
  214. if (inb(EC_LPC_ADDR_MEMMAP + EC_MEMMAP_ID) == 'E' &&
  215. inb(EC_LPC_ADDR_MEMMAP + EC_MEMMAP_ID + 1)
  216. == 'C' &&
  217. (inb(EC_LPC_ADDR_MEMMAP +
  218. EC_MEMMAP_HOST_CMD_FLAGS) &
  219. EC_HOST_CMD_FLAG_LPC_ARGS_SUPPORTED)) {
  220. dev->cmd_version_is_supported = 1;
  221. } else {
  222. /* We are going to use the old IO ports */
  223. dev->cmd_version_is_supported = 0;
  224. }
  225. debug("lpc: version %s\n", dev->cmd_version_is_supported ?
  226. "new" : "old");
  227. return 0;
  228. }