cmis_fw_update.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. #include <linux/ethtool.h>
  3. #include <linux/firmware.h>
  4. #include "common.h"
  5. #include "module_fw.h"
  6. #include "cmis.h"
  7. struct cmis_fw_update_fw_mng_features {
  8. u8 start_cmd_payload_size;
  9. u16 max_duration_start;
  10. u16 max_duration_write;
  11. u16 max_duration_complete;
  12. };
  13. /* See section 9.4.2 "CMD 0041h: Firmware Management Features" in CMIS standard
  14. * revision 5.2.
  15. * struct cmis_cdb_fw_mng_features_rpl is a structured layout of the flat
  16. * array, ethtool_cmis_cdb_rpl::payload.
  17. */
  18. struct cmis_cdb_fw_mng_features_rpl {
  19. u8 resv1;
  20. u8 resv2;
  21. u8 start_cmd_payload_size;
  22. u8 resv3;
  23. u8 read_write_len_ext;
  24. u8 write_mechanism;
  25. u8 resv4;
  26. u8 resv5;
  27. __be16 max_duration_start;
  28. __be16 resv6;
  29. __be16 max_duration_write;
  30. __be16 max_duration_complete;
  31. __be16 resv7;
  32. };
  33. enum cmis_cdb_fw_write_mechanism {
  34. CMIS_CDB_FW_WRITE_MECHANISM_LPL = 0x01,
  35. CMIS_CDB_FW_WRITE_MECHANISM_BOTH = 0x11,
  36. };
  37. static int
  38. cmis_fw_update_fw_mng_features_get(struct ethtool_cmis_cdb *cdb,
  39. struct net_device *dev,
  40. struct cmis_fw_update_fw_mng_features *fw_mng,
  41. struct ethnl_module_fw_flash_ntf_params *ntf_params)
  42. {
  43. struct ethtool_cmis_cdb_cmd_args args = {};
  44. struct cmis_cdb_fw_mng_features_rpl *rpl;
  45. u8 flags = CDB_F_STATUS_VALID;
  46. int err;
  47. ethtool_cmis_cdb_check_completion_flag(cdb->cmis_rev, &flags);
  48. ethtool_cmis_cdb_compose_args(&args,
  49. ETHTOOL_CMIS_CDB_CMD_FW_MANAGMENT_FEATURES,
  50. NULL, 0, cdb->max_completion_time,
  51. cdb->read_write_len_ext, 1000,
  52. sizeof(*rpl), flags);
  53. err = ethtool_cmis_cdb_execute_cmd(dev, &args);
  54. if (err < 0) {
  55. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  56. "FW Management Features command failed",
  57. args.err_msg);
  58. return err;
  59. }
  60. rpl = (struct cmis_cdb_fw_mng_features_rpl *)args.req.payload;
  61. if (!(rpl->write_mechanism == CMIS_CDB_FW_WRITE_MECHANISM_LPL ||
  62. rpl->write_mechanism == CMIS_CDB_FW_WRITE_MECHANISM_BOTH)) {
  63. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  64. "Write LPL is not supported",
  65. NULL);
  66. return -EOPNOTSUPP;
  67. }
  68. /* Above, we used read_write_len_ext that we got from CDB
  69. * advertisement. Update it with the value that we got from module
  70. * features query, which is specific for Firmware Management Commands
  71. * (IDs 0100h-01FFh).
  72. */
  73. cdb->read_write_len_ext = rpl->read_write_len_ext;
  74. fw_mng->start_cmd_payload_size = rpl->start_cmd_payload_size;
  75. fw_mng->max_duration_start = be16_to_cpu(rpl->max_duration_start);
  76. fw_mng->max_duration_write = be16_to_cpu(rpl->max_duration_write);
  77. fw_mng->max_duration_complete = be16_to_cpu(rpl->max_duration_complete);
  78. return 0;
  79. }
  80. /* See section 9.7.2 "CMD 0101h: Start Firmware Download" in CMIS standard
  81. * revision 5.2.
  82. * struct cmis_cdb_start_fw_download_pl is a structured layout of the
  83. * flat array, ethtool_cmis_cdb_request::payload.
  84. */
  85. struct cmis_cdb_start_fw_download_pl {
  86. __struct_group(cmis_cdb_start_fw_download_pl_h, head, /* no attrs */,
  87. __be32 image_size;
  88. __be32 resv1;
  89. );
  90. u8 vendor_data[ETHTOOL_CMIS_CDB_LPL_MAX_PL_LENGTH -
  91. sizeof(struct cmis_cdb_start_fw_download_pl_h)];
  92. };
  93. static int
  94. cmis_fw_update_start_download(struct ethtool_cmis_cdb *cdb,
  95. struct ethtool_cmis_fw_update_params *fw_update,
  96. struct cmis_fw_update_fw_mng_features *fw_mng)
  97. {
  98. u8 vendor_data_size = fw_mng->start_cmd_payload_size;
  99. struct cmis_cdb_start_fw_download_pl pl = {};
  100. struct ethtool_cmis_cdb_cmd_args args = {};
  101. u8 lpl_len;
  102. int err;
  103. pl.image_size = cpu_to_be32(fw_update->fw->size);
  104. memcpy(pl.vendor_data, fw_update->fw->data, vendor_data_size);
  105. lpl_len = offsetof(struct cmis_cdb_start_fw_download_pl,
  106. vendor_data[vendor_data_size]);
  107. ethtool_cmis_cdb_compose_args(&args,
  108. ETHTOOL_CMIS_CDB_CMD_START_FW_DOWNLOAD,
  109. (u8 *)&pl, lpl_len,
  110. fw_mng->max_duration_start,
  111. cdb->read_write_len_ext, 1000, 0,
  112. CDB_F_COMPLETION_VALID | CDB_F_STATUS_VALID);
  113. err = ethtool_cmis_cdb_execute_cmd(fw_update->dev, &args);
  114. if (err < 0)
  115. ethnl_module_fw_flash_ntf_err(fw_update->dev,
  116. &fw_update->ntf_params,
  117. "Start FW download command failed",
  118. args.err_msg);
  119. return err;
  120. }
  121. /* See section 9.7.4 "CMD 0103h: Write Firmware Block LPL" in CMIS standard
  122. * revision 5.2.
  123. * struct cmis_cdb_write_fw_block_lpl_pl is a structured layout of the
  124. * flat array, ethtool_cmis_cdb_request::payload.
  125. */
  126. struct cmis_cdb_write_fw_block_lpl_pl {
  127. __be32 block_address;
  128. u8 fw_block[ETHTOOL_CMIS_CDB_LPL_MAX_PL_LENGTH - sizeof(__be32)];
  129. };
  130. static int
  131. cmis_fw_update_write_image(struct ethtool_cmis_cdb *cdb,
  132. struct ethtool_cmis_fw_update_params *fw_update,
  133. struct cmis_fw_update_fw_mng_features *fw_mng)
  134. {
  135. u8 start = fw_mng->start_cmd_payload_size;
  136. u32 offset, max_block_size, max_lpl_len;
  137. u32 image_size = fw_update->fw->size;
  138. int err;
  139. max_lpl_len = min_t(u32,
  140. ethtool_cmis_get_max_payload_size(cdb->read_write_len_ext),
  141. ETHTOOL_CMIS_CDB_LPL_MAX_PL_LENGTH);
  142. max_block_size =
  143. max_lpl_len - sizeof_field(struct cmis_cdb_write_fw_block_lpl_pl,
  144. block_address);
  145. for (offset = start; offset < image_size; offset += max_block_size) {
  146. struct cmis_cdb_write_fw_block_lpl_pl pl = {
  147. .block_address = cpu_to_be32(offset - start),
  148. };
  149. struct ethtool_cmis_cdb_cmd_args args = {};
  150. u32 block_size, lpl_len;
  151. ethnl_module_fw_flash_ntf_in_progress(fw_update->dev,
  152. &fw_update->ntf_params,
  153. offset - start,
  154. image_size);
  155. block_size = min_t(u32, max_block_size, image_size - offset);
  156. memcpy(pl.fw_block, &fw_update->fw->data[offset], block_size);
  157. lpl_len = block_size +
  158. sizeof_field(struct cmis_cdb_write_fw_block_lpl_pl,
  159. block_address);
  160. ethtool_cmis_cdb_compose_args(&args,
  161. ETHTOOL_CMIS_CDB_CMD_WRITE_FW_BLOCK_LPL,
  162. (u8 *)&pl, lpl_len,
  163. fw_mng->max_duration_write,
  164. cdb->read_write_len_ext, 1, 0,
  165. CDB_F_COMPLETION_VALID | CDB_F_STATUS_VALID);
  166. err = ethtool_cmis_cdb_execute_cmd(fw_update->dev, &args);
  167. if (err < 0) {
  168. ethnl_module_fw_flash_ntf_err(fw_update->dev,
  169. &fw_update->ntf_params,
  170. "Write FW block LPL command failed",
  171. args.err_msg);
  172. return err;
  173. }
  174. }
  175. return 0;
  176. }
  177. static int
  178. cmis_fw_update_complete_download(struct ethtool_cmis_cdb *cdb,
  179. struct net_device *dev,
  180. struct cmis_fw_update_fw_mng_features *fw_mng,
  181. struct ethnl_module_fw_flash_ntf_params *ntf_params)
  182. {
  183. struct ethtool_cmis_cdb_cmd_args args = {};
  184. int err;
  185. ethtool_cmis_cdb_compose_args(&args,
  186. ETHTOOL_CMIS_CDB_CMD_COMPLETE_FW_DOWNLOAD,
  187. NULL, 0, fw_mng->max_duration_complete,
  188. cdb->read_write_len_ext, 1000, 0,
  189. CDB_F_COMPLETION_VALID | CDB_F_STATUS_VALID);
  190. err = ethtool_cmis_cdb_execute_cmd(dev, &args);
  191. if (err < 0)
  192. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  193. "Complete FW download command failed",
  194. args.err_msg);
  195. return err;
  196. }
  197. static int
  198. cmis_fw_update_download_image(struct ethtool_cmis_cdb *cdb,
  199. struct ethtool_cmis_fw_update_params *fw_update,
  200. struct cmis_fw_update_fw_mng_features *fw_mng)
  201. {
  202. int err;
  203. err = cmis_fw_update_start_download(cdb, fw_update, fw_mng);
  204. if (err < 0)
  205. return err;
  206. err = cmis_fw_update_write_image(cdb, fw_update, fw_mng);
  207. if (err < 0)
  208. return err;
  209. err = cmis_fw_update_complete_download(cdb, fw_update->dev, fw_mng,
  210. &fw_update->ntf_params);
  211. if (err < 0)
  212. return err;
  213. return 0;
  214. }
  215. enum {
  216. CMIS_MODULE_LOW_PWR = 1,
  217. CMIS_MODULE_READY = 3,
  218. };
  219. static bool module_is_ready(u8 data)
  220. {
  221. u8 state = (data >> 1) & 7;
  222. return state == CMIS_MODULE_READY || state == CMIS_MODULE_LOW_PWR;
  223. }
  224. #define CMIS_MODULE_READY_MAX_DURATION_MSEC 1000
  225. #define CMIS_MODULE_STATE_OFFSET 3
  226. static int
  227. cmis_fw_update_wait_for_module_state(struct net_device *dev, u8 flags)
  228. {
  229. u8 state;
  230. return ethtool_cmis_wait_for_cond(dev, flags, CDB_F_MODULE_STATE_VALID,
  231. CMIS_MODULE_READY_MAX_DURATION_MSEC,
  232. CMIS_MODULE_STATE_OFFSET,
  233. module_is_ready, NULL, &state);
  234. }
  235. /* See section 9.7.10 "CMD 0109h: Run Firmware Image" in CMIS standard
  236. * revision 5.2.
  237. * struct cmis_cdb_run_fw_image_pl is a structured layout of the flat
  238. * array, ethtool_cmis_cdb_request::payload.
  239. */
  240. struct cmis_cdb_run_fw_image_pl {
  241. u8 resv1;
  242. u8 image_to_run;
  243. u16 delay_to_reset;
  244. };
  245. static int
  246. cmis_fw_update_run_image(struct ethtool_cmis_cdb *cdb, struct net_device *dev,
  247. struct ethnl_module_fw_flash_ntf_params *ntf_params)
  248. {
  249. struct ethtool_cmis_cdb_cmd_args args = {};
  250. struct cmis_cdb_run_fw_image_pl pl = {0};
  251. int err;
  252. ethtool_cmis_cdb_compose_args(&args, ETHTOOL_CMIS_CDB_CMD_RUN_FW_IMAGE,
  253. (u8 *)&pl, sizeof(pl),
  254. cdb->max_completion_time,
  255. cdb->read_write_len_ext, 1000, 0,
  256. CDB_F_MODULE_STATE_VALID);
  257. err = ethtool_cmis_cdb_execute_cmd(dev, &args);
  258. if (err < 0) {
  259. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  260. "Run image command failed",
  261. args.err_msg);
  262. return err;
  263. }
  264. err = cmis_fw_update_wait_for_module_state(dev, args.flags);
  265. if (err < 0)
  266. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  267. "Module is not ready on time after reset",
  268. NULL);
  269. return err;
  270. }
  271. static int
  272. cmis_fw_update_commit_image(struct ethtool_cmis_cdb *cdb,
  273. struct net_device *dev,
  274. struct ethnl_module_fw_flash_ntf_params *ntf_params)
  275. {
  276. struct ethtool_cmis_cdb_cmd_args args = {};
  277. int err;
  278. ethtool_cmis_cdb_compose_args(&args,
  279. ETHTOOL_CMIS_CDB_CMD_COMMIT_FW_IMAGE,
  280. NULL, 0, cdb->max_completion_time,
  281. cdb->read_write_len_ext, 1000, 0,
  282. CDB_F_COMPLETION_VALID | CDB_F_STATUS_VALID);
  283. err = ethtool_cmis_cdb_execute_cmd(dev, &args);
  284. if (err < 0)
  285. ethnl_module_fw_flash_ntf_err(dev, ntf_params,
  286. "Commit image command failed",
  287. args.err_msg);
  288. return err;
  289. }
  290. static int cmis_fw_update_reset(struct net_device *dev)
  291. {
  292. __u32 reset_data = ETH_RESET_PHY;
  293. return dev->ethtool_ops->reset(dev, &reset_data);
  294. }
  295. void
  296. ethtool_cmis_fw_update(struct ethtool_cmis_fw_update_params *fw_update)
  297. {
  298. struct ethnl_module_fw_flash_ntf_params *ntf_params =
  299. &fw_update->ntf_params;
  300. struct cmis_fw_update_fw_mng_features fw_mng = {0};
  301. struct net_device *dev = fw_update->dev;
  302. struct ethtool_cmis_cdb *cdb;
  303. int err;
  304. cdb = ethtool_cmis_cdb_init(dev, &fw_update->params, ntf_params);
  305. if (IS_ERR(cdb))
  306. goto err_send_ntf;
  307. ethnl_module_fw_flash_ntf_start(dev, ntf_params);
  308. err = cmis_fw_update_fw_mng_features_get(cdb, dev, &fw_mng, ntf_params);
  309. if (err < 0)
  310. goto err_cdb_fini;
  311. err = cmis_fw_update_download_image(cdb, fw_update, &fw_mng);
  312. if (err < 0)
  313. goto err_cdb_fini;
  314. err = cmis_fw_update_run_image(cdb, dev, ntf_params);
  315. if (err < 0)
  316. goto err_cdb_fini;
  317. /* The CDB command "Run Firmware Image" resets the firmware, so the new
  318. * one might have different settings.
  319. * Free the old CDB instance, and init a new one.
  320. */
  321. ethtool_cmis_cdb_fini(cdb);
  322. cdb = ethtool_cmis_cdb_init(dev, &fw_update->params, ntf_params);
  323. if (IS_ERR(cdb))
  324. goto err_send_ntf;
  325. err = cmis_fw_update_commit_image(cdb, dev, ntf_params);
  326. if (err < 0)
  327. goto err_cdb_fini;
  328. err = cmis_fw_update_reset(dev);
  329. if (err < 0)
  330. goto err_cdb_fini;
  331. ethnl_module_fw_flash_ntf_complete(dev, ntf_params);
  332. ethtool_cmis_cdb_fini(cdb);
  333. return;
  334. err_cdb_fini:
  335. ethtool_cmis_cdb_fini(cdb);
  336. err_send_ntf:
  337. ethnl_module_fw_flash_ntf_err(dev, ntf_params, NULL, NULL);
  338. }