rolling_flaws_subghz_receive.c 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. #include "rolling_flaws_subghz_receive.h"
  2. static SubGhzEnvironment* load_environment() {
  3. SubGhzEnvironment* environment = subghz_environment_alloc();
  4. subghz_environment_load_keystore(environment, SUBGHZ_KEYSTORE_DIR_NAME);
  5. subghz_environment_load_keystore(environment, SUBGHZ_KEYSTORE_DIR_USER_NAME);
  6. subghz_environment_set_alutech_at_4n_rainbow_table_file_name(
  7. environment, SUBGHZ_ALUTECH_AT_4N_DIR_NAME);
  8. subghz_environment_set_nice_flor_s_rainbow_table_file_name(
  9. environment, SUBGHZ_NICE_FLOR_S_DIR_NAME);
  10. subghz_environment_set_protocol_registry(environment, (void*)&subghz_protocol_registry);
  11. return environment;
  12. }
  13. RollingFlawsSubGhz* rolling_flaws_subghz_alloc() {
  14. RollingFlawsSubGhz* subghz = malloc(sizeof(RollingFlawsSubGhz));
  15. subghz->status = SUBGHZ_RECEIVER_UNINITIALIZED;
  16. subghz->environment = load_environment();
  17. subghz->stream = furi_stream_buffer_alloc(sizeof(LevelDuration) * 1024, sizeof(LevelDuration));
  18. furi_check(subghz->stream);
  19. subghz->overrun = false;
  20. return subghz;
  21. }
  22. void rolling_flaws_subghz_free(RollingFlawsSubGhz* subghz) {
  23. subghz_environment_free(subghz->environment);
  24. furi_stream_buffer_free(subghz->stream);
  25. free(subghz);
  26. }
  27. static void
  28. rx_callback(SubGhzReceiver* receiver, SubGhzProtocolDecoderBase* decoder_base, void* cxt) {
  29. RollingFlawsSubGhz* context = (RollingFlawsSubGhz*)cxt;
  30. FuriString* buffer = furi_string_alloc();
  31. subghz_protocol_decoder_base_get_string(decoder_base, buffer);
  32. subghz_receiver_reset(receiver);
  33. FURI_LOG_I(TAG, "PACKET:\r\n%s", furi_string_get_cstr(buffer));
  34. if(context->callback) {
  35. if(!context->callback(buffer, context->callback_context)) {
  36. context->status = SUBGHZ_RECEIVER_SYNCHRONIZED;
  37. }
  38. }
  39. furi_string_free(buffer);
  40. }
  41. static void rx_capture_callback(bool level, uint32_t duration, void* context) {
  42. RollingFlawsSubGhz* instance = context;
  43. LevelDuration level_duration = level_duration_make(level, duration);
  44. if(instance->overrun) {
  45. instance->overrun = false;
  46. level_duration = level_duration_reset();
  47. }
  48. size_t ret =
  49. furi_stream_buffer_send(instance->stream, &level_duration, sizeof(LevelDuration), 0);
  50. if(sizeof(LevelDuration) != ret) {
  51. instance->overrun = true;
  52. }
  53. }
  54. static int32_t listen_rx(void* ctx) {
  55. RollingFlawsSubGhz* context = (RollingFlawsSubGhz*)ctx;
  56. context->status = SUBGHZ_RECEIVER_LISTENING;
  57. LevelDuration level_duration;
  58. FURI_LOG_I(TAG, "listen_rx started...");
  59. while(context->status == SUBGHZ_RECEIVER_LISTENING) {
  60. int ret = furi_stream_buffer_receive(
  61. context->stream, &level_duration, sizeof(LevelDuration), 10);
  62. if(ret == sizeof(LevelDuration)) {
  63. if(level_duration_is_reset(level_duration)) {
  64. subghz_receiver_reset(context->receiver);
  65. } else {
  66. bool level = level_duration_get_level(level_duration);
  67. uint32_t duration = level_duration_get_duration(level_duration);
  68. subghz_receiver_decode(context->receiver, level, duration);
  69. }
  70. }
  71. }
  72. FURI_LOG_I(TAG, "listen_rx exiting...");
  73. context->status = SUBGHZ_RECEIVER_NOTLISTENING;
  74. return 0;
  75. }
  76. void start_listening(
  77. RollingFlawsSubGhz* context,
  78. uint32_t frequency,
  79. SubghzPacketCallback callback,
  80. void* callback_context) {
  81. context->status = SUBGHZ_RECEIVER_INITIALIZING;
  82. context->callback = callback;
  83. context->callback_context = callback_context;
  84. subghz_devices_init();
  85. const SubGhzDevice* device = subghz_devices_get_by_name(SUBGHZ_DEVICE_CC1101_INT_NAME);
  86. if(!subghz_devices_is_frequency_valid(device, frequency)) {
  87. FURI_LOG_E(TAG, "Frequency not in range. %lu\r\n", frequency);
  88. subghz_devices_deinit();
  89. return;
  90. }
  91. context->receiver = subghz_receiver_alloc_init(context->environment);
  92. subghz_receiver_set_filter(context->receiver, SubGhzProtocolFlag_Decodable);
  93. subghz_receiver_set_rx_callback(context->receiver, rx_callback, context);
  94. subghz_devices_begin(device);
  95. subghz_devices_reset(device);
  96. subghz_devices_load_preset(device, FuriHalSubGhzPresetOok650Async, NULL);
  97. frequency = subghz_devices_set_frequency(device, frequency);
  98. furi_hal_power_suppress_charge_enter();
  99. subghz_devices_start_async_rx(device, rx_capture_callback, context);
  100. FURI_LOG_I(TAG, "Listening at frequency: %lu\r\n", frequency);
  101. context->thread = furi_thread_alloc_ex("RX", 1024, listen_rx, context);
  102. furi_thread_start(context->thread);
  103. }
  104. void stop_listening(RollingFlawsSubGhz* context) {
  105. if(context->status == SUBGHZ_RECEIVER_UNINITIALIZED) {
  106. return;
  107. }
  108. context->status = SUBGHZ_RECEIVER_UNINITIALING;
  109. FURI_LOG_D(TAG, "Stopping listening...");
  110. furi_thread_join(context->thread);
  111. furi_thread_free(context->thread);
  112. furi_hal_power_suppress_charge_exit();
  113. const SubGhzDevice* device = subghz_devices_get_by_name(SUBGHZ_DEVICE_CC1101_INT_NAME);
  114. subghz_devices_stop_async_rx(device);
  115. subghz_receiver_free(context->receiver);
  116. subghz_devices_deinit();
  117. context->status = SUBGHZ_RECEIVER_UNINITIALIZED;
  118. }