subghz_protocol_ido.c 6.6 KB

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