subghz_frequency_analyzer_worker.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. #include "subghz_frequency_analyzer_worker.h"
  2. #include <lib/drivers/cc1101_regs.h>
  3. #include <furi.h>
  4. #include "../subghz_i.h"
  5. static const uint8_t subghz_preset_ook_58khz[][2] = {
  6. {CC1101_FIFOTHR, 0x47}, // The only important bit is ADC_RETENTION, FIFO Tx=33 Rx=32
  7. {CC1101_MDMCFG4, 0xF5}, // Rx BW filter is 58.035714kHz
  8. {CC1101_TEST2, 0x81}, // FIFOTHR ADC_RETENTION=1 matched value
  9. {CC1101_TEST1, 0x35}, // FIFOTHR ADC_RETENTION=1 matched value
  10. /* End */
  11. {0, 0},
  12. };
  13. static const uint8_t subghz_preset_ook_650khz[][2] = {
  14. {CC1101_FIFOTHR, 0x07}, // The only important bit is ADC_RETENTION
  15. {CC1101_MDMCFG4, 0x17}, // Rx BW filter is 650.000kHz
  16. {CC1101_TEST2, 0x88},
  17. {CC1101_TEST1, 0x31},
  18. /* End */
  19. {0, 0},
  20. };
  21. struct SubGhzFrequencyAnalyzerWorker {
  22. FuriThread* thread;
  23. volatile bool worker_running;
  24. uint8_t count_repet;
  25. FrequencyRSSI frequency_rssi_buf;
  26. SubGhzSetting* setting;
  27. float filVal;
  28. SubGhzFrequencyAnalyzerWorkerPairCallback pair_callback;
  29. void* context;
  30. };
  31. // running average with adaptive coefficient
  32. static uint32_t subghz_frequency_analyzer_worker_expRunningAverageAdaptive(
  33. SubGhzFrequencyAnalyzerWorker* instance,
  34. uint32_t newVal) {
  35. float k;
  36. float newValFloat = newVal;
  37. // the sharpness of the filter depends on the absolute value of the difference
  38. if(abs(newValFloat - instance->filVal) > 500000)
  39. k = 0.9;
  40. else
  41. k = 0.03;
  42. instance->filVal += (newValFloat - instance->filVal) * k;
  43. return (uint32_t)instance->filVal;
  44. }
  45. /** Worker thread
  46. *
  47. * @param context
  48. * @return exit code
  49. */
  50. static int32_t subghz_frequency_analyzer_worker_thread(void* context) {
  51. SubGhzFrequencyAnalyzerWorker* instance = context;
  52. FrequencyRSSI frequency_rssi = {.frequency = 0, .rssi = 0};
  53. float rssi;
  54. uint32_t frequency;
  55. uint32_t frequency_start;
  56. //Start CC1101
  57. furi_hal_subghz_reset();
  58. furi_hal_subghz_load_preset(FuriHalSubGhzPresetOok650Async);
  59. furi_hal_subghz_set_frequency(433920000);
  60. furi_hal_subghz_flush_rx();
  61. furi_hal_subghz_set_path(FuriHalSubGhzPathIsolate);
  62. furi_hal_subghz_rx();
  63. while(instance->worker_running) {
  64. osDelay(10);
  65. frequency_rssi.rssi = -127.0f;
  66. furi_hal_subghz_idle();
  67. furi_hal_subghz_load_registers(subghz_preset_ook_650khz);
  68. for(size_t i = 0; i < subghz_setting_get_frequency_count(instance->setting); i++) {
  69. if(furi_hal_subghz_is_frequency_valid(
  70. subghz_setting_get_frequency(instance->setting, i))) {
  71. furi_hal_subghz_idle();
  72. frequency = furi_hal_subghz_set_frequency(
  73. subghz_setting_get_frequency(instance->setting, i));
  74. furi_hal_subghz_rx();
  75. osDelay(3);
  76. rssi = furi_hal_subghz_get_rssi();
  77. if(frequency_rssi.rssi < rssi) {
  78. frequency_rssi.rssi = rssi;
  79. frequency_rssi.frequency = frequency;
  80. }
  81. }
  82. }
  83. if(frequency_rssi.rssi > -90.0) {
  84. // -0.5 ... 433.92 ... +0.5
  85. frequency_start = frequency_rssi.frequency - 250000;
  86. //step 10KHz
  87. frequency_rssi.rssi = -127.0;
  88. furi_hal_subghz_idle();
  89. furi_hal_subghz_load_registers(subghz_preset_ook_58khz);
  90. for(uint32_t i = frequency_start; i < frequency_start + 500000; i += 10000) {
  91. if(furi_hal_subghz_is_frequency_valid(i)) {
  92. furi_hal_subghz_idle();
  93. frequency = furi_hal_subghz_set_frequency(i);
  94. furi_hal_subghz_rx();
  95. osDelay(3);
  96. rssi = furi_hal_subghz_get_rssi();
  97. if(frequency_rssi.rssi < rssi) {
  98. frequency_rssi.rssi = rssi;
  99. frequency_rssi.frequency = frequency;
  100. }
  101. }
  102. }
  103. }
  104. if(frequency_rssi.rssi > -90.0) {
  105. instance->count_repet = 20;
  106. if(instance->filVal) {
  107. frequency_rssi.frequency =
  108. subghz_frequency_analyzer_worker_expRunningAverageAdaptive(
  109. instance, frequency_rssi.frequency);
  110. }
  111. if(instance->pair_callback)
  112. instance->pair_callback(
  113. instance->context, frequency_rssi.frequency, frequency_rssi.rssi);
  114. } else {
  115. if(instance->count_repet > 0) {
  116. instance->count_repet--;
  117. } else {
  118. instance->filVal = 0;
  119. if(instance->pair_callback) instance->pair_callback(instance->context, 0, 0);
  120. }
  121. }
  122. }
  123. //Stop CC1101
  124. furi_hal_subghz_idle();
  125. furi_hal_subghz_sleep();
  126. return 0;
  127. }
  128. SubGhzFrequencyAnalyzerWorker* subghz_frequency_analyzer_worker_alloc() {
  129. SubGhzFrequencyAnalyzerWorker* instance = malloc(sizeof(SubGhzFrequencyAnalyzerWorker));
  130. instance->thread = furi_thread_alloc();
  131. furi_thread_set_name(instance->thread, "SubGhzFAWorker");
  132. furi_thread_set_stack_size(instance->thread, 2048);
  133. furi_thread_set_context(instance->thread, instance);
  134. furi_thread_set_callback(instance->thread, subghz_frequency_analyzer_worker_thread);
  135. instance->setting = subghz_setting_alloc();
  136. subghz_setting_load(instance->setting, "/ext/subghz/assets/setting_frequency_analyzer_user");
  137. return instance;
  138. }
  139. void subghz_frequency_analyzer_worker_free(SubGhzFrequencyAnalyzerWorker* instance) {
  140. furi_assert(instance);
  141. furi_thread_free(instance->thread);
  142. subghz_setting_free(instance->setting);
  143. free(instance);
  144. }
  145. void subghz_frequency_analyzer_worker_set_pair_callback(
  146. SubGhzFrequencyAnalyzerWorker* instance,
  147. SubGhzFrequencyAnalyzerWorkerPairCallback callback,
  148. void* context) {
  149. furi_assert(instance);
  150. furi_assert(context);
  151. instance->pair_callback = callback;
  152. instance->context = context;
  153. }
  154. void subghz_frequency_analyzer_worker_start(SubGhzFrequencyAnalyzerWorker* instance) {
  155. furi_assert(instance);
  156. furi_assert(!instance->worker_running);
  157. instance->worker_running = true;
  158. furi_thread_start(instance->thread);
  159. }
  160. void subghz_frequency_analyzer_worker_stop(SubGhzFrequencyAnalyzerWorker* instance) {
  161. furi_assert(instance);
  162. furi_assert(instance->worker_running);
  163. instance->worker_running = false;
  164. furi_thread_join(instance->thread);
  165. }
  166. bool subghz_frequency_analyzer_worker_is_running(SubGhzFrequencyAnalyzerWorker* instance) {
  167. furi_assert(instance);
  168. return instance->worker_running;
  169. }