subghz_history.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. #include "subghz_history.h"
  2. #include <lib/subghz/receiver.h>
  3. #include <lib/subghz/protocols/came.h>
  4. #include <furi.h>
  5. #include <m-string.h>
  6. #define SUBGHZ_HISTORY_MAX 50
  7. #define TAG "SubGhzHistory"
  8. typedef struct {
  9. string_t item_str;
  10. FlipperFormat* flipper_string;
  11. uint8_t type;
  12. SubGhzPresetDefinition* preset;
  13. } SubGhzHistoryItem;
  14. ARRAY_DEF(SubGhzHistoryItemArray, SubGhzHistoryItem, M_POD_OPLIST)
  15. #define M_OPL_SubGhzHistoryItemArray_t() ARRAY_OPLIST(SubGhzHistoryItemArray, M_POD_OPLIST)
  16. typedef struct {
  17. SubGhzHistoryItemArray_t data;
  18. } SubGhzHistoryStruct;
  19. struct SubGhzHistory {
  20. uint32_t last_update_timestamp;
  21. uint16_t last_index_write;
  22. uint8_t code_last_hash_data;
  23. string_t tmp_string;
  24. SubGhzHistoryStruct* history;
  25. };
  26. SubGhzHistory* subghz_history_alloc(void) {
  27. SubGhzHistory* instance = malloc(sizeof(SubGhzHistory));
  28. string_init(instance->tmp_string);
  29. instance->history = malloc(sizeof(SubGhzHistoryStruct));
  30. SubGhzHistoryItemArray_init(instance->history->data);
  31. return instance;
  32. }
  33. void subghz_history_free(SubGhzHistory* instance) {
  34. furi_assert(instance);
  35. string_clear(instance->tmp_string);
  36. for
  37. M_EACH(item, instance->history->data, SubGhzHistoryItemArray_t) {
  38. string_clear(item->item_str);
  39. string_clear(item->preset->name);
  40. free(item->preset);
  41. flipper_format_free(item->flipper_string);
  42. item->type = 0;
  43. }
  44. SubGhzHistoryItemArray_clear(instance->history->data);
  45. free(instance->history);
  46. free(instance);
  47. }
  48. uint32_t subghz_history_get_frequency(SubGhzHistory* instance, uint16_t idx) {
  49. furi_assert(instance);
  50. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  51. return item->preset->frequency;
  52. }
  53. SubGhzPresetDefinition* subghz_history_get_preset_def(SubGhzHistory* instance, uint16_t idx) {
  54. furi_assert(instance);
  55. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  56. return item->preset;
  57. }
  58. const char* subghz_history_get_preset(SubGhzHistory* instance, uint16_t idx) {
  59. furi_assert(instance);
  60. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  61. return string_get_cstr(item->preset->name);
  62. }
  63. void subghz_history_reset(SubGhzHistory* instance) {
  64. furi_assert(instance);
  65. string_reset(instance->tmp_string);
  66. for
  67. M_EACH(item, instance->history->data, SubGhzHistoryItemArray_t) {
  68. string_clear(item->item_str);
  69. string_clear(item->preset->name);
  70. free(item->preset);
  71. flipper_format_free(item->flipper_string);
  72. item->type = 0;
  73. }
  74. SubGhzHistoryItemArray_reset(instance->history->data);
  75. instance->last_index_write = 0;
  76. instance->code_last_hash_data = 0;
  77. }
  78. uint16_t subghz_history_get_item(SubGhzHistory* instance) {
  79. furi_assert(instance);
  80. return instance->last_index_write;
  81. }
  82. uint8_t subghz_history_get_type_protocol(SubGhzHistory* instance, uint16_t idx) {
  83. furi_assert(instance);
  84. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  85. return item->type;
  86. }
  87. const char* subghz_history_get_protocol_name(SubGhzHistory* instance, uint16_t idx) {
  88. furi_assert(instance);
  89. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  90. flipper_format_rewind(item->flipper_string);
  91. if(!flipper_format_read_string(item->flipper_string, "Protocol", instance->tmp_string)) {
  92. FURI_LOG_E(TAG, "Missing Protocol");
  93. string_reset(instance->tmp_string);
  94. }
  95. return string_get_cstr(instance->tmp_string);
  96. }
  97. FlipperFormat* subghz_history_get_raw_data(SubGhzHistory* instance, uint16_t idx) {
  98. furi_assert(instance);
  99. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  100. if(item->flipper_string) {
  101. return item->flipper_string;
  102. } else {
  103. return NULL;
  104. }
  105. }
  106. bool subghz_history_get_text_space_left(SubGhzHistory* instance, string_t output) {
  107. furi_assert(instance);
  108. if(instance->last_index_write == SUBGHZ_HISTORY_MAX) {
  109. if(output != NULL) string_printf(output, "Memory is FULL");
  110. return true;
  111. }
  112. if(output != NULL)
  113. string_printf(output, "%02u/%02u", instance->last_index_write, SUBGHZ_HISTORY_MAX);
  114. return false;
  115. }
  116. void subghz_history_get_text_item_menu(SubGhzHistory* instance, string_t output, uint16_t idx) {
  117. SubGhzHistoryItem* item = SubGhzHistoryItemArray_get(instance->history->data, idx);
  118. string_set(output, item->item_str);
  119. }
  120. bool subghz_history_add_to_history(
  121. SubGhzHistory* instance,
  122. void* context,
  123. SubGhzPresetDefinition* preset) {
  124. furi_assert(instance);
  125. furi_assert(context);
  126. if(instance->last_index_write >= SUBGHZ_HISTORY_MAX) return false;
  127. SubGhzProtocolDecoderBase* decoder_base = context;
  128. if((instance->code_last_hash_data ==
  129. subghz_protocol_decoder_base_get_hash_data(decoder_base)) &&
  130. ((furi_get_tick() - instance->last_update_timestamp) < 500)) {
  131. instance->last_update_timestamp = furi_get_tick();
  132. return false;
  133. }
  134. instance->code_last_hash_data = subghz_protocol_decoder_base_get_hash_data(decoder_base);
  135. instance->last_update_timestamp = furi_get_tick();
  136. string_t text;
  137. string_init(text);
  138. SubGhzHistoryItem* item = SubGhzHistoryItemArray_push_raw(instance->history->data);
  139. item->preset = malloc(sizeof(SubGhzPresetDefinition));
  140. item->type = decoder_base->protocol->type;
  141. item->preset->frequency = preset->frequency;
  142. string_init(item->preset->name);
  143. string_set(item->preset->name, preset->name);
  144. item->preset->data = preset->data;
  145. item->preset->data_size = preset->data_size;
  146. string_init(item->item_str);
  147. item->flipper_string = flipper_format_string_alloc();
  148. subghz_protocol_decoder_base_serialize(decoder_base, item->flipper_string, preset);
  149. do {
  150. if(!flipper_format_rewind(item->flipper_string)) {
  151. FURI_LOG_E(TAG, "Rewind error");
  152. break;
  153. }
  154. if(!flipper_format_read_string(item->flipper_string, "Protocol", instance->tmp_string)) {
  155. FURI_LOG_E(TAG, "Missing Protocol");
  156. break;
  157. }
  158. if(!strcmp(string_get_cstr(instance->tmp_string), "KeeLoq")) {
  159. string_set_str(instance->tmp_string, "KL ");
  160. if(!flipper_format_read_string(item->flipper_string, "Manufacture", text)) {
  161. FURI_LOG_E(TAG, "Missing Protocol");
  162. break;
  163. }
  164. string_cat(instance->tmp_string, text);
  165. } else if(!strcmp(string_get_cstr(instance->tmp_string), "Star Line")) {
  166. string_set_str(instance->tmp_string, "SL ");
  167. if(!flipper_format_read_string(item->flipper_string, "Manufacture", text)) {
  168. FURI_LOG_E(TAG, "Missing Protocol");
  169. break;
  170. }
  171. string_cat(instance->tmp_string, text);
  172. }
  173. if(!flipper_format_rewind(item->flipper_string)) {
  174. FURI_LOG_E(TAG, "Rewind error");
  175. break;
  176. }
  177. uint8_t key_data[sizeof(uint64_t)] = {0};
  178. if(!flipper_format_read_hex(item->flipper_string, "Key", key_data, sizeof(uint64_t))) {
  179. FURI_LOG_E(TAG, "Missing Key");
  180. break;
  181. }
  182. uint64_t data = 0;
  183. for(uint8_t i = 0; i < sizeof(uint64_t); i++) {
  184. data = (data << 8) | key_data[i];
  185. }
  186. if(!(uint32_t)(data >> 32)) {
  187. string_printf(
  188. item->item_str,
  189. "%s %lX",
  190. string_get_cstr(instance->tmp_string),
  191. (uint32_t)(data & 0xFFFFFFFF));
  192. } else {
  193. string_printf(
  194. item->item_str,
  195. "%s %lX%08lX",
  196. string_get_cstr(instance->tmp_string),
  197. (uint32_t)(data >> 32),
  198. (uint32_t)(data & 0xFFFFFFFF));
  199. }
  200. } while(false);
  201. string_clear(text);
  202. instance->last_index_write++;
  203. return true;
  204. }