subghz_frequency_analyzer_worker.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. #include "subghz_frequency_analyzer_worker.h"
  2. #include <lib/drivers/cc1101.h>
  3. #include <furi.h>
  4. #include "../subghz_i.h"
  5. #define TAG "SubghzFrequencyAnalyzerWorker"
  6. #define SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD -95.0f
  7. static const uint8_t subghz_preset_ook_58khz[][2] = {
  8. {CC1101_MDMCFG4, 0b11110111}, // Rx BW filter is 58.035714kHz
  9. /* End */
  10. {0, 0},
  11. };
  12. static const uint8_t subghz_preset_ook_650khz[][2] = {
  13. {CC1101_MDMCFG4, 0b00010111}, // Rx BW filter is 650.000kHz
  14. /* End */
  15. {0, 0},
  16. };
  17. struct SubGhzFrequencyAnalyzerWorker {
  18. FuriThread* thread;
  19. volatile bool worker_running;
  20. uint8_t sample_hold_counter;
  21. FrequencyRSSI frequency_rssi_buf;
  22. SubGhzSetting* setting;
  23. float filVal;
  24. SubGhzFrequencyAnalyzerWorkerPairCallback pair_callback;
  25. void* context;
  26. };
  27. static void subghz_frequency_analyzer_worker_load_registers(const uint8_t data[][2]) {
  28. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  29. size_t i = 0;
  30. while(data[i][0]) {
  31. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, data[i][0], data[i][1]);
  32. i++;
  33. }
  34. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  35. }
  36. // running average with adaptive coefficient
  37. static uint32_t subghz_frequency_analyzer_worker_expRunningAverageAdaptive(
  38. SubGhzFrequencyAnalyzerWorker* instance,
  39. uint32_t newVal) {
  40. float k;
  41. float newValFloat = newVal;
  42. // the sharpness of the filter depends on the absolute value of the difference
  43. if(fabs(newValFloat - instance->filVal) > 500000)
  44. k = 0.9;
  45. else
  46. k = 0.03;
  47. instance->filVal += (newValFloat - instance->filVal) * k;
  48. return (uint32_t)instance->filVal;
  49. }
  50. /** Worker thread
  51. *
  52. * @param context
  53. * @return exit code
  54. */
  55. static int32_t subghz_frequency_analyzer_worker_thread(void* context) {
  56. SubGhzFrequencyAnalyzerWorker* instance = context;
  57. FrequencyRSSI frequency_rssi = {.frequency = 0, .rssi = 0};
  58. float rssi = 0;
  59. uint32_t frequency = 0;
  60. CC1101Status status;
  61. //Start CC1101
  62. furi_hal_subghz_reset();
  63. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  64. cc1101_flush_rx(&furi_hal_spi_bus_handle_subghz);
  65. cc1101_flush_tx(&furi_hal_spi_bus_handle_subghz);
  66. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHW);
  67. cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_MDMCFG3,
  68. 0b11111111); // symbol rate
  69. cc1101_write_reg(
  70. &furi_hal_spi_bus_handle_subghz,
  71. CC1101_AGCCTRL2,
  72. 0b00000111); // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAGN_TARGET 42 dB
  73. cc1101_write_reg(
  74. &furi_hal_spi_bus_handle_subghz,
  75. CC1101_AGCCTRL1,
  76. 0b00001000); // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 1000 - Absolute carrier sense threshold disabled
  77. cc1101_write_reg(
  78. &furi_hal_spi_bus_handle_subghz,
  79. CC1101_AGCCTRL0,
  80. 0b00110000); // 00 - No hysteresis, medium asymmetric dead zone, medium gain ; 11 - 64 samples agc; 00 - Normal AGC, 00 - 4dB boundary
  81. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  82. furi_hal_subghz_set_path(FuriHalSubGhzPathIsolate);
  83. while(instance->worker_running) {
  84. osDelay(10);
  85. float rssi_min = 26.0f;
  86. float rssi_avg = 0;
  87. size_t rssi_avg_samples = 0;
  88. frequency_rssi.rssi = -127.0f;
  89. furi_hal_subghz_idle();
  90. subghz_frequency_analyzer_worker_load_registers(subghz_preset_ook_650khz);
  91. // First stage: coarse scan
  92. for(size_t i = 0; i < subghz_setting_get_frequency_count(instance->setting); i++) {
  93. if(furi_hal_subghz_is_frequency_valid(
  94. subghz_setting_get_frequency(instance->setting, i))) {
  95. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  96. cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz);
  97. frequency = cc1101_set_frequency(
  98. &furi_hal_spi_bus_handle_subghz,
  99. subghz_setting_get_frequency(instance->setting, i));
  100. cc1101_calibrate(&furi_hal_spi_bus_handle_subghz);
  101. do {
  102. status = cc1101_get_status(&furi_hal_spi_bus_handle_subghz);
  103. } while(status.STATE != CC1101StateIDLE);
  104. cc1101_switch_to_rx(&furi_hal_spi_bus_handle_subghz);
  105. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  106. // delay will be in range between 1 and 2ms
  107. osDelay(2);
  108. rssi = furi_hal_subghz_get_rssi();
  109. rssi_avg += rssi;
  110. rssi_avg_samples++;
  111. if(rssi < rssi_min) rssi_min = rssi;
  112. if(frequency_rssi.rssi < rssi) {
  113. frequency_rssi.rssi = rssi;
  114. frequency_rssi.frequency = frequency;
  115. }
  116. }
  117. }
  118. FURI_LOG_T(
  119. TAG,
  120. "RSSI: avg %f, max %f at %u, min %f",
  121. (double)(rssi_avg / rssi_avg_samples),
  122. (double)frequency_rssi.rssi,
  123. frequency_rssi.frequency,
  124. (double)rssi_min);
  125. // Second stage: fine scan
  126. if(frequency_rssi.rssi > SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD) {
  127. FURI_LOG_D(TAG, "~:%u:%f", frequency_rssi.frequency, (double)frequency_rssi.rssi);
  128. frequency_rssi.rssi = -127.0;
  129. furi_hal_subghz_idle();
  130. subghz_frequency_analyzer_worker_load_registers(subghz_preset_ook_58khz);
  131. //-0.3 ... 433.92 ... +0.3 step 10KHz
  132. for(uint32_t i = frequency_rssi.frequency - 300000;
  133. i < frequency_rssi.frequency + 300000;
  134. i += 20000) {
  135. if(furi_hal_subghz_is_frequency_valid(i)) {
  136. furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz);
  137. cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz);
  138. frequency = cc1101_set_frequency(&furi_hal_spi_bus_handle_subghz, i);
  139. cc1101_calibrate(&furi_hal_spi_bus_handle_subghz);
  140. do {
  141. status = cc1101_get_status(&furi_hal_spi_bus_handle_subghz);
  142. } while(status.STATE != CC1101StateIDLE);
  143. cc1101_switch_to_rx(&furi_hal_spi_bus_handle_subghz);
  144. furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz);
  145. // delay will be in range between 1 and 2ms
  146. osDelay(2);
  147. rssi = furi_hal_subghz_get_rssi();
  148. if(frequency_rssi.rssi < rssi) {
  149. frequency_rssi.rssi = rssi;
  150. frequency_rssi.frequency = frequency;
  151. }
  152. }
  153. }
  154. }
  155. // Deliver results
  156. if(frequency_rssi.rssi > SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD) {
  157. FURI_LOG_D(TAG, "=:%u:%f", frequency_rssi.frequency, (double)frequency_rssi.rssi);
  158. instance->sample_hold_counter = 20;
  159. if(instance->filVal) {
  160. frequency_rssi.frequency =
  161. subghz_frequency_analyzer_worker_expRunningAverageAdaptive(
  162. instance, frequency_rssi.frequency);
  163. }
  164. // Deliver callback
  165. if(instance->pair_callback) {
  166. instance->pair_callback(
  167. instance->context, frequency_rssi.frequency, frequency_rssi.rssi);
  168. }
  169. } else {
  170. if(instance->sample_hold_counter > 0) {
  171. instance->sample_hold_counter--;
  172. } else {
  173. instance->filVal = 0;
  174. if(instance->pair_callback) instance->pair_callback(instance->context, 0, 0);
  175. }
  176. }
  177. }
  178. //Stop CC1101
  179. furi_hal_subghz_idle();
  180. furi_hal_subghz_sleep();
  181. return 0;
  182. }
  183. SubGhzFrequencyAnalyzerWorker* subghz_frequency_analyzer_worker_alloc() {
  184. SubGhzFrequencyAnalyzerWorker* instance = malloc(sizeof(SubGhzFrequencyAnalyzerWorker));
  185. instance->thread = furi_thread_alloc();
  186. furi_thread_set_name(instance->thread, "SubGhzFAWorker");
  187. furi_thread_set_stack_size(instance->thread, 2048);
  188. furi_thread_set_context(instance->thread, instance);
  189. furi_thread_set_callback(instance->thread, subghz_frequency_analyzer_worker_thread);
  190. instance->setting = subghz_setting_alloc();
  191. subghz_setting_load(instance->setting, "/ext/subghz/assets/setting_frequency_analyzer_user");
  192. return instance;
  193. }
  194. void subghz_frequency_analyzer_worker_free(SubGhzFrequencyAnalyzerWorker* instance) {
  195. furi_assert(instance);
  196. furi_thread_free(instance->thread);
  197. subghz_setting_free(instance->setting);
  198. free(instance);
  199. }
  200. void subghz_frequency_analyzer_worker_set_pair_callback(
  201. SubGhzFrequencyAnalyzerWorker* instance,
  202. SubGhzFrequencyAnalyzerWorkerPairCallback callback,
  203. void* context) {
  204. furi_assert(instance);
  205. furi_assert(context);
  206. instance->pair_callback = callback;
  207. instance->context = context;
  208. }
  209. void subghz_frequency_analyzer_worker_start(SubGhzFrequencyAnalyzerWorker* instance) {
  210. furi_assert(instance);
  211. furi_assert(!instance->worker_running);
  212. instance->worker_running = true;
  213. furi_thread_start(instance->thread);
  214. }
  215. void subghz_frequency_analyzer_worker_stop(SubGhzFrequencyAnalyzerWorker* instance) {
  216. furi_assert(instance);
  217. furi_assert(instance->worker_running);
  218. instance->worker_running = false;
  219. furi_thread_join(instance->thread);
  220. }
  221. bool subghz_frequency_analyzer_worker_is_running(SubGhzFrequencyAnalyzerWorker* instance) {
  222. furi_assert(instance);
  223. return instance->worker_running;
  224. }