subghz_history.c 7.4 KB

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