subghz_protocol_faac_slh.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. #include "subghz_protocol_faac_slh.h"
  2. struct SubGhzProtocolFaacSLH {
  3. SubGhzProtocolCommon common;
  4. };
  5. typedef enum {
  6. FaacSLHDecoderStepReset = 0,
  7. FaacSLHDecoderStepFoundPreambula,
  8. FaacSLHDecoderStepSaveDuration,
  9. FaacSLHDecoderStepCheckDuration,
  10. } FaacSLHDecoderStep;
  11. SubGhzProtocolFaacSLH* subghz_protocol_faac_slh_alloc(void) {
  12. SubGhzProtocolFaacSLH* instance = furi_alloc(sizeof(SubGhzProtocolFaacSLH));
  13. instance->common.name = "Faac SLH";
  14. instance->common.code_min_count_bit_for_found = 64;
  15. instance->common.te_short = 255;
  16. instance->common.te_long = 595;
  17. instance->common.te_delta = 100;
  18. instance->common.type_protocol = SubGhzProtocolCommonTypeDynamic;
  19. instance->common.to_string = (SubGhzProtocolCommonToStr)subghz_protocol_faac_slh_to_str;
  20. instance->common.to_load_protocol =
  21. (SubGhzProtocolCommonLoadFromRAW)subghz_decoder_faac_slh_to_load_protocol;
  22. return instance;
  23. }
  24. void subghz_protocol_faac_slh_free(SubGhzProtocolFaacSLH* instance) {
  25. furi_assert(instance);
  26. free(instance);
  27. }
  28. /** Send bit
  29. *
  30. * @param instance - SubGhzProtocolFaacSLH instance
  31. * @param bit - bit
  32. */
  33. void subghz_protocol_faac_slh_send_bit(SubGhzProtocolFaacSLH* instance, uint8_t bit) {
  34. if(bit) {
  35. //send bit 1
  36. SUBGHZ_TX_PIN_HIGH();
  37. delay_us(instance->common.te_long);
  38. SUBGHZ_TX_PIN_LOW();
  39. delay_us(instance->common.te_short);
  40. } else {
  41. //send bit 0
  42. SUBGHZ_TX_PIN_HIGH();
  43. delay_us(instance->common.te_short);
  44. SUBGHZ_TX_PIN_LOW();
  45. delay_us(instance->common.te_long);
  46. }
  47. }
  48. void subghz_protocol_faac_slh_send_key(
  49. SubGhzProtocolFaacSLH* instance,
  50. uint64_t key,
  51. uint8_t bit,
  52. uint8_t repeat) {
  53. while(repeat--) {
  54. SUBGHZ_TX_PIN_HIGH();
  55. //Send header
  56. delay_us(instance->common.te_long * 2);
  57. SUBGHZ_TX_PIN_LOW();
  58. delay_us(instance->common.te_long * 2);
  59. //Send key data
  60. for(uint8_t i = bit; i > 0; i--) {
  61. subghz_protocol_faac_slh_send_bit(instance, bit_read(key, i - 1));
  62. }
  63. }
  64. }
  65. void subghz_protocol_faac_slh_reset(SubGhzProtocolFaacSLH* instance) {
  66. instance->common.parser_step = FaacSLHDecoderStepReset;
  67. }
  68. /** Analysis of received data
  69. *
  70. * @param instance SubGhzProtocolFaacSLH instance
  71. */
  72. void subghz_protocol_faac_slh_check_remote_controller(SubGhzProtocolFaacSLH* instance) {
  73. uint64_t code_found_reverse = subghz_protocol_common_reverse_key(
  74. instance->common.code_last_found, instance->common.code_last_count_bit);
  75. uint32_t code_fix = code_found_reverse & 0xFFFFFFFF;
  76. //uint32_t code_hop = (code_found_reverse >> 24) & 0xFFFFF;
  77. instance->common.serial = code_fix & 0xFFFFFFF;
  78. instance->common.btn = (code_fix >> 28) & 0x0F;
  79. }
  80. void subghz_protocol_faac_slh_parse(SubGhzProtocolFaacSLH* instance, bool level, uint32_t duration) {
  81. switch(instance->common.parser_step) {
  82. case FaacSLHDecoderStepReset:
  83. if((level) && (DURATION_DIFF(duration, instance->common.te_long * 2) <
  84. instance->common.te_delta * 3)) {
  85. instance->common.parser_step = FaacSLHDecoderStepFoundPreambula;
  86. }
  87. break;
  88. case FaacSLHDecoderStepFoundPreambula:
  89. if((!level) && (DURATION_DIFF(duration, instance->common.te_long * 2) <
  90. instance->common.te_delta * 3)) {
  91. //Found Preambula
  92. instance->common.parser_step = FaacSLHDecoderStepSaveDuration;
  93. instance->common.code_found = 0;
  94. instance->common.code_count_bit = 0;
  95. } else {
  96. instance->common.parser_step = FaacSLHDecoderStepReset;
  97. }
  98. break;
  99. case FaacSLHDecoderStepSaveDuration:
  100. if(level) {
  101. if(duration >= (instance->common.te_short * 3 + instance->common.te_delta)) {
  102. instance->common.parser_step = FaacSLHDecoderStepFoundPreambula;
  103. if(instance->common.code_count_bit >=
  104. instance->common.code_min_count_bit_for_found) {
  105. instance->common.code_last_found = instance->common.code_found;
  106. instance->common.code_last_count_bit = instance->common.code_count_bit;
  107. if(instance->common.callback)
  108. instance->common.callback(
  109. (SubGhzProtocolCommon*)instance, instance->common.context);
  110. }
  111. instance->common.code_found = 0;
  112. instance->common.code_count_bit = 0;
  113. break;
  114. } else {
  115. instance->common.te_last = duration;
  116. instance->common.parser_step = FaacSLHDecoderStepCheckDuration;
  117. }
  118. } else {
  119. instance->common.parser_step = FaacSLHDecoderStepReset;
  120. }
  121. break;
  122. case FaacSLHDecoderStepCheckDuration:
  123. if(!level) {
  124. if((DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  125. instance->common.te_delta) &&
  126. (DURATION_DIFF(duration, instance->common.te_long) < instance->common.te_delta)) {
  127. subghz_protocol_common_add_bit(&instance->common, 0);
  128. instance->common.parser_step = FaacSLHDecoderStepSaveDuration;
  129. } else if(
  130. (DURATION_DIFF(instance->common.te_last, instance->common.te_long) <
  131. instance->common.te_delta) &&
  132. (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) {
  133. subghz_protocol_common_add_bit(&instance->common, 1);
  134. instance->common.parser_step = FaacSLHDecoderStepSaveDuration;
  135. } else {
  136. instance->common.parser_step = FaacSLHDecoderStepReset;
  137. }
  138. } else {
  139. instance->common.parser_step = FaacSLHDecoderStepReset;
  140. }
  141. break;
  142. }
  143. }
  144. void subghz_protocol_faac_slh_to_str(SubGhzProtocolFaacSLH* instance, string_t output) {
  145. subghz_protocol_faac_slh_check_remote_controller(instance);
  146. uint64_t code_found_reverse = subghz_protocol_common_reverse_key(
  147. instance->common.code_last_found, instance->common.code_last_count_bit);
  148. uint32_t code_fix = code_found_reverse & 0xFFFFFFFF;
  149. uint32_t code_hop = (code_found_reverse >> 32) & 0xFFFFFFFF;
  150. string_cat_printf(
  151. output,
  152. "%s %dbit\r\n"
  153. "Key:%lX%08lX\r\n"
  154. "Fix:%08lX \r\n"
  155. "Hop:%08lX \r\n"
  156. "Sn:%07lX Btn:%lX\r\n",
  157. instance->common.name,
  158. instance->common.code_last_count_bit,
  159. (uint32_t)(instance->common.code_last_found >> 32),
  160. (uint32_t)instance->common.code_last_found,
  161. code_fix,
  162. code_hop,
  163. instance->common.serial,
  164. instance->common.btn);
  165. }
  166. void subghz_decoder_faac_slh_to_load_protocol(SubGhzProtocolFaacSLH* instance, void* context) {
  167. furi_assert(context);
  168. furi_assert(instance);
  169. SubGhzProtocolCommonLoad* data = context;
  170. instance->common.code_last_found = data->code_found;
  171. instance->common.code_last_count_bit = data->code_count_bit;
  172. subghz_protocol_faac_slh_check_remote_controller(instance);
  173. }