subghz_frequency_analyzer_worker.c 11 KB

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