SubGhz: refactoring frequency analyzer and MegaCode display changes (#1221)
* SubGhz: MegaCode display changes * SubGhz: refactoring frequency analyzer * SubGhz: use one stage detection in frequency analyzer, tune bw, datarate and etc * SubGhz: tune analyzer threshold * SubGhz: raise frequency analyzer threshold and rssi sampling config * SubGhz: fix frequency analyzer, small step frequency analysis * SubGhz: subghz_frequency_analyzer_worker * SubGhz: fix SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD * SubGhz: debug logging in frequency analyzer, increase MAGN_TARGET to max value * SubGhz: reduce RSSI delay in frequency scanner * SubGhz: fix delays, remove trace logging from frequency analyzer * SubGhz: cleanup variable names and add comments Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
This commit is contained in:
		| @@ -1,24 +1,22 @@ | ||||
| #include "subghz_frequency_analyzer_worker.h" | ||||
| #include <lib/drivers/cc1101_regs.h> | ||||
| #include <lib/drivers/cc1101.h> | ||||
|  | ||||
| #include <furi.h> | ||||
|  | ||||
| #include "../subghz_i.h" | ||||
|  | ||||
| #define TAG "SubghzFrequencyAnalyzerWorker" | ||||
|  | ||||
| #define SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD -95.0f | ||||
|  | ||||
| static const uint8_t subghz_preset_ook_58khz[][2] = { | ||||
|     {CC1101_FIFOTHR, 0x47}, // The only important bit is ADC_RETENTION, FIFO Tx=33 Rx=32 | ||||
|     {CC1101_MDMCFG4, 0xF5}, // Rx BW filter is 58.035714kHz | ||||
|     {CC1101_TEST2, 0x81}, // FIFOTHR ADC_RETENTION=1 matched value | ||||
|     {CC1101_TEST1, 0x35}, // FIFOTHR ADC_RETENTION=1 matched value | ||||
|     {CC1101_MDMCFG4, 0b11110111}, // Rx BW filter is 58.035714kHz | ||||
|     /* End  */ | ||||
|     {0, 0}, | ||||
| }; | ||||
|  | ||||
| static const uint8_t subghz_preset_ook_650khz[][2] = { | ||||
|     {CC1101_FIFOTHR, 0x07}, // The only important bit is ADC_RETENTION | ||||
|     {CC1101_MDMCFG4, 0x17}, // Rx BW filter is 650.000kHz | ||||
|     {CC1101_TEST2, 0x88}, | ||||
|     {CC1101_TEST1, 0x31}, | ||||
|     {CC1101_MDMCFG4, 0b00010111}, // Rx BW filter is 650.000kHz | ||||
|     /* End  */ | ||||
|     {0, 0}, | ||||
| }; | ||||
| @@ -27,7 +25,7 @@ struct SubGhzFrequencyAnalyzerWorker { | ||||
|     FuriThread* thread; | ||||
|  | ||||
|     volatile bool worker_running; | ||||
|     uint8_t count_repet; | ||||
|     uint8_t sample_hold_counter; | ||||
|     FrequencyRSSI frequency_rssi_buf; | ||||
|     SubGhzSetting* setting; | ||||
|  | ||||
| @@ -37,6 +35,16 @@ struct SubGhzFrequencyAnalyzerWorker { | ||||
|     void* context; | ||||
| }; | ||||
|  | ||||
| static void subghz_frequency_analyzer_worker_load_registers(const uint8_t data[][2]) { | ||||
|     furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz); | ||||
|     size_t i = 0; | ||||
|     while(data[i][0]) { | ||||
|         cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, data[i][0], data[i][1]); | ||||
|         i++; | ||||
|     } | ||||
|     furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz); | ||||
| } | ||||
|  | ||||
| // running average with adaptive coefficient | ||||
| static uint32_t subghz_frequency_analyzer_worker_expRunningAverageAdaptive( | ||||
|     SubGhzFrequencyAnalyzerWorker* instance, | ||||
| @@ -62,32 +70,75 @@ static int32_t subghz_frequency_analyzer_worker_thread(void* context) { | ||||
|     SubGhzFrequencyAnalyzerWorker* instance = context; | ||||
|  | ||||
|     FrequencyRSSI frequency_rssi = {.frequency = 0, .rssi = 0}; | ||||
|     float rssi; | ||||
|     uint32_t frequency; | ||||
|     uint32_t frequency_start; | ||||
|     float rssi = 0; | ||||
|     uint32_t frequency = 0; | ||||
|     CC1101Status status; | ||||
|  | ||||
|     //Start CC1101 | ||||
|     furi_hal_subghz_reset(); | ||||
|     furi_hal_subghz_load_preset(FuriHalSubGhzPresetOok650Async); | ||||
|     furi_hal_subghz_set_frequency(433920000); | ||||
|     furi_hal_subghz_flush_rx(); | ||||
|  | ||||
|     furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz); | ||||
|     cc1101_flush_rx(&furi_hal_spi_bus_handle_subghz); | ||||
|     cc1101_flush_tx(&furi_hal_spi_bus_handle_subghz); | ||||
|     cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_IOCFG0, CC1101IocfgHW); | ||||
|     cc1101_write_reg(&furi_hal_spi_bus_handle_subghz, CC1101_MDMCFG3, | ||||
|                      0b11111111); // symbol rate | ||||
|     cc1101_write_reg( | ||||
|         &furi_hal_spi_bus_handle_subghz, | ||||
|         CC1101_AGCCTRL2, | ||||
|         0b00000111); // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAGN_TARGET 42 dB | ||||
|     cc1101_write_reg( | ||||
|         &furi_hal_spi_bus_handle_subghz, | ||||
|         CC1101_AGCCTRL1, | ||||
|         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 | ||||
|     cc1101_write_reg( | ||||
|         &furi_hal_spi_bus_handle_subghz, | ||||
|         CC1101_AGCCTRL0, | ||||
|         0b00110000); // 00 - No hysteresis, medium asymmetric dead zone, medium gain ; 11 - 64 samples agc; 00 - Normal AGC, 00 - 4dB boundary | ||||
|  | ||||
|     furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz); | ||||
|  | ||||
|     furi_hal_subghz_set_path(FuriHalSubGhzPathIsolate); | ||||
|     furi_hal_subghz_rx(); | ||||
|  | ||||
|     while(instance->worker_running) { | ||||
|         osDelay(10); | ||||
|  | ||||
|         float rssi_min = 26.0f; | ||||
|         float rssi_avg = 0; | ||||
|         size_t rssi_avg_samples = 0; | ||||
|  | ||||
|         frequency_rssi.rssi = -127.0f; | ||||
|         furi_hal_subghz_idle(); | ||||
|         furi_hal_subghz_load_registers(subghz_preset_ook_650khz); | ||||
|         subghz_frequency_analyzer_worker_load_registers(subghz_preset_ook_650khz); | ||||
|  | ||||
|         // First stage: coarse scan | ||||
|         for(size_t i = 0; i < subghz_setting_get_frequency_count(instance->setting); i++) { | ||||
|             if(furi_hal_subghz_is_frequency_valid( | ||||
|                    subghz_setting_get_frequency(instance->setting, i))) { | ||||
|                 furi_hal_subghz_idle(); | ||||
|                 frequency = furi_hal_subghz_set_frequency( | ||||
|                 furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz); | ||||
|                 cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz); | ||||
|                 frequency = cc1101_set_frequency( | ||||
|                     &furi_hal_spi_bus_handle_subghz, | ||||
|                     subghz_setting_get_frequency(instance->setting, i)); | ||||
|                 furi_hal_subghz_rx(); | ||||
|                 osDelay(3); | ||||
|  | ||||
|                 cc1101_calibrate(&furi_hal_spi_bus_handle_subghz); | ||||
|                 do { | ||||
|                     status = cc1101_get_status(&furi_hal_spi_bus_handle_subghz); | ||||
|                 } while(status.STATE != CC1101StateIDLE); | ||||
|  | ||||
|                 cc1101_switch_to_rx(&furi_hal_spi_bus_handle_subghz); | ||||
|                 furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz); | ||||
|  | ||||
|                 // delay will be in range between 1 and 2ms | ||||
|                 osDelay(2); | ||||
|  | ||||
|                 rssi = furi_hal_subghz_get_rssi(); | ||||
|  | ||||
|                 rssi_avg += rssi; | ||||
|                 rssi_avg_samples++; | ||||
|  | ||||
|                 if(rssi < rssi_min) rssi_min = rssi; | ||||
|  | ||||
|                 if(frequency_rssi.rssi < rssi) { | ||||
|                     frequency_rssi.rssi = rssi; | ||||
|                     frequency_rssi.frequency = frequency; | ||||
| @@ -95,19 +146,41 @@ static int32_t subghz_frequency_analyzer_worker_thread(void* context) { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if(frequency_rssi.rssi > -90.0) { | ||||
|             //  -0.5 ... 433.92 ... +0.5 | ||||
|             frequency_start = frequency_rssi.frequency - 250000; | ||||
|             //step 10KHz | ||||
|         FURI_LOG_T( | ||||
|             TAG, | ||||
|             "RSSI: avg %f, max %f at %u, min %f", | ||||
|             (double)(rssi_avg / rssi_avg_samples), | ||||
|             (double)frequency_rssi.rssi, | ||||
|             frequency_rssi.frequency, | ||||
|             (double)rssi_min); | ||||
|  | ||||
|         // Second stage: fine scan | ||||
|         if(frequency_rssi.rssi > SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD) { | ||||
|             FURI_LOG_D(TAG, "~:%u:%f", frequency_rssi.frequency, (double)frequency_rssi.rssi); | ||||
|  | ||||
|             frequency_rssi.rssi = -127.0; | ||||
|             furi_hal_subghz_idle(); | ||||
|             furi_hal_subghz_load_registers(subghz_preset_ook_58khz); | ||||
|             for(uint32_t i = frequency_start; i < frequency_start + 500000; i += 10000) { | ||||
|             subghz_frequency_analyzer_worker_load_registers(subghz_preset_ook_58khz); | ||||
|             //-0.3 ... 433.92 ... +0.3 step 10KHz | ||||
|             for(uint32_t i = frequency_rssi.frequency - 300000; | ||||
|                 i < frequency_rssi.frequency + 300000; | ||||
|                 i += 20000) { | ||||
|                 if(furi_hal_subghz_is_frequency_valid(i)) { | ||||
|                     furi_hal_subghz_idle(); | ||||
|                     frequency = furi_hal_subghz_set_frequency(i); | ||||
|                     furi_hal_subghz_rx(); | ||||
|                     osDelay(3); | ||||
|                     furi_hal_spi_acquire(&furi_hal_spi_bus_handle_subghz); | ||||
|                     cc1101_switch_to_idle(&furi_hal_spi_bus_handle_subghz); | ||||
|                     frequency = cc1101_set_frequency(&furi_hal_spi_bus_handle_subghz, i); | ||||
|  | ||||
|                     cc1101_calibrate(&furi_hal_spi_bus_handle_subghz); | ||||
|                     do { | ||||
|                         status = cc1101_get_status(&furi_hal_spi_bus_handle_subghz); | ||||
|                     } while(status.STATE != CC1101StateIDLE); | ||||
|  | ||||
|                     cc1101_switch_to_rx(&furi_hal_spi_bus_handle_subghz); | ||||
|                     furi_hal_spi_release(&furi_hal_spi_bus_handle_subghz); | ||||
|  | ||||
|                     // delay will be in range between 1 and 2ms | ||||
|                     osDelay(2); | ||||
|  | ||||
|                     rssi = furi_hal_subghz_get_rssi(); | ||||
|                     if(frequency_rssi.rssi < rssi) { | ||||
|                         frequency_rssi.rssi = rssi; | ||||
| @@ -117,20 +190,24 @@ static int32_t subghz_frequency_analyzer_worker_thread(void* context) { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if(frequency_rssi.rssi > -90.0) { | ||||
|             instance->count_repet = 20; | ||||
|         // Deliver results | ||||
|         if(frequency_rssi.rssi > SUBGHZ_FREQUENCY_ANALYZER_THRESHOLD) { | ||||
|             FURI_LOG_D(TAG, "=:%u:%f", frequency_rssi.frequency, (double)frequency_rssi.rssi); | ||||
|  | ||||
|             instance->sample_hold_counter = 20; | ||||
|             if(instance->filVal) { | ||||
|                 frequency_rssi.frequency = | ||||
|                     subghz_frequency_analyzer_worker_expRunningAverageAdaptive( | ||||
|                         instance, frequency_rssi.frequency); | ||||
|             } | ||||
|             if(instance->pair_callback) | ||||
|             // Deliver callback | ||||
|             if(instance->pair_callback) { | ||||
|                 instance->pair_callback( | ||||
|                     instance->context, frequency_rssi.frequency, frequency_rssi.rssi); | ||||
|  | ||||
|             } | ||||
|         } else { | ||||
|             if(instance->count_repet > 0) { | ||||
|                 instance->count_repet--; | ||||
|             if(instance->sample_hold_counter > 0) { | ||||
|                 instance->sample_hold_counter--; | ||||
|             } else { | ||||
|                 instance->filVal = 0; | ||||
|                 if(instance->pair_callback) instance->pair_callback(instance->context, 0, 0); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user