subghz_protocol_ido.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. #include "subghz_protocol_ido.h"
  2. struct SubGhzProtocolIDo {
  3. SubGhzProtocolCommon common;
  4. };
  5. typedef enum {
  6. IDoDecoderStepReset = 0,
  7. IDoDecoderStepFoundPreambula,
  8. IDoDecoderStepSaveDuration,
  9. IDoDecoderStepCheckDuration,
  10. } IDoDecoderStep;
  11. SubGhzProtocolIDo* subghz_protocol_ido_alloc(void) {
  12. SubGhzProtocolIDo* instance = furi_alloc(sizeof(SubGhzProtocolIDo));
  13. instance->common.name = "iDo 117/111"; // PT4301-X";
  14. instance->common.code_min_count_bit_for_found = 48;
  15. instance->common.te_short = 450;
  16. instance->common.te_long = 1450;
  17. instance->common.te_delta = 150;
  18. instance->common.type_protocol = SubGhzProtocolCommonTypeDynamic;
  19. instance->common.to_string = (SubGhzProtocolCommonToStr)subghz_protocol_ido_to_str;
  20. instance->common.to_load_protocol =
  21. (SubGhzProtocolCommonLoadFromRAW)subghz_decoder_ido_to_load_protocol;
  22. return instance;
  23. }
  24. void subghz_protocol_ido_free(SubGhzProtocolIDo* instance) {
  25. furi_assert(instance);
  26. free(instance);
  27. }
  28. /** Send bit
  29. *
  30. * @param instance - SubGhzProtocolIDo instance
  31. * @param bit - bit
  32. */
  33. void subghz_protocol_ido_send_bit(SubGhzProtocolIDo* instance, uint8_t bit) {
  34. if(bit) {
  35. //send bit 1
  36. SUBGHZ_TX_PIN_HIGH();
  37. delay_us(instance->common.te_short);
  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_ido_send_key(
  49. SubGhzProtocolIDo* 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_short * 10);
  57. SUBGHZ_TX_PIN_LOW();
  58. delay_us(instance->common.te_short * 10);
  59. //Send key data
  60. for(uint8_t i = bit; i > 0; i--) {
  61. subghz_protocol_ido_send_bit(instance, bit_read(key, i - 1));
  62. }
  63. }
  64. }
  65. void subghz_protocol_ido_reset(SubGhzProtocolIDo* instance) {
  66. instance->common.parser_step = IDoDecoderStepReset;
  67. }
  68. /** Analysis of received data
  69. *
  70. * @param instance SubGhzProtocolIDo instance
  71. */
  72. void subghz_protocol_ido_check_remote_controller(SubGhzProtocolIDo* 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 & 0xFFFFFF;
  76. instance->common.serial = code_fix & 0xFFFFF;
  77. instance->common.btn = (code_fix >> 20) & 0x0F;
  78. }
  79. void subghz_protocol_ido_parse(SubGhzProtocolIDo* instance, bool level, uint32_t duration) {
  80. switch(instance->common.parser_step) {
  81. case IDoDecoderStepReset:
  82. if((level) && (DURATION_DIFF(duration, instance->common.te_short * 10) <
  83. instance->common.te_delta * 5)) {
  84. instance->common.parser_step = IDoDecoderStepFoundPreambula;
  85. }
  86. break;
  87. case IDoDecoderStepFoundPreambula:
  88. if((!level) && (DURATION_DIFF(duration, instance->common.te_short * 10) <
  89. instance->common.te_delta * 5)) {
  90. //Found Preambula
  91. instance->common.parser_step = IDoDecoderStepSaveDuration;
  92. instance->common.code_found = 0;
  93. instance->common.code_count_bit = 0;
  94. } else {
  95. instance->common.parser_step = IDoDecoderStepReset;
  96. }
  97. break;
  98. case IDoDecoderStepSaveDuration:
  99. if(level) {
  100. if(duration >= (instance->common.te_short * 5 + instance->common.te_delta)) {
  101. instance->common.parser_step = IDoDecoderStepFoundPreambula;
  102. if(instance->common.code_count_bit >=
  103. instance->common.code_min_count_bit_for_found) {
  104. instance->common.code_last_found = instance->common.code_found;
  105. instance->common.code_last_count_bit = instance->common.code_count_bit;
  106. if(instance->common.callback)
  107. instance->common.callback(
  108. (SubGhzProtocolCommon*)instance, instance->common.context);
  109. }
  110. instance->common.code_found = 0;
  111. instance->common.code_count_bit = 0;
  112. break;
  113. } else {
  114. instance->common.te_last = duration;
  115. instance->common.parser_step = IDoDecoderStepCheckDuration;
  116. }
  117. } else {
  118. instance->common.parser_step = IDoDecoderStepReset;
  119. }
  120. break;
  121. case IDoDecoderStepCheckDuration:
  122. if(!level) {
  123. if((DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  124. instance->common.te_delta) &&
  125. (DURATION_DIFF(duration, instance->common.te_long) <
  126. instance->common.te_delta * 3)) {
  127. subghz_protocol_common_add_bit(&instance->common, 0);
  128. instance->common.parser_step = IDoDecoderStepSaveDuration;
  129. } else if(
  130. (DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  131. instance->common.te_delta * 3) &&
  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 = IDoDecoderStepSaveDuration;
  135. } else {
  136. instance->common.parser_step = IDoDecoderStepReset;
  137. }
  138. } else {
  139. instance->common.parser_step = IDoDecoderStepReset;
  140. }
  141. break;
  142. }
  143. }
  144. void subghz_protocol_ido_to_str(SubGhzProtocolIDo* instance, string_t output) {
  145. subghz_protocol_ido_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 & 0xFFFFFF;
  149. uint32_t code_hop = (code_found_reverse >> 24) & 0xFFFFFF;
  150. string_cat_printf(
  151. output,
  152. "%s %dbit\r\n"
  153. "Key:0x%lX%08lX\r\n"
  154. "Fix:%06lX \r\n"
  155. "Hop:%06lX \r\n"
  156. "Sn:%05lX 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_ido_to_load_protocol(SubGhzProtocolIDo* 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_ido_check_remote_controller(instance);
  173. }