canopen_storage.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. * Copyright (c) 2019 Vestas Wind Systems A/S
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <settings/settings.h>
  7. #include <CANopen.h>
  8. #include <CO_Emergency.h>
  9. #include <CO_SDO.h>
  10. #include <canopennode.h>
  11. #define LOG_LEVEL CONFIG_CANOPEN_LOG_LEVEL
  12. #include <logging/log.h>
  13. LOG_MODULE_REGISTER(canopen_storage);
  14. /* 's', 'a', 'v', 'e' from LSB to MSB */
  15. #define STORE_PARAM_MAGIC 0x65766173UL
  16. /* 'l', 'o', 'a', 'd' from LSB to MSB */
  17. #define RESTORE_PARAM_MAGIC 0x64616F6CUL
  18. /* Variables for reporing errors through CANopen once the stack is up */
  19. static int canopen_storage_rom_error;
  20. static int canopen_storage_eeprom_error;
  21. static CO_SDO_abortCode_t canopen_odf_1010(CO_ODF_arg_t *odf_arg)
  22. {
  23. CO_EM_t *em = odf_arg->object;
  24. uint32_t value;
  25. int err;
  26. value = CO_getUint32(odf_arg->data);
  27. if (odf_arg->reading) {
  28. return CO_SDO_AB_NONE;
  29. }
  30. /* Preserve old value */
  31. memcpy(odf_arg->data, odf_arg->ODdataStorage, sizeof(uint32_t));
  32. if (odf_arg->subIndex != 1U) {
  33. return CO_SDO_AB_NONE;
  34. }
  35. if (value != STORE_PARAM_MAGIC) {
  36. return CO_SDO_AB_DATA_TRANSF;
  37. }
  38. err = canopen_storage_save(CANOPEN_STORAGE_ROM);
  39. if (err) {
  40. LOG_ERR("failed to save object dictionary ROM entries (err %d)",
  41. err);
  42. CO_errorReport(em, CO_EM_NON_VOLATILE_MEMORY, CO_EMC_HARDWARE,
  43. err);
  44. return CO_SDO_AB_HW;
  45. } else {
  46. LOG_DBG("saved object dictionary ROM entries");
  47. }
  48. return CO_SDO_AB_NONE;
  49. }
  50. static CO_SDO_abortCode_t canopen_odf_1011(CO_ODF_arg_t *odf_arg)
  51. {
  52. CO_EM_t *em = odf_arg->object;
  53. bool failed = false;
  54. uint32_t value;
  55. int err;
  56. value = CO_getUint32(odf_arg->data);
  57. if (odf_arg->reading) {
  58. return CO_SDO_AB_NONE;
  59. }
  60. /* Preserve old value */
  61. memcpy(odf_arg->data, odf_arg->ODdataStorage, sizeof(uint32_t));
  62. if (odf_arg->subIndex < 1U) {
  63. return CO_SDO_AB_NONE;
  64. }
  65. if (value != RESTORE_PARAM_MAGIC) {
  66. return CO_SDO_AB_DATA_TRANSF;
  67. }
  68. err = canopen_storage_erase(CANOPEN_STORAGE_ROM);
  69. if (err == -ENOENT) {
  70. LOG_DBG("no object dictionary ROM entries to delete");
  71. } else if (err) {
  72. LOG_ERR("failed to delete object dictionary ROM entries"
  73. " (err %d)", err);
  74. CO_errorReport(em, CO_EM_NON_VOLATILE_MEMORY, CO_EMC_HARDWARE,
  75. err);
  76. failed = true;
  77. } else {
  78. LOG_DBG("deleted object dictionary ROM entries");
  79. }
  80. #ifdef CONFIG_CANOPENNODE_STORAGE_HANDLER_ERASES_EEPROM
  81. err = canopen_storage_erase(CANOPEN_STORAGE_EEPROM);
  82. if (err == -ENOENT) {
  83. LOG_DBG("no object dictionary EEPROM entries to delete");
  84. } else if (err) {
  85. LOG_ERR("failed to delete object dictionary EEPROM entries"
  86. " (err %d)", err);
  87. CO_errorReport(em, CO_EM_NON_VOLATILE_MEMORY, CO_EMC_HARDWARE,
  88. err);
  89. failed = true;
  90. } else {
  91. LOG_DBG("deleted object dictionary EEPROM entries");
  92. }
  93. #endif
  94. if (failed) {
  95. return CO_SDO_AB_HW;
  96. }
  97. return CO_SDO_AB_NONE;
  98. }
  99. static int canopen_settings_set(const char *key, size_t len_rd,
  100. settings_read_cb read_cb, void *cb_arg)
  101. {
  102. const char *next;
  103. int nlen;
  104. ssize_t len;
  105. nlen = settings_name_next(key, &next);
  106. if (!strncmp(key, "eeprom", nlen)) {
  107. struct sCO_OD_EEPROM eeprom;
  108. len = read_cb(cb_arg, &eeprom, sizeof(eeprom));
  109. if (len < 0) {
  110. LOG_ERR("failed to restore object dictionary EEPROM"
  111. " entries (err %d)", len);
  112. canopen_storage_eeprom_error = len;
  113. } else {
  114. if ((eeprom.FirstWord == CO_OD_FIRST_LAST_WORD) &&
  115. (eeprom.LastWord == CO_OD_FIRST_LAST_WORD)) {
  116. memcpy(&CO_OD_EEPROM, &eeprom,
  117. sizeof(CO_OD_EEPROM));
  118. LOG_DBG("restored object dictionary EEPROM"
  119. " entries");
  120. } else {
  121. LOG_WRN("object dictionary EEPROM entries"
  122. " signature mismatch, skipping"
  123. " restore");
  124. }
  125. }
  126. return 0;
  127. } else if (!strncmp(key, "rom", nlen)) {
  128. struct sCO_OD_ROM rom;
  129. len = read_cb(cb_arg, &rom, sizeof(rom));
  130. if (len < 0) {
  131. LOG_ERR("failed to restore object dictionary ROM"
  132. " entries (err %d)", len);
  133. canopen_storage_rom_error = len;
  134. } else {
  135. if ((rom.FirstWord == CO_OD_FIRST_LAST_WORD) &&
  136. (rom.LastWord == CO_OD_FIRST_LAST_WORD)) {
  137. memcpy(&CO_OD_ROM, &rom, sizeof(CO_OD_ROM));
  138. LOG_DBG("restored object dictionary ROM"
  139. " entries");
  140. } else {
  141. LOG_WRN("object dictionary ROM entries"
  142. " signature mismatch, skipping"
  143. " restore");
  144. }
  145. }
  146. return 0;
  147. }
  148. return 0;
  149. }
  150. SETTINGS_STATIC_HANDLER_DEFINE(canopen, "canopen", NULL,
  151. canopen_settings_set, NULL, NULL);
  152. void canopen_storage_attach(CO_SDO_t *sdo, CO_EM_t *em)
  153. {
  154. CO_OD_configure(sdo, OD_H1010_STORE_PARAM_FUNC, canopen_odf_1010,
  155. em, 0U, 0U);
  156. CO_OD_configure(sdo, OD_H1011_REST_PARAM_FUNC, canopen_odf_1011,
  157. em, 0U, 0U);
  158. if (canopen_storage_eeprom_error) {
  159. CO_errorReport(em, CO_EM_NON_VOLATILE_MEMORY, CO_EMC_HARDWARE,
  160. canopen_storage_eeprom_error);
  161. }
  162. if (canopen_storage_rom_error) {
  163. CO_errorReport(em, CO_EM_NON_VOLATILE_MEMORY, CO_EMC_HARDWARE,
  164. canopen_storage_rom_error);
  165. }
  166. }
  167. int canopen_storage_save(enum canopen_storage storage)
  168. {
  169. int ret = 0;
  170. if (storage == CANOPEN_STORAGE_ROM) {
  171. ret = settings_save_one("canopen/rom", &CO_OD_ROM,
  172. sizeof(CO_OD_ROM));
  173. } else if (storage == CANOPEN_STORAGE_EEPROM) {
  174. ret = settings_save_one("canopen/eeprom", &CO_OD_EEPROM,
  175. sizeof(CO_OD_EEPROM));
  176. }
  177. return ret;
  178. }
  179. int canopen_storage_erase(enum canopen_storage storage)
  180. {
  181. int ret = 0;
  182. if (storage == CANOPEN_STORAGE_ROM) {
  183. ret = settings_delete("canopen/rom");
  184. } else if (storage == CANOPEN_STORAGE_EEPROM) {
  185. ret = settings_delete("canopen/eeprom");
  186. }
  187. return ret;
  188. }