subghz_protocol_common.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. #include "subghz_protocol_common.h"
  2. #include <stdio.h>
  3. #include <lib/toolbox/hex.h>
  4. SubGhzProtocolCommonEncoder* subghz_protocol_encoder_common_alloc() {
  5. SubGhzProtocolCommonEncoder* instance = furi_alloc(sizeof(SubGhzProtocolCommonEncoder));
  6. instance->upload = furi_alloc(SUBGHZ_ENCODER_UPLOAD_MAX_SIZE * sizeof(LevelDuration));
  7. instance->start = true;
  8. instance->repeat = 10; //default number of repeat
  9. return instance;
  10. }
  11. void subghz_protocol_encoder_common_free(SubGhzProtocolCommonEncoder* instance) {
  12. furi_assert(instance);
  13. if(instance->callback_end) {
  14. instance->callback_end((SubGhzProtocolCommon*)instance->context_end);
  15. }
  16. free(instance->upload);
  17. free(instance);
  18. }
  19. size_t subghz_encoder_common_get_repeat_left(SubGhzProtocolCommonEncoder* instance) {
  20. furi_assert(instance);
  21. return instance->repeat;
  22. }
  23. void subghz_protocol_encoder_common_set_callback(
  24. SubGhzProtocolCommonEncoder* instance,
  25. SubGhzProtocolCommonEncoderCallback callback,
  26. void* context) {
  27. furi_assert(instance);
  28. furi_assert(callback);
  29. instance->callback = callback;
  30. instance->context = context;
  31. }
  32. void subghz_protocol_encoder_common_set_callback_end(
  33. SubGhzProtocolCommonEncoder* instance,
  34. SubGhzProtocolCommonEncoderCallbackEnd callback_end,
  35. void* context_end) {
  36. furi_assert(instance);
  37. furi_assert(callback_end);
  38. instance->callback_end = callback_end;
  39. instance->context_end = context_end;
  40. }
  41. LevelDuration subghz_protocol_encoder_common_yield(void* context) {
  42. SubGhzProtocolCommonEncoder* instance = context;
  43. if(instance->callback) {
  44. return instance->callback((SubGhzProtocolCommon*)instance->context);
  45. }
  46. if(instance->repeat == 0) {
  47. return level_duration_reset();
  48. }
  49. LevelDuration ret = instance->upload[instance->front];
  50. if(++instance->front == instance->size_upload) {
  51. instance->repeat--;
  52. instance->front = 0;
  53. }
  54. return ret;
  55. }
  56. void subghz_protocol_common_add_bit(SubGhzProtocolCommon* common, uint8_t bit) {
  57. common->code_found = common->code_found << 1 | bit;
  58. common->code_count_bit++;
  59. }
  60. bool subghz_protocol_common_check_interval(
  61. SubGhzProtocolCommon* common,
  62. uint32_t duration,
  63. uint16_t duration_check) {
  64. if((duration_check >= (duration - common->te_delta)) &&
  65. (duration_check <= (duration + common->te_delta))) {
  66. return true;
  67. } else {
  68. return false;
  69. }
  70. }
  71. uint64_t subghz_protocol_common_reverse_key(uint64_t key, uint8_t count_bit) {
  72. uint64_t key_reverse = 0;
  73. for(uint8_t i = 0; i < count_bit; i++) {
  74. key_reverse = key_reverse << 1 | bit_read(key, i);
  75. }
  76. return key_reverse;
  77. }
  78. void subghz_protocol_common_set_callback(
  79. SubGhzProtocolCommon* common,
  80. SubGhzProtocolCommonCallback callback,
  81. void* context) {
  82. common->callback = callback;
  83. common->context = context;
  84. }
  85. void subghz_protocol_common_to_str(SubGhzProtocolCommon* instance, string_t output) {
  86. if(instance->to_string) {
  87. instance->to_string(instance, output);
  88. } else {
  89. uint32_t code_found_hi = instance->code_found >> 32;
  90. uint32_t code_found_lo = instance->code_found & 0x00000000ffffffff;
  91. uint64_t code_found_reverse =
  92. subghz_protocol_common_reverse_key(instance->code_found, instance->code_count_bit);
  93. uint32_t code_found_reverse_hi = code_found_reverse >> 32;
  94. uint32_t code_found_reverse_lo = code_found_reverse & 0x00000000ffffffff;
  95. if(code_found_hi > 0) {
  96. string_cat_printf(
  97. output,
  98. "Protocol %s, %d Bit\r\n"
  99. " KEY:0x%lX%08lX\r\n"
  100. " YEK:0x%lX%08lX\r\n"
  101. " SN:0x%05lX BTN:%02X\r\n",
  102. instance->name,
  103. instance->code_count_bit,
  104. code_found_hi,
  105. code_found_lo,
  106. code_found_reverse_hi,
  107. code_found_reverse_lo,
  108. instance->serial,
  109. instance->btn);
  110. } else {
  111. string_cat_printf(
  112. output,
  113. "Protocol %s, %d Bit\r\n"
  114. " KEY:0x%lX%lX\r\n"
  115. " YEK:0x%lX%lX\r\n"
  116. " SN:0x%05lX BTN:%02X\r\n",
  117. instance->name,
  118. instance->code_count_bit,
  119. code_found_hi,
  120. code_found_lo,
  121. code_found_reverse_hi,
  122. code_found_reverse_lo,
  123. instance->serial,
  124. instance->btn);
  125. }
  126. }
  127. }
  128. bool subghz_protocol_common_read_hex(string_t str, uint8_t* buff, uint16_t len) {
  129. string_strim(str);
  130. uint8_t nibble_high = 0;
  131. uint8_t nibble_low = 0;
  132. bool parsed = true;
  133. for(uint16_t i = 0; i < len; i++) {
  134. if(hex_char_to_hex_nibble(string_get_char(str, 0), &nibble_high) &&
  135. hex_char_to_hex_nibble(string_get_char(str, 1), &nibble_low)) {
  136. buff[i] = (nibble_high << 4) | nibble_low;
  137. if(string_size(str) > 2) {
  138. string_right(str, 2);
  139. } else if(i < len - 1) {
  140. parsed = false;
  141. break;
  142. };
  143. } else {
  144. parsed = false;
  145. break;
  146. }
  147. }
  148. return parsed;
  149. }
  150. bool subghz_protocol_common_to_save_file(SubGhzProtocolCommon* instance, FlipperFile* flipper_file) {
  151. furi_assert(instance);
  152. furi_assert(flipper_file);
  153. bool res = false;
  154. do {
  155. if(!flipper_file_write_string_cstr(flipper_file, "Protocol", instance->name)) {
  156. FURI_LOG_E(SUBGHZ_PARSER_TAG, "Unable to add Protocol");
  157. break;
  158. }
  159. uint32_t temp = instance->code_last_count_bit;
  160. if(!flipper_file_write_uint32(
  161. flipper_file, "Bit", &temp, 1)) {
  162. FURI_LOG_E(SUBGHZ_PARSER_TAG, "Unable to add Bit");
  163. break;
  164. }
  165. uint8_t key_data[sizeof(uint64_t)] = {0};
  166. for(size_t i = 0; i < sizeof(uint64_t); i++) {
  167. key_data[sizeof(uint64_t) - i - 1] = (instance->code_last_found >> i * 8) & 0xFF;
  168. }
  169. if(!flipper_file_write_hex(flipper_file, "Key", key_data, sizeof(uint64_t))) {
  170. FURI_LOG_E(SUBGHZ_PARSER_TAG, "Unable to add Key");
  171. break;
  172. }
  173. res = true;
  174. } while(false);
  175. return res;
  176. }
  177. bool subghz_protocol_common_to_load_protocol_from_file(
  178. SubGhzProtocolCommon* instance,
  179. FlipperFile* flipper_file) {
  180. furi_assert(instance);
  181. furi_assert(flipper_file);
  182. bool loaded = false;
  183. string_t temp_str;
  184. string_init(temp_str);
  185. uint32_t temp_data = 0;
  186. do {
  187. if(!flipper_file_read_uint32(flipper_file, "Bit", (uint32_t*)&temp_data, 1)) {
  188. FURI_LOG_E(SUBGHZ_PARSER_TAG, "Missing Bit");
  189. break;
  190. }
  191. instance->code_last_count_bit = (uint8_t)temp_data;
  192. uint8_t key_data[sizeof(uint64_t)] = {0};
  193. if(!flipper_file_read_hex(flipper_file, "Key", key_data, sizeof(uint64_t))) {
  194. FURI_LOG_E(SUBGHZ_PARSER_TAG, "Missing Key");
  195. break;
  196. }
  197. for(uint8_t i = 0; i < sizeof(uint64_t); i++) {
  198. instance->code_last_found = instance->code_last_found << 8 | key_data[i];
  199. }
  200. loaded = true;
  201. } while(0);
  202. string_clear(temp_str);
  203. return loaded;
  204. }