weather_station_history.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. #include "weather_station_history.h"
  2. #include <flipper_format/flipper_format_i.h>
  3. #include <lib/toolbox/stream/stream.h>
  4. #include <lib/subghz/receiver.h>
  5. #include <furi.h>
  6. #define WS_HISTORY_MAX 50
  7. #define TAG "WSHistory"
  8. typedef struct {
  9. FuriString* item_str;
  10. FlipperFormat* flipper_string;
  11. uint8_t type;
  12. uint32_t id;
  13. SubGhzRadioPreset* preset;
  14. } WSHistoryItem;
  15. ARRAY_DEF(WSHistoryItemArray, WSHistoryItem, M_POD_OPLIST)
  16. #define M_OPL_WSHistoryItemArray_t() ARRAY_OPLIST(WSHistoryItemArray, M_POD_OPLIST)
  17. typedef struct {
  18. WSHistoryItemArray_t data;
  19. } WSHistoryStruct;
  20. struct WSHistory {
  21. uint32_t last_update_timestamp;
  22. uint16_t last_index_write;
  23. uint8_t code_last_hash_data;
  24. FuriString* tmp_string;
  25. WSHistoryStruct* history;
  26. };
  27. WSHistory* ws_history_alloc(void) {
  28. WSHistory* instance = malloc(sizeof(WSHistory));
  29. instance->tmp_string = furi_string_alloc();
  30. instance->history = malloc(sizeof(WSHistoryStruct));
  31. WSHistoryItemArray_init(instance->history->data);
  32. return instance;
  33. }
  34. void ws_history_free(WSHistory* instance) {
  35. furi_assert(instance);
  36. furi_string_free(instance->tmp_string);
  37. for
  38. M_EACH(item, instance->history->data, WSHistoryItemArray_t) {
  39. furi_string_free(item->item_str);
  40. furi_string_free(item->preset->name);
  41. free(item->preset);
  42. flipper_format_free(item->flipper_string);
  43. item->type = 0;
  44. }
  45. WSHistoryItemArray_clear(instance->history->data);
  46. free(instance->history);
  47. free(instance);
  48. }
  49. uint32_t ws_history_get_frequency(WSHistory* instance, uint16_t idx) {
  50. furi_assert(instance);
  51. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  52. return item->preset->frequency;
  53. }
  54. SubGhzRadioPreset* ws_history_get_radio_preset(WSHistory* instance, uint16_t idx) {
  55. furi_assert(instance);
  56. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  57. return item->preset;
  58. }
  59. const char* ws_history_get_preset(WSHistory* instance, uint16_t idx) {
  60. furi_assert(instance);
  61. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  62. return furi_string_get_cstr(item->preset->name);
  63. }
  64. void ws_history_reset(WSHistory* instance) {
  65. furi_assert(instance);
  66. furi_string_reset(instance->tmp_string);
  67. for
  68. M_EACH(item, instance->history->data, WSHistoryItemArray_t) {
  69. furi_string_free(item->item_str);
  70. furi_string_free(item->preset->name);
  71. free(item->preset);
  72. flipper_format_free(item->flipper_string);
  73. item->type = 0;
  74. }
  75. WSHistoryItemArray_reset(instance->history->data);
  76. instance->last_index_write = 0;
  77. instance->code_last_hash_data = 0;
  78. }
  79. uint16_t ws_history_get_item(WSHistory* instance) {
  80. furi_assert(instance);
  81. return instance->last_index_write;
  82. }
  83. uint8_t ws_history_get_type_protocol(WSHistory* instance, uint16_t idx) {
  84. furi_assert(instance);
  85. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  86. return item->type;
  87. }
  88. const char* ws_history_get_protocol_name(WSHistory* instance, uint16_t idx) {
  89. furi_assert(instance);
  90. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  91. flipper_format_rewind(item->flipper_string);
  92. if(!flipper_format_read_string(item->flipper_string, "Protocol", instance->tmp_string)) {
  93. FURI_LOG_E(TAG, "Missing Protocol");
  94. furi_string_reset(instance->tmp_string);
  95. }
  96. return furi_string_get_cstr(instance->tmp_string);
  97. }
  98. FlipperFormat* ws_history_get_raw_data(WSHistory* instance, uint16_t idx) {
  99. furi_assert(instance);
  100. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  101. if(item->flipper_string) {
  102. return item->flipper_string;
  103. } else {
  104. return NULL;
  105. }
  106. }
  107. bool ws_history_get_text_space_left(WSHistory* instance, FuriString* output) {
  108. furi_assert(instance);
  109. if(instance->last_index_write == WS_HISTORY_MAX) {
  110. if(output != NULL) furi_string_printf(output, "Memory is FULL");
  111. return true;
  112. }
  113. if(output != NULL)
  114. furi_string_printf(output, "%02u/%02u", instance->last_index_write, WS_HISTORY_MAX);
  115. return false;
  116. }
  117. void ws_history_get_text_item_menu(WSHistory* instance, FuriString* output, uint16_t idx) {
  118. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, idx);
  119. furi_string_set(output, item->item_str);
  120. }
  121. WSHistoryStateAddKey
  122. ws_history_add_to_history(WSHistory* instance, void* context, SubGhzRadioPreset* preset) {
  123. furi_assert(instance);
  124. furi_assert(context);
  125. if(instance->last_index_write >= WS_HISTORY_MAX) return WSHistoryStateAddKeyOverflow;
  126. SubGhzProtocolDecoderBase* decoder_base = context;
  127. if((instance->code_last_hash_data ==
  128. subghz_protocol_decoder_base_get_hash_data(decoder_base)) &&
  129. ((furi_get_tick() - instance->last_update_timestamp) < 500)) {
  130. instance->last_update_timestamp = furi_get_tick();
  131. return WSHistoryStateAddKeyTimeOut;
  132. }
  133. instance->code_last_hash_data = subghz_protocol_decoder_base_get_hash_data(decoder_base);
  134. instance->last_update_timestamp = furi_get_tick();
  135. FlipperFormat* fff = flipper_format_string_alloc();
  136. uint32_t id = 0;
  137. subghz_protocol_decoder_base_serialize(decoder_base, fff, preset);
  138. do {
  139. if(!flipper_format_rewind(fff)) {
  140. FURI_LOG_E(TAG, "Rewind error");
  141. break;
  142. }
  143. if(!flipper_format_read_uint32(fff, "Id", (uint32_t*)&id, 1)) {
  144. FURI_LOG_E(TAG, "Missing Id");
  145. break;
  146. }
  147. } while(false);
  148. flipper_format_free(fff);
  149. //Update record if found
  150. bool sensor_found = false;
  151. for(size_t i = 0; i < WSHistoryItemArray_size(instance->history->data); i++) {
  152. WSHistoryItem* item = WSHistoryItemArray_get(instance->history->data, i);
  153. if(item->id == id) {
  154. sensor_found = true;
  155. Stream* flipper_string_stream = flipper_format_get_raw_stream(item->flipper_string);
  156. stream_clean(flipper_string_stream);
  157. subghz_protocol_decoder_base_serialize(decoder_base, item->flipper_string, preset);
  158. return WSHistoryStateAddKeyUpdateData;
  159. }
  160. }
  161. // or add new record
  162. if(!sensor_found) {
  163. WSHistoryItem* item = WSHistoryItemArray_push_raw(instance->history->data);
  164. item->preset = malloc(sizeof(SubGhzRadioPreset));
  165. item->type = decoder_base->protocol->type;
  166. item->preset->frequency = preset->frequency;
  167. item->preset->name = furi_string_alloc();
  168. furi_string_set(item->preset->name, preset->name);
  169. item->preset->data = preset->data;
  170. item->preset->data_size = preset->data_size;
  171. item->id = id;
  172. item->item_str = furi_string_alloc();
  173. item->flipper_string = flipper_format_string_alloc();
  174. subghz_protocol_decoder_base_serialize(decoder_base, item->flipper_string, preset);
  175. do {
  176. if(!flipper_format_rewind(item->flipper_string)) {
  177. FURI_LOG_E(TAG, "Rewind error");
  178. break;
  179. }
  180. if(!flipper_format_read_string(
  181. item->flipper_string, "Protocol", instance->tmp_string)) {
  182. FURI_LOG_E(TAG, "Missing Protocol");
  183. break;
  184. }
  185. if(!flipper_format_rewind(item->flipper_string)) {
  186. FURI_LOG_E(TAG, "Rewind error");
  187. break;
  188. }
  189. uint8_t key_data[sizeof(uint64_t)] = {0};
  190. if(!flipper_format_read_hex(item->flipper_string, "Data", key_data, sizeof(uint64_t))) {
  191. FURI_LOG_E(TAG, "Missing Data");
  192. break;
  193. }
  194. uint64_t data = 0;
  195. for(uint8_t i = 0; i < sizeof(uint64_t); i++) {
  196. data = (data << 8) | key_data[i];
  197. }
  198. if(!(uint32_t)(data >> 32)) {
  199. furi_string_printf(
  200. item->item_str,
  201. "%s %lX",
  202. furi_string_get_cstr(instance->tmp_string),
  203. (uint32_t)(data & 0xFFFFFFFF));
  204. } else {
  205. furi_string_printf(
  206. item->item_str,
  207. "%s %lX%08lX",
  208. furi_string_get_cstr(instance->tmp_string),
  209. (uint32_t)(data >> 32),
  210. (uint32_t)(data & 0xFFFFFFFF));
  211. }
  212. } while(false);
  213. instance->last_index_write++;
  214. return WSHistoryStateAddKeyNewDada;
  215. }
  216. return WSHistoryStateAddKeyUnknown;
  217. }