canopen_program.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /*
  2. * Copyright (c) 2020 Vestas Wind Systems A/S
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <CANopen.h>
  7. #include <canopennode.h>
  8. #include <dfu/flash_img.h>
  9. #include <dfu/mcuboot.h>
  10. #include <storage/flash_map.h>
  11. #include <sys/crc.h>
  12. #define LOG_LEVEL CONFIG_CANOPEN_LOG_LEVEL
  13. #include <logging/log.h>
  14. LOG_MODULE_REGISTER(canopen_program);
  15. /* Object dictionary indexes */
  16. #define OD_H1F50_PROGRAM_DATA 0x1F50
  17. #define OD_H1F51_PROGRAM_CTRL 0x1F51
  18. #define OD_H1F56_PROGRAM_SWID 0x1F56
  19. #define OD_H1F57_FLASH_STATUS 0x1F57
  20. /* Common program control commands and status */
  21. #define PROGRAM_CTRL_STOP 0x00
  22. #define PROGRAM_CTRL_START 0x01
  23. #define PROGRAM_CTRL_RESET 0x02
  24. #define PROGRAM_CTRL_CLEAR 0x03
  25. /* Zephyr specific program control and status */
  26. #define PROGRAM_CTRL_ZEPHYR_CONFIRM 0x80
  27. /* Flash status bits */
  28. #define FLASH_STATUS_IN_PROGRESS BIT(0)
  29. /* Flash common error bits values */
  30. #define FLASH_STATUS_NO_ERROR (0U << 1U)
  31. #define FLASH_STATUS_NO_VALID_PROGRAM (1U << 1U)
  32. #define FLASH_STATUS_DATA_FORMAT_UNKNOWN (2U << 1U)
  33. #define FLASH_STATUS_DATA_FORMAT_ERROR (3U << 1U)
  34. #define FLASH_STATUS_FLASH_NOT_CLEARED (4U << 1U)
  35. #define FLASH_STATUS_FLASH_WRITE_ERROR (5U << 1U)
  36. #define FLASH_STATUS_GENERAL_ADDR_ERROR (6U << 1U)
  37. #define FLASH_STATUS_FLASH_SECURED (7U << 1U)
  38. #define FLASH_STATUS_UNSPECIFIED_ERROR (63U << 1)
  39. struct canopen_program_context {
  40. uint32_t flash_status;
  41. size_t total;
  42. CO_NMT_t *nmt;
  43. CO_EM_t *em;
  44. struct flash_img_context flash_img_ctx;
  45. uint8_t program_status;
  46. bool flash_written;
  47. };
  48. static struct canopen_program_context ctx;
  49. static void canopen_program_set_status(uint32_t status)
  50. {
  51. ctx.program_status = status;
  52. }
  53. static uint32_t canopen_program_get_status(void)
  54. {
  55. /*
  56. * Non-confirmed boot image takes precedence over other
  57. * status. This must be checked on every invocation since the
  58. * app may be using other means of confirming the image.
  59. */
  60. if (!boot_is_img_confirmed()) {
  61. return PROGRAM_CTRL_ZEPHYR_CONFIRM;
  62. }
  63. return ctx.program_status;
  64. }
  65. static CO_SDO_abortCode_t canopen_odf_1f50(CO_ODF_arg_t *odf_arg)
  66. {
  67. int err;
  68. if (odf_arg->subIndex != 1U) {
  69. return CO_SDO_AB_NONE;
  70. }
  71. if (odf_arg->reading) {
  72. return CO_SDO_AB_WRITEONLY;
  73. }
  74. if (canopen_program_get_status() != PROGRAM_CTRL_CLEAR) {
  75. ctx.flash_status = FLASH_STATUS_FLASH_NOT_CLEARED;
  76. return CO_SDO_AB_DATA_DEV_STATE;
  77. }
  78. if (odf_arg->firstSegment) {
  79. err = flash_img_init(&ctx.flash_img_ctx);
  80. if (err) {
  81. LOG_ERR("failed to initialize flash img (err %d)", err);
  82. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  83. CO_EMC_HARDWARE, err);
  84. ctx.flash_status = FLASH_STATUS_FLASH_WRITE_ERROR;
  85. return CO_SDO_AB_HW;
  86. }
  87. ctx.flash_status = FLASH_STATUS_IN_PROGRESS;
  88. if (IS_ENABLED(CONFIG_CANOPENNODE_LEDS)) {
  89. canopen_leds_program_download(true);
  90. }
  91. ctx.total = odf_arg->dataLengthTotal;
  92. LOG_DBG("total = %d", ctx.total);
  93. }
  94. err = flash_img_buffered_write(&ctx.flash_img_ctx, odf_arg->data,
  95. odf_arg->dataLength,
  96. odf_arg->lastSegment);
  97. if (err) {
  98. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  99. CO_EMC_HARDWARE, err);
  100. ctx.flash_status = FLASH_STATUS_FLASH_WRITE_ERROR;
  101. canopen_leds_program_download(false);
  102. return CO_SDO_AB_HW;
  103. }
  104. if (odf_arg->lastSegment) {
  105. /* ctx.total is zero if not provided by download process */
  106. if (ctx.total != 0 &&
  107. ctx.total != flash_img_bytes_written(&ctx.flash_img_ctx)) {
  108. LOG_WRN("premature end of program download");
  109. ctx.flash_status = FLASH_STATUS_DATA_FORMAT_ERROR;
  110. } else {
  111. LOG_DBG("program downloaded");
  112. ctx.flash_written = true;
  113. ctx.flash_status = FLASH_STATUS_NO_ERROR;
  114. }
  115. canopen_program_set_status(PROGRAM_CTRL_STOP);
  116. canopen_leds_program_download(false);
  117. }
  118. return CO_SDO_AB_NONE;
  119. }
  120. static inline CO_SDO_abortCode_t canopen_program_cmd_stop(void)
  121. {
  122. if (canopen_program_get_status() == PROGRAM_CTRL_ZEPHYR_CONFIRM) {
  123. return CO_SDO_AB_DATA_DEV_STATE;
  124. }
  125. LOG_DBG("program stopped");
  126. canopen_program_set_status(PROGRAM_CTRL_STOP);
  127. return CO_SDO_AB_NONE;
  128. }
  129. static inline CO_SDO_abortCode_t canopen_program_cmd_start(void)
  130. {
  131. int err;
  132. if (canopen_program_get_status() == PROGRAM_CTRL_ZEPHYR_CONFIRM) {
  133. return CO_SDO_AB_DATA_DEV_STATE;
  134. }
  135. if (ctx.flash_written) {
  136. LOG_DBG("requesting upgrade and reset");
  137. err = boot_request_upgrade(BOOT_UPGRADE_TEST);
  138. if (err) {
  139. LOG_ERR("failed to request upgrade (err %d)", err);
  140. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  141. CO_EMC_HARDWARE, err);
  142. return CO_SDO_AB_HW;
  143. }
  144. ctx.nmt->resetCommand = CO_RESET_APP;
  145. } else {
  146. LOG_DBG("program started");
  147. canopen_program_set_status(PROGRAM_CTRL_START);
  148. }
  149. return CO_SDO_AB_NONE;
  150. }
  151. static inline CO_SDO_abortCode_t canopen_program_cmd_clear(void)
  152. {
  153. int err;
  154. if (canopen_program_get_status() != PROGRAM_CTRL_STOP) {
  155. return CO_SDO_AB_DATA_DEV_STATE;
  156. }
  157. if (!IS_ENABLED(CONFIG_IMG_ERASE_PROGRESSIVELY)) {
  158. LOG_DBG("erasing flash area");
  159. err = boot_erase_img_bank(FLASH_AREA_ID(image_1));
  160. if (err) {
  161. LOG_ERR("failed to erase image bank (err %d)", err);
  162. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  163. CO_EMC_HARDWARE, err);
  164. return CO_SDO_AB_HW;
  165. }
  166. }
  167. LOG_DBG("program cleared");
  168. canopen_program_set_status(PROGRAM_CTRL_CLEAR);
  169. ctx.flash_status = FLASH_STATUS_NO_ERROR;
  170. ctx.flash_written = false;
  171. return CO_SDO_AB_NONE;
  172. }
  173. static inline CO_SDO_abortCode_t canopen_program_cmd_confirm(void)
  174. {
  175. int err;
  176. if (canopen_program_get_status() == PROGRAM_CTRL_ZEPHYR_CONFIRM) {
  177. err = boot_write_img_confirmed();
  178. if (err) {
  179. LOG_ERR("failed to confirm image (err %d)", err);
  180. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  181. CO_EMC_HARDWARE, err);
  182. return CO_SDO_AB_HW;
  183. }
  184. LOG_DBG("program confirmed");
  185. canopen_program_set_status(PROGRAM_CTRL_START);
  186. }
  187. return CO_SDO_AB_NONE;
  188. }
  189. static CO_SDO_abortCode_t canopen_odf_1f51(CO_ODF_arg_t *odf_arg)
  190. {
  191. CO_SDO_abortCode_t ab;
  192. uint8_t cmd;
  193. if (odf_arg->subIndex != 1U) {
  194. return CO_SDO_AB_NONE;
  195. }
  196. if (odf_arg->reading) {
  197. odf_arg->data[0] = canopen_program_get_status();
  198. return CO_SDO_AB_NONE;
  199. }
  200. if (CO_NMT_getInternalState(ctx.nmt) != CO_NMT_PRE_OPERATIONAL) {
  201. LOG_DBG("not in pre-operational state");
  202. return CO_SDO_AB_DATA_DEV_STATE;
  203. }
  204. /* Preserve old value */
  205. cmd = odf_arg->data[0];
  206. memcpy(odf_arg->data, odf_arg->ODdataStorage, sizeof(uint8_t));
  207. LOG_DBG("program status = %d, cmd = %d", canopen_program_get_status(),
  208. cmd);
  209. switch (cmd) {
  210. case PROGRAM_CTRL_STOP:
  211. ab = canopen_program_cmd_stop();
  212. break;
  213. case PROGRAM_CTRL_START:
  214. ab = canopen_program_cmd_start();
  215. break;
  216. case PROGRAM_CTRL_CLEAR:
  217. ab = canopen_program_cmd_clear();
  218. break;
  219. case PROGRAM_CTRL_ZEPHYR_CONFIRM:
  220. ab = canopen_program_cmd_confirm();
  221. break;
  222. case PROGRAM_CTRL_RESET:
  223. __fallthrough;
  224. default:
  225. LOG_DBG("unsupported command '%d'", cmd);
  226. ab = CO_SDO_AB_INVALID_VALUE;
  227. }
  228. return ab;
  229. }
  230. #ifdef CONFIG_BOOTLOADER_MCUBOOT
  231. /** @brief Calculate crc for region in flash
  232. *
  233. * @param flash_area Flash area to read from, must be open
  234. * @offset Offset to read from
  235. * @size Number of bytes to include in calculation
  236. * @pcrc Pointer to uint32_t where crc will be written if return value is 0
  237. *
  238. * @return 0 if successful, negative errno on failure
  239. */
  240. static int flash_crc(const struct flash_area *flash_area,
  241. off_t offset, size_t size, uint32_t *pcrc)
  242. {
  243. uint32_t crc = 0;
  244. uint8_t buffer[32];
  245. while (size > 0) {
  246. size_t len = MIN(size, sizeof(buffer));
  247. int err = flash_area_read(flash_area, offset, buffer, len);
  248. if (err) {
  249. return err;
  250. }
  251. crc = crc32_ieee_update(crc, buffer, len);
  252. offset += len;
  253. size -= len;
  254. }
  255. *pcrc = crc;
  256. return 0;
  257. }
  258. static CO_SDO_abortCode_t canopen_odf_1f56(CO_ODF_arg_t *odf_arg)
  259. {
  260. const struct flash_area *flash_area;
  261. struct mcuboot_img_header header;
  262. off_t offset = 0;
  263. uint32_t crc = 0;
  264. uint8_t fa_id;
  265. uint32_t len;
  266. int err;
  267. if (odf_arg->subIndex != 1U) {
  268. return CO_SDO_AB_NONE;
  269. }
  270. if (!odf_arg->reading) {
  271. /* Preserve old value */
  272. memcpy(odf_arg->data, odf_arg->ODdataStorage, sizeof(uint32_t));
  273. return CO_SDO_AB_READONLY;
  274. }
  275. /* Reading from flash and calculating crc can take 100ms or more, and
  276. * this function is called with the can od lock taken.
  277. *
  278. * Release the lock before performing time consuming work, and reacquire
  279. * before return.
  280. */
  281. CO_UNLOCK_OD();
  282. /*
  283. * Calculate the CRC32 of the image that is running or will be
  284. * started upon receiveing the next 'start' command.
  285. */
  286. if (ctx.flash_written) {
  287. fa_id = FLASH_AREA_ID(image_1);
  288. } else {
  289. fa_id = FLASH_AREA_ID(image_0);
  290. }
  291. err = boot_read_bank_header(fa_id, &header, sizeof(header));
  292. if (err) {
  293. LOG_WRN("failed to read bank header (err %d)", err);
  294. CO_setUint32(odf_arg->data, 0U);
  295. CO_LOCK_OD();
  296. return CO_SDO_AB_NONE;
  297. }
  298. if (header.mcuboot_version != 1) {
  299. LOG_WRN("unsupported mcuboot header version %d",
  300. header.mcuboot_version);
  301. CO_setUint32(odf_arg->data, 0U);
  302. CO_LOCK_OD();
  303. return CO_SDO_AB_NONE;
  304. }
  305. len = header.h.v1.image_size;
  306. err = flash_area_open(fa_id, &flash_area);
  307. if (err) {
  308. LOG_ERR("failed to open flash area (err %d)", err);
  309. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  310. CO_EMC_HARDWARE, err);
  311. CO_LOCK_OD();
  312. return CO_SDO_AB_HW;
  313. }
  314. err = flash_crc(flash_area, offset, len, &crc);
  315. flash_area_close(flash_area);
  316. if (err) {
  317. LOG_ERR("failed to read flash (err %d)", err);
  318. CO_errorReport(ctx.em, CO_EM_NON_VOLATILE_MEMORY,
  319. CO_EMC_HARDWARE, err);
  320. CO_LOCK_OD();
  321. return CO_SDO_AB_HW;
  322. }
  323. CO_setUint32(odf_arg->data, crc);
  324. CO_LOCK_OD();
  325. return CO_SDO_AB_NONE;
  326. }
  327. #endif /* CONFIG_BOOTLOADER_MCUBOOT */
  328. static CO_SDO_abortCode_t canopen_odf_1f57(CO_ODF_arg_t *odf_arg)
  329. {
  330. if (odf_arg->subIndex != 1U) {
  331. return CO_SDO_AB_NONE;
  332. }
  333. if (!odf_arg->reading) {
  334. /* Preserve old value */
  335. memcpy(odf_arg->data, odf_arg->ODdataStorage, sizeof(uint32_t));
  336. return CO_SDO_AB_READONLY;
  337. }
  338. CO_setUint32(odf_arg->data, ctx.flash_status);
  339. return CO_SDO_AB_NONE;
  340. }
  341. void canopen_program_download_attach(CO_NMT_t *nmt, CO_SDO_t *sdo, CO_EM_t *em)
  342. {
  343. canopen_program_set_status(PROGRAM_CTRL_START);
  344. ctx.flash_status = FLASH_STATUS_NO_ERROR;
  345. ctx.flash_written = false;
  346. ctx.nmt = nmt;
  347. ctx.em = em;
  348. CO_OD_configure(sdo, OD_H1F50_PROGRAM_DATA, canopen_odf_1f50,
  349. NULL, 0U, 0U);
  350. CO_OD_configure(sdo, OD_H1F51_PROGRAM_CTRL, canopen_odf_1f51,
  351. NULL, 0U, 0U);
  352. if (IS_ENABLED(CONFIG_BOOTLOADER_MCUBOOT)) {
  353. CO_OD_configure(sdo, OD_H1F56_PROGRAM_SWID, canopen_odf_1f56,
  354. NULL, 0U, 0U);
  355. }
  356. CO_OD_configure(sdo, OD_H1F57_FLASH_STATUS, canopen_odf_1f57,
  357. NULL, 0U, 0U);
  358. }