subghz_protocol_ido.c 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  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. } else {
  86. instance->common.parser_step = IDoDecoderStepReset;
  87. }
  88. break;
  89. case IDoDecoderStepFoundPreambula:
  90. if((!level) && (DURATION_DIFF(duration, instance->common.te_short * 10) <
  91. instance->common.te_delta * 5)) {
  92. //Found Preambula
  93. instance->common.parser_step = IDoDecoderStepSaveDuration;
  94. instance->common.code_found = 0;
  95. instance->common.code_count_bit = 0;
  96. } else {
  97. instance->common.parser_step = IDoDecoderStepReset;
  98. }
  99. break;
  100. case IDoDecoderStepSaveDuration:
  101. if(level) {
  102. if(duration >= (instance->common.te_short * 5 + instance->common.te_delta)) {
  103. instance->common.parser_step = IDoDecoderStepFoundPreambula;
  104. if(instance->common.code_count_bit >=
  105. instance->common.code_min_count_bit_for_found) {
  106. instance->common.code_last_found = instance->common.code_found;
  107. instance->common.code_last_count_bit = instance->common.code_count_bit;
  108. if(instance->common.callback)
  109. instance->common.callback(
  110. (SubGhzProtocolCommon*)instance, instance->common.context);
  111. }
  112. instance->common.code_found = 0;
  113. instance->common.code_count_bit = 0;
  114. break;
  115. } else {
  116. instance->common.te_last = duration;
  117. instance->common.parser_step = IDoDecoderStepCheckDuration;
  118. }
  119. } else {
  120. instance->common.parser_step = IDoDecoderStepReset;
  121. }
  122. break;
  123. case IDoDecoderStepCheckDuration:
  124. if(!level) {
  125. if((DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  126. instance->common.te_delta) &&
  127. (DURATION_DIFF(duration, instance->common.te_long) <
  128. instance->common.te_delta * 3)) {
  129. subghz_protocol_common_add_bit(&instance->common, 0);
  130. instance->common.parser_step = IDoDecoderStepSaveDuration;
  131. } else if(
  132. (DURATION_DIFF(instance->common.te_last, instance->common.te_short) <
  133. instance->common.te_delta * 3) &&
  134. (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) {
  135. subghz_protocol_common_add_bit(&instance->common, 1);
  136. instance->common.parser_step = IDoDecoderStepSaveDuration;
  137. } else {
  138. instance->common.parser_step = IDoDecoderStepReset;
  139. }
  140. } else {
  141. instance->common.parser_step = IDoDecoderStepReset;
  142. }
  143. break;
  144. }
  145. }
  146. void subghz_protocol_ido_to_str(SubGhzProtocolIDo* instance, string_t output) {
  147. subghz_protocol_ido_check_remote_controller(instance);
  148. uint64_t code_found_reverse = subghz_protocol_common_reverse_key(
  149. instance->common.code_last_found, instance->common.code_last_count_bit);
  150. uint32_t code_fix = code_found_reverse & 0xFFFFFF;
  151. uint32_t code_hop = (code_found_reverse >> 24) & 0xFFFFFF;
  152. string_cat_printf(
  153. output,
  154. "%s %dbit\r\n"
  155. "Key:0x%lX%08lX\r\n"
  156. "Fix:%06lX \r\n"
  157. "Hop:%06lX \r\n"
  158. "Sn:%05lX Btn:%lX\r\n",
  159. instance->common.name,
  160. instance->common.code_last_count_bit,
  161. (uint32_t)(instance->common.code_last_found >> 32),
  162. (uint32_t)instance->common.code_last_found,
  163. code_fix,
  164. code_hop,
  165. instance->common.serial,
  166. instance->common.btn);
  167. }
  168. void subghz_decoder_ido_to_load_protocol(SubGhzProtocolIDo* instance, void* context) {
  169. furi_assert(context);
  170. furi_assert(instance);
  171. SubGhzProtocolCommonLoad* data = context;
  172. instance->common.code_last_found = data->code_found;
  173. instance->common.code_last_count_bit = data->code_count_bit;
  174. subghz_protocol_ido_check_remote_controller(instance);
  175. }