subghz_test_packet.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. #include "subghz_test_packet.h"
  2. #include "../subghz_i.h"
  3. #include <math.h>
  4. #include <furi.h>
  5. #include <furi-hal.h>
  6. #include <input/input.h>
  7. #include <toolbox/level_duration.h>
  8. #include <lib/subghz/protocols/subghz_protocol_princeton.h>
  9. #define SUBGHZ_TEST_PACKET_COUNT 1000
  10. struct SubghzTestPacket {
  11. View* view;
  12. osTimerId timer;
  13. SubGhzDecoderPrinceton* decoder;
  14. SubGhzEncoderPrinceton* encoder;
  15. volatile size_t packet_rx;
  16. };
  17. typedef enum {
  18. SubghzTestPacketModelStatusRx,
  19. SubghzTestPacketModelStatusTx,
  20. } SubghzTestPacketModelStatus;
  21. typedef struct {
  22. uint8_t frequency;
  23. uint32_t real_frequency;
  24. FuriHalSubGhzPath path;
  25. float rssi;
  26. size_t packets;
  27. SubghzTestPacketModelStatus status;
  28. } SubghzTestPacketModel;
  29. volatile bool subghz_test_packet_overrun = false;
  30. static void subghz_test_packet_rx_callback(bool level, uint32_t duration, void* context) {
  31. furi_assert(context);
  32. SubghzTestPacket* instance = context;
  33. subghz_decoder_princeton_parse(instance->decoder, level, duration);
  34. }
  35. static void subghz_test_packet_rx_pt_callback(SubGhzProtocolCommon* parser, void* context) {
  36. furi_assert(context);
  37. SubghzTestPacket* instance = context;
  38. instance->packet_rx++;
  39. }
  40. static void subghz_test_packet_rssi_timer_callback(void* context) {
  41. furi_assert(context);
  42. SubghzTestPacket* instance = context;
  43. with_view_model(
  44. instance->view, (SubghzTestPacketModel * model) {
  45. if(model->status == SubghzTestPacketModelStatusRx) {
  46. model->rssi = furi_hal_subghz_get_rssi();
  47. model->packets = instance->packet_rx;
  48. } else {
  49. model->packets = SUBGHZ_TEST_PACKET_COUNT -
  50. subghz_encoder_princeton_get_repeat_left(instance->encoder);
  51. }
  52. return true;
  53. });
  54. }
  55. static void subghz_test_packet_draw(Canvas* canvas, SubghzTestPacketModel* model) {
  56. char buffer[64];
  57. canvas_set_color(canvas, ColorBlack);
  58. canvas_set_font(canvas, FontPrimary);
  59. canvas_draw_str(canvas, 0, 8, "CC1101 Packet Test");
  60. canvas_set_font(canvas, FontSecondary);
  61. // Frequency
  62. snprintf(
  63. buffer,
  64. sizeof(buffer),
  65. "Freq: %03ld.%03ld.%03ld Hz",
  66. model->real_frequency / 1000000 % 1000,
  67. model->real_frequency / 1000 % 1000,
  68. model->real_frequency % 1000);
  69. canvas_draw_str(canvas, 0, 20, buffer);
  70. // Path
  71. char* path_name = "Unknown";
  72. if(model->path == FuriHalSubGhzPathIsolate) {
  73. path_name = "isolate";
  74. } else if(model->path == FuriHalSubGhzPath433) {
  75. path_name = "433MHz";
  76. } else if(model->path == FuriHalSubGhzPath315) {
  77. path_name = "315MHz";
  78. } else if(model->path == FuriHalSubGhzPath868) {
  79. path_name = "868MHz";
  80. }
  81. snprintf(buffer, sizeof(buffer), "Path: %d - %s", model->path, path_name);
  82. canvas_draw_str(canvas, 0, 31, buffer);
  83. snprintf(buffer, sizeof(buffer), "Packets: %d", model->packets);
  84. canvas_draw_str(canvas, 0, 42, buffer);
  85. if(model->status == SubghzTestPacketModelStatusRx) {
  86. snprintf(
  87. buffer,
  88. sizeof(buffer),
  89. "RSSI: %ld.%ld dBm",
  90. (int32_t)(model->rssi),
  91. (int32_t)fabs(model->rssi * 10) % 10);
  92. canvas_draw_str(canvas, 0, 53, buffer);
  93. } else {
  94. canvas_draw_str(canvas, 0, 53, "TX");
  95. }
  96. }
  97. static bool subghz_test_packet_input(InputEvent* event, void* context) {
  98. furi_assert(context);
  99. SubghzTestPacket* instance = context;
  100. if(event->key == InputKeyBack || event->type != InputTypeShort) {
  101. return false;
  102. }
  103. with_view_model(
  104. instance->view, (SubghzTestPacketModel * model) {
  105. if(model->status == SubghzTestPacketModelStatusRx) {
  106. furi_hal_subghz_stop_async_rx();
  107. } else {
  108. furi_hal_subghz_stop_async_tx();
  109. }
  110. if(event->key == InputKeyLeft) {
  111. if(model->frequency > 0) model->frequency--;
  112. } else if(event->key == InputKeyRight) {
  113. if(model->frequency < subghz_frequencies_count - 1) model->frequency++;
  114. } else if(event->key == InputKeyDown) {
  115. if(model->path > 0) model->path--;
  116. } else if(event->key == InputKeyUp) {
  117. if(model->path < FuriHalSubGhzPath868) model->path++;
  118. } else if(event->key == InputKeyOk) {
  119. if(model->status == SubghzTestPacketModelStatusTx) {
  120. model->status = SubghzTestPacketModelStatusRx;
  121. } else {
  122. model->status = SubghzTestPacketModelStatusTx;
  123. }
  124. }
  125. model->real_frequency =
  126. furi_hal_subghz_set_frequency(subghz_frequencies[model->frequency]);
  127. furi_hal_subghz_set_path(model->path);
  128. if(model->status == SubghzTestPacketModelStatusRx) {
  129. furi_hal_subghz_start_async_rx(subghz_test_packet_rx_callback, instance);
  130. } else {
  131. subghz_encoder_princeton_set(instance->encoder, 0x00AABBCC, 1000);
  132. furi_hal_subghz_start_async_tx(subghz_encoder_princeton_yield, instance->encoder);
  133. }
  134. return true;
  135. });
  136. return true;
  137. }
  138. void subghz_test_packet_enter(void* context) {
  139. furi_assert(context);
  140. SubghzTestPacket* instance = context;
  141. furi_hal_subghz_reset();
  142. furi_hal_subghz_load_preset(FuriHalSubGhzPresetOokAsync);
  143. with_view_model(
  144. instance->view, (SubghzTestPacketModel * model) {
  145. model->frequency = subghz_frequencies_433_92;
  146. model->real_frequency =
  147. furi_hal_subghz_set_frequency(subghz_frequencies[model->frequency]);
  148. model->path = FuriHalSubGhzPathIsolate; // isolate
  149. model->rssi = 0.0f;
  150. model->status = SubghzTestPacketModelStatusRx;
  151. return true;
  152. });
  153. furi_hal_subghz_start_async_rx(subghz_test_packet_rx_callback, instance);
  154. osTimerStart(instance->timer, 1024 / 4);
  155. }
  156. void subghz_test_packet_exit(void* context) {
  157. furi_assert(context);
  158. SubghzTestPacket* instance = context;
  159. osTimerStop(instance->timer);
  160. // Reinitialize IC to default state
  161. with_view_model(
  162. instance->view, (SubghzTestPacketModel * model) {
  163. if(model->status == SubghzTestPacketModelStatusRx) {
  164. furi_hal_subghz_stop_async_rx();
  165. } else {
  166. furi_hal_subghz_stop_async_tx();
  167. }
  168. return true;
  169. });
  170. furi_hal_subghz_sleep();
  171. }
  172. SubghzTestPacket* subghz_test_packet_alloc() {
  173. SubghzTestPacket* instance = furi_alloc(sizeof(SubghzTestPacket));
  174. // View allocation and configuration
  175. instance->view = view_alloc();
  176. view_allocate_model(instance->view, ViewModelTypeLocking, sizeof(SubghzTestPacketModel));
  177. view_set_context(instance->view, instance);
  178. view_set_draw_callback(instance->view, (ViewDrawCallback)subghz_test_packet_draw);
  179. view_set_input_callback(instance->view, subghz_test_packet_input);
  180. view_set_enter_callback(instance->view, subghz_test_packet_enter);
  181. view_set_exit_callback(instance->view, subghz_test_packet_exit);
  182. instance->timer =
  183. osTimerNew(subghz_test_packet_rssi_timer_callback, osTimerPeriodic, instance, NULL);
  184. instance->decoder = subghz_decoder_princeton_alloc();
  185. subghz_protocol_common_set_callback(
  186. (SubGhzProtocolCommon*)instance->decoder, subghz_test_packet_rx_pt_callback, instance);
  187. instance->encoder = subghz_encoder_princeton_alloc();
  188. return instance;
  189. }
  190. void subghz_test_packet_free(SubghzTestPacket* instance) {
  191. furi_assert(instance);
  192. subghz_decoder_princeton_free(instance->decoder);
  193. subghz_encoder_princeton_free(instance->encoder);
  194. osTimerDelete(instance->timer);
  195. view_free(instance->view);
  196. free(instance);
  197. }
  198. View* subghz_test_packet_get_view(SubghzTestPacket* instance) {
  199. furi_assert(instance);
  200. return instance->view;
  201. }