PBW.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. #define LOG_LOCAL_LEVEL ESP_LOG_INFO
  2. #include "Locking.h"
  3. #include "cpp_tools.h"
  4. #include "freertos/FreeRTOS.h"
  5. #include "freertos/event_groups.h"
  6. #include "PBW.h"
  7. #include "accessors.h"
  8. #include "esp_log.h"
  9. #include "network_manager.h"
  10. #include "pb.h"
  11. #include "pb_decode.h" // Nanopb header for decoding (deserialization)
  12. #include "pb_encode.h" // Nanopb header for encoding (serialization)
  13. #include "tools.h"
  14. #include <functional>
  15. #include <sstream>
  16. #include <vector>
  17. namespace System {
  18. const char* PBHelper::PROTOTAG = "PB";
  19. void PBHelper::ResetModified() {
  20. ESP_LOGD(PROTOTAG, "Resetting the global commit flag for %s.", name.c_str());
  21. SetGroupBit(Flags::COMMITTED, true);
  22. }
  23. bool PBHelper::SetGroupBit(Flags flags, bool flag) {
  24. bool result = true;
  25. int bit_num = static_cast<int>(flags);
  26. int curFlags = xEventGroupGetBits(_group);
  27. if ((curFlags & static_cast<int>(Flags::LOAD)) && flags == Flags::COMMITTED) {
  28. ESP_LOGD(PROTOTAG, "Loading %s, ignoring changes", name.c_str());
  29. result = false;
  30. }
  31. if (result) {
  32. if ((curFlags & bit_num) == flag) {
  33. ESP_LOGV(PROTOTAG, "Flag %d already %s", bit_num, flag ? "Set" : "Cleared");
  34. result = false;
  35. }
  36. }
  37. if (result) {
  38. ESP_LOGV(PROTOTAG, "%s Flag %d ", flag ? "Setting" : "Clearing", bit_num);
  39. if (!flag) {
  40. xEventGroupClearBits(_group, bit_num);
  41. } else {
  42. xEventGroupSetBits(_group, bit_num);
  43. }
  44. }
  45. return result;
  46. }
  47. void PBHelper::SyncCommit(void* protoWrapper) {
  48. IPBBase* protoWrapperBase = reinterpret_cast<IPBBase*>(protoWrapper);
  49. if (protoWrapperBase) {
  50. protoWrapperBase->CommitChanges();
  51. }
  52. }
  53. std::vector<pb_byte_t> PBHelper::EncodeData(
  54. const pb_msgdesc_t* fields, const void* src_struct) {
  55. size_t datasz;
  56. ESP_LOGV(PROTOTAG, "EncodeData: getting size");
  57. if (!pb_get_encoded_size(&datasz, fields, src_struct)) {
  58. throw std::runtime_error("Failed to get encoded size.");
  59. }
  60. ESP_LOGV(PROTOTAG, "EncodeData: size: %d. Encoding", datasz);
  61. std::vector<pb_byte_t> data(datasz);
  62. pb_ostream_t stream = pb_ostream_from_buffer(data.data(), datasz);
  63. if (!pb_encode(&stream, fields, src_struct)) {
  64. throw std::runtime_error("Failed to encode data.");
  65. }
  66. ESP_LOGV(PROTOTAG, "EncodeData: Done");
  67. return data;
  68. }
  69. void PBHelper::DecodeData(
  70. std::vector<pb_byte_t> data, const pb_msgdesc_t* fields, void* target, bool noinit) {
  71. pb_istream_t stream = pb_istream_from_buffer((uint8_t*)data.data(), data.size());
  72. bool result = false;
  73. // Decode the Protocol Buffers message
  74. if (noinit) {
  75. ESP_LOGD(PROTOTAG, "Decoding WITHOUT initialization");
  76. result = pb_decode_noinit(&stream, fields, data.data());
  77. } else {
  78. ESP_LOGD(PROTOTAG, "Decoding WITH initialization");
  79. result = pb_decode(&stream, fields, target);
  80. }
  81. if (!result) {
  82. throw std::runtime_error(
  83. std::string("Failed to decode settings: %s", PB_GET_ERROR(&stream)));
  84. }
  85. ESP_LOGD(PROTOTAG, "Data decoded");
  86. }
  87. void PBHelper::CommitFile(
  88. const std::string& filename, const pb_msgdesc_t* fields, const void* src_struct) {
  89. size_t datasz = 0;
  90. ESP_LOGD(PROTOTAG, "Committing data to file File %s", filename.c_str());
  91. if (!pb_get_encoded_size(&datasz, fields, src_struct)) {
  92. throw std::runtime_error("Failed to get encoded size.");
  93. }
  94. if (datasz == 0) {
  95. ESP_LOGW(PROTOTAG, "File %s not written. Data size is zero", filename.c_str());
  96. return;
  97. }
  98. ESP_LOGD(PROTOTAG, "Committing to file %s", filename.c_str());
  99. if (!src_struct) {
  100. throw std::runtime_error("Null pointer received.");
  101. }
  102. FILE* file = fopen(filename.c_str(), "wb");
  103. if (file == nullptr) {
  104. throw std::runtime_error(std::string("Error opening file ") + filename.c_str());
  105. }
  106. pb_ostream_t filestream = PB_OSTREAM_SIZING;
  107. filestream.callback = &out_file_binding;
  108. filestream.state = file;
  109. filestream.max_size = SIZE_MAX;
  110. ESP_LOGD(PROTOTAG, "Starting file encode for %s", filename.c_str());
  111. if (!pb_encode(&filestream, fields, (void*)src_struct)) {
  112. fclose(file);
  113. throw std::runtime_error("Encoding file failed");
  114. }
  115. ESP_LOGD(PROTOTAG, "Encoded size: %d", filestream.bytes_written);
  116. if (filestream.bytes_written == 0) {
  117. ESP_LOGW(PROTOTAG, "Empty structure for file %s", filename.c_str());
  118. }
  119. fclose(file);
  120. }
  121. bool PBHelper::IsDataDifferent(
  122. const pb_msgdesc_t* fields, const void* src_struct, const void* other_struct) {
  123. bool changed = false;
  124. try {
  125. ESP_LOGV(PROTOTAG, "Encoding Source data");
  126. auto src_data = EncodeData(fields, src_struct);
  127. ESP_LOGV(PROTOTAG, "Encoding Compared data");
  128. auto other_data = EncodeData(fields, other_struct);
  129. if (src_data.size() != other_data.size()) {
  130. ESP_LOGD(PROTOTAG, "IsDataDifferent: source and target size different: [%d!=%d]",
  131. src_data.size(), other_data.size());
  132. changed = true;
  133. } else if (src_data != other_data) {
  134. ESP_LOGD(PROTOTAG, "IsDataDifferent: source and target not the same");
  135. changed = true;
  136. }
  137. if (changed && esp_log_level_get(PROTOTAG) >= ESP_LOG_DEBUG) {
  138. ESP_LOGD(PROTOTAG, "Source data: ");
  139. dump_data((const uint8_t*)src_data.data(), src_data.size());
  140. ESP_LOGD(PROTOTAG, "Compared data: ");
  141. dump_data((const uint8_t*)other_data.data(), src_data.size());
  142. }
  143. } catch (const std::runtime_error& e) {
  144. throw std::runtime_error(std::string("Comparison failed: ") + e.what());
  145. }
  146. ESP_LOGD(PROTOTAG, "IsDataDifferent: %s", changed ? "TRUE" : "FALSE");
  147. return changed;
  148. }
  149. void PBHelper::CopyStructure(
  150. const void* src_data, const pb_msgdesc_t* fields, void* target_data) {
  151. try {
  152. auto src = EncodeData(fields, src_data);
  153. ESP_LOGD(PROTOTAG, "Encoded structure to copy has %d bytes", src.size());
  154. DecodeData(src, fields, target_data, false);
  155. } catch (const std::runtime_error& e) {
  156. throw std::runtime_error(std::string("Copy failed: ") + e.what());
  157. }
  158. }
  159. void PBHelper::LoadFile(
  160. const std::string& filename, const pb_msgdesc_t* fields, void* target_data, bool noinit) {
  161. struct stat fileInformation;
  162. if (!get_file_info(&fileInformation, filename.c_str()) || fileInformation.st_size == 0) {
  163. throw FileNotFoundException("filename");
  164. }
  165. FILE* file = fopen(filename.c_str(), "rb");
  166. ESP_LOGI(PROTOTAG, "Loading file %s", filename.c_str());
  167. if (file == nullptr) {
  168. int errNum = errno;
  169. ESP_LOGE(
  170. PROTOTAG, "Unable to open file: %s. Error: %s", filename.c_str(), strerror(errNum));
  171. throw std::runtime_error(
  172. "Unable to open file: " + filename + ". Error: " + strerror(errNum));
  173. }
  174. pb_istream_t filestream = PB_ISTREAM_EMPTY;
  175. filestream.callback = &in_file_binding;
  176. filestream.state = file;
  177. filestream.bytes_left = fileInformation.st_size;
  178. ESP_LOGV(PROTOTAG, "Starting decode.");
  179. bool result = false;
  180. if (noinit) {
  181. ESP_LOGV(PROTOTAG, "Decoding WITHOUT initialization");
  182. result = pb_decode_noinit(&filestream, fields, target_data);
  183. } else {
  184. ESP_LOGV(PROTOTAG, "Decoding WITH initialization");
  185. result = pb_decode(&filestream, fields, target_data);
  186. }
  187. fclose(file);
  188. if (!result) {
  189. throw System::DecodeError(PB_GET_ERROR(&filestream));
  190. }
  191. ESP_LOGV(PROTOTAG, "Decode done.");
  192. }
  193. bool PBHelper::FileExists(std::string filename) {
  194. struct stat fileInformation;
  195. ESP_LOGD(PROTOTAG, "Checking if file %s exists", filename.c_str());
  196. return get_file_info(&fileInformation, filename.c_str()) && fileInformation.st_size > 0;
  197. }
  198. bool PBHelper::FileExists() { return FileExists(filename); }
  199. bool PBHelper::IsLoading() { return xEventGroupGetBits(_group) & static_cast<int>(Flags::LOAD); }
  200. void PBHelper::SetLoading(bool active) { SetGroupBit(Flags::LOAD, active); }
  201. bool PBHelper::WaitForCommit(uint8_t retries=2) {
  202. auto remain= retries;
  203. auto bits = xEventGroupGetBits(_group);
  204. bool commit_pending = HasChanges();
  205. ESP_LOGD(PROTOTAG, "Entering WaitForCommit bits: %d, changes? %s", bits,
  206. commit_pending ? "YES" : "NO");
  207. while (commit_pending && remain-->0) {
  208. ESP_LOGD(PROTOTAG, "Waiting for config commit ...");
  209. auto bits = xEventGroupWaitBits(
  210. _group, static_cast<int>(Flags::COMMITTED), pdFALSE, pdTRUE, (MaxDelay * 2) / portTICK_PERIOD_MS);
  211. commit_pending = !(bits & static_cast<int>(Flags::COMMITTED));
  212. ESP_LOGD(
  213. PROTOTAG, "WaitForCommit bits: %d, changes? %s", bits, commit_pending ? "YES" : "NO");
  214. if (commit_pending) {
  215. ESP_LOGW(PROTOTAG, "Timeout waiting for config commit for [%s]", name.c_str());
  216. } else {
  217. ESP_LOGI(PROTOTAG, "Changes to %s committed", name.c_str());
  218. }
  219. }
  220. return !commit_pending;
  221. }
  222. void PBHelper::RaiseChangedAsync() {
  223. if(_no_save){
  224. ESP_LOGD(PROTOTAG,"Ignoring changes for %s, as it is marked not to be saved", name.c_str());
  225. }
  226. ESP_LOGI(PROTOTAG, "Changes made to %s", name.c_str());
  227. if (IsLoading()) {
  228. ESP_LOGD(PROTOTAG, "Ignoring raise modified during load of %s", name.c_str());
  229. } else {
  230. SetGroupBit(Flags::COMMITTED, false);
  231. network_async_commit_protowrapper(
  232. static_cast<void*>(static_cast<IPBBase*>(this)));
  233. }
  234. }
  235. const std::string& PBHelper::GetFileName() { return filename; }
  236. bool PBHelper::HasChanges() { return !(xEventGroupGetBits(_group) & static_cast<int>(Flags::COMMITTED)); }
  237. } // namespace PlatformConfig
  238. bool proto_load_file(
  239. const char* filename, const pb_msgdesc_t* fields, void* target_data, bool noinit = false) {
  240. try {
  241. ESP_LOGI(System::PBHelper::PROTOTAG,"Loading file %s",filename);
  242. System::PBHelper::LoadFile(std::string(filename), fields, target_data, noinit);
  243. } catch (const std::exception& e) {
  244. if(!noinit){
  245. // initialize the structure
  246. pb_istream_t stream = pb_istream_from_buffer(nullptr, 0);
  247. pb_decode(&stream, fields, target_data);
  248. }
  249. ESP_LOGE(System::PBHelper::PROTOTAG, "Error loading file %s: %s", filename, e.what());
  250. return false;
  251. }
  252. return true;
  253. }