|
|
@@ -7,20 +7,19 @@
|
|
|
|
|
|
#define TAG "SubGhzSetting"
|
|
|
|
|
|
-#define SUBGHZ_SETTING_FILE_VERSION 1
|
|
|
#define SUBGHZ_SETTING_FILE_TYPE "Flipper SubGhz Setting File"
|
|
|
+#define SUBGHZ_SETTING_FILE_VERSION 1
|
|
|
|
|
|
-typedef enum {
|
|
|
- SubGhzSettingStateNoLoad = 0,
|
|
|
- SubGhzSettingStateLoadFrequencyDefault,
|
|
|
- SubGhzSettingStateOkLoad,
|
|
|
-} SubGhzSettingState;
|
|
|
+#define FREQUENCY_FLAG_DEFAULT (1 << 31)
|
|
|
+#define FREQUENCY_MASK (0xFFFFFFFF ^ FREQUENCY_FLAG_DEFAULT)
|
|
|
|
|
|
-static const uint32_t subghz_frequencies[] = {
|
|
|
+/* Default */
|
|
|
+static const uint32_t subghz_frequency_list[] = {
|
|
|
/* 300 - 348 */
|
|
|
300000000,
|
|
|
303875000,
|
|
|
304250000,
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
|
|
|
@@ -29,7 +28,7 @@ static const uint32_t subghz_frequencies[] = {
|
|
|
418000000,
|
|
|
433075000, /* LPD433 first */
|
|
|
433420000,
|
|
|
- 433920000, /* LPD433 mid */
|
|
|
+ 433920000 | FREQUENCY_FLAG_DEFAULT, /* LPD433 mid */
|
|
|
434420000,
|
|
|
434775000, /* LPD433 last channels */
|
|
|
438900000,
|
|
|
@@ -40,7 +39,9 @@ static const uint32_t subghz_frequencies[] = {
|
|
|
925000000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_hopper_frequencies[] = {
|
|
|
+
|
|
|
+static const uint32_t subghz_hopper_frequency_list[] = {
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
390000000,
|
|
|
@@ -48,13 +49,14 @@ static const uint32_t subghz_hopper_frequencies[] = {
|
|
|
868350000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_frequency_default_index = 9;
|
|
|
|
|
|
-static const uint32_t subghz_frequencies_region_eu_ru[] = {
|
|
|
+/* Europe and Russia */
|
|
|
+static const uint32_t subghz_frequency_list_region_eu_ru[] = {
|
|
|
/* 300 - 348 */
|
|
|
300000000,
|
|
|
303875000,
|
|
|
304250000,
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
|
|
|
@@ -63,7 +65,7 @@ static const uint32_t subghz_frequencies_region_eu_ru[] = {
|
|
|
418000000,
|
|
|
433075000, /* LPD433 first */
|
|
|
433420000,
|
|
|
- 433920000, /* LPD433 mid */
|
|
|
+ 433920000 | FREQUENCY_FLAG_DEFAULT, /* LPD433 mid */
|
|
|
434420000,
|
|
|
434775000, /* LPD433 last channels */
|
|
|
438900000,
|
|
|
@@ -74,7 +76,8 @@ static const uint32_t subghz_frequencies_region_eu_ru[] = {
|
|
|
925000000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_hopper_frequencies_region_eu_ru[] = {
|
|
|
+static const uint32_t subghz_hopper_frequency_list_region_eu_ru[] = {
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
390000000,
|
|
|
@@ -82,13 +85,14 @@ static const uint32_t subghz_hopper_frequencies_region_eu_ru[] = {
|
|
|
868350000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_frequency_default_index_region_eu_ru = 9;
|
|
|
|
|
|
-static const uint32_t subghz_frequencies_region_us_ca_au[] = {
|
|
|
+/* Region 0 */
|
|
|
+static const uint32_t subghz_frequency_list_region_us_ca_au[] = {
|
|
|
/* 300 - 348 */
|
|
|
300000000,
|
|
|
303875000,
|
|
|
304250000,
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
|
|
|
@@ -97,7 +101,7 @@ static const uint32_t subghz_frequencies_region_us_ca_au[] = {
|
|
|
418000000,
|
|
|
433075000, /* LPD433 first */
|
|
|
433420000,
|
|
|
- 433920000, /* LPD433 mid */
|
|
|
+ 433920000 | FREQUENCY_FLAG_DEFAULT, /* LPD433 mid */
|
|
|
434420000,
|
|
|
434775000, /* LPD433 last channels */
|
|
|
438900000,
|
|
|
@@ -108,7 +112,8 @@ static const uint32_t subghz_frequencies_region_us_ca_au[] = {
|
|
|
925000000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_hopper_frequencies_region_us_ca_au[] = {
|
|
|
+static const uint32_t subghz_hopper_frequency_list_region_us_ca_au[] = {
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
390000000,
|
|
|
@@ -116,13 +121,13 @@ static const uint32_t subghz_hopper_frequencies_region_us_ca_au[] = {
|
|
|
868350000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_frequency_default_index_region_us_ca_au = 9;
|
|
|
|
|
|
-static const uint32_t subghz_frequencies_region_jp[] = {
|
|
|
+static const uint32_t subghz_frequency_list_region_jp[] = {
|
|
|
/* 300 - 348 */
|
|
|
300000000,
|
|
|
303875000,
|
|
|
304250000,
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
|
|
|
@@ -131,7 +136,7 @@ static const uint32_t subghz_frequencies_region_jp[] = {
|
|
|
418000000,
|
|
|
433075000, /* LPD433 first */
|
|
|
433420000,
|
|
|
- 433920000, /* LPD433 mid */
|
|
|
+ 433920000 | FREQUENCY_FLAG_DEFAULT, /* LPD433 mid */
|
|
|
434420000,
|
|
|
434775000, /* LPD433 last channels */
|
|
|
438900000,
|
|
|
@@ -142,7 +147,8 @@ static const uint32_t subghz_frequencies_region_jp[] = {
|
|
|
925000000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_hopper_frequencies_region_jp[] = {
|
|
|
+static const uint32_t subghz_hopper_frequency_list_region_jp[] = {
|
|
|
+ 310000000,
|
|
|
315000000,
|
|
|
318000000,
|
|
|
390000000,
|
|
|
@@ -150,72 +156,88 @@ static const uint32_t subghz_hopper_frequencies_region_jp[] = {
|
|
|
868350000,
|
|
|
0,
|
|
|
};
|
|
|
-static const uint32_t subghz_frequency_default_index_region_jp = 9;
|
|
|
|
|
|
-LIST_DEF(FrequenciesList, uint32_t)
|
|
|
+LIST_DEF(FrequencyList, uint32_t)
|
|
|
+
|
|
|
+#define M_OPL_FrequencyList_t() LIST_OPLIST(FrequencyList)
|
|
|
|
|
|
struct SubGhzSetting {
|
|
|
- FrequenciesList_t frequencies;
|
|
|
- FrequenciesList_t hopper_frequencies;
|
|
|
- size_t frequencies_count;
|
|
|
- size_t hopper_frequencies_count;
|
|
|
- uint32_t frequency_default_index;
|
|
|
+ FrequencyList_t frequencies;
|
|
|
+ FrequencyList_t hopper_frequencies;
|
|
|
};
|
|
|
|
|
|
SubGhzSetting* subghz_setting_alloc(void) {
|
|
|
SubGhzSetting* instance = malloc(sizeof(SubGhzSetting));
|
|
|
- FrequenciesList_init(instance->frequencies);
|
|
|
- FrequenciesList_init(instance->hopper_frequencies);
|
|
|
+ FrequencyList_init(instance->frequencies);
|
|
|
+ FrequencyList_init(instance->hopper_frequencies);
|
|
|
return instance;
|
|
|
}
|
|
|
|
|
|
void subghz_setting_free(SubGhzSetting* instance) {
|
|
|
furi_assert(instance);
|
|
|
- FrequenciesList_clear(instance->frequencies);
|
|
|
- FrequenciesList_clear(instance->hopper_frequencies);
|
|
|
+ FrequencyList_clear(instance->frequencies);
|
|
|
+ FrequencyList_clear(instance->hopper_frequencies);
|
|
|
free(instance);
|
|
|
}
|
|
|
|
|
|
-void subghz_setting_load_default(
|
|
|
+static void subghz_setting_load_default_region(
|
|
|
SubGhzSetting* instance,
|
|
|
const uint32_t frequencies[],
|
|
|
- const uint32_t hopper_frequencies[],
|
|
|
- const uint32_t frequency_default_index) {
|
|
|
+ const uint32_t hopper_frequencies[]) {
|
|
|
furi_assert(instance);
|
|
|
- size_t i = 0;
|
|
|
- FrequenciesList_clear(instance->frequencies);
|
|
|
- FrequenciesList_clear(instance->hopper_frequencies);
|
|
|
- i = 0;
|
|
|
- while(frequencies[i]) {
|
|
|
- FrequenciesList_push_back(instance->frequencies, frequencies[i]);
|
|
|
- i++;
|
|
|
+
|
|
|
+ FrequencyList_reset(instance->frequencies);
|
|
|
+ FrequencyList_reset(instance->hopper_frequencies);
|
|
|
+
|
|
|
+ while(*frequencies) {
|
|
|
+ FrequencyList_push_back(instance->frequencies, *frequencies);
|
|
|
+ frequencies++;
|
|
|
}
|
|
|
- instance->frequencies_count = i;
|
|
|
|
|
|
- i = 0;
|
|
|
- while(hopper_frequencies[i]) {
|
|
|
- FrequenciesList_push_back(instance->hopper_frequencies, hopper_frequencies[i]);
|
|
|
- i++;
|
|
|
+ while(*hopper_frequencies) {
|
|
|
+ FrequencyList_push_back(instance->hopper_frequencies, *hopper_frequencies);
|
|
|
+ hopper_frequencies++;
|
|
|
}
|
|
|
- instance->hopper_frequencies_count = i;
|
|
|
+}
|
|
|
|
|
|
- instance->frequency_default_index = frequency_default_index;
|
|
|
+void subghz_setting_load_default(SubGhzSetting* instance) {
|
|
|
+ switch(furi_hal_version_get_hw_region()) {
|
|
|
+ case FuriHalVersionRegionEuRu:
|
|
|
+ subghz_setting_load_default_region(
|
|
|
+ instance,
|
|
|
+ subghz_frequency_list_region_eu_ru,
|
|
|
+ subghz_hopper_frequency_list_region_eu_ru);
|
|
|
+ break;
|
|
|
+ case FuriHalVersionRegionUsCaAu:
|
|
|
+ subghz_setting_load_default_region(
|
|
|
+ instance,
|
|
|
+ subghz_frequency_list_region_us_ca_au,
|
|
|
+ subghz_hopper_frequency_list_region_us_ca_au);
|
|
|
+ break;
|
|
|
+ case FuriHalVersionRegionJp:
|
|
|
+ subghz_setting_load_default_region(
|
|
|
+ instance, subghz_frequency_list_region_jp, subghz_hopper_frequency_list_region_jp);
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ subghz_setting_load_default_region(
|
|
|
+ instance, subghz_frequency_list, subghz_hopper_frequency_list);
|
|
|
+ break;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
void subghz_setting_load(SubGhzSetting* instance, const char* file_path) {
|
|
|
furi_assert(instance);
|
|
|
|
|
|
- FrequenciesList_clear(instance->frequencies);
|
|
|
- FrequenciesList_clear(instance->hopper_frequencies);
|
|
|
-
|
|
|
Storage* storage = furi_record_open("storage");
|
|
|
FlipperFormat* fff_data_file = flipper_format_file_alloc(storage);
|
|
|
|
|
|
string_t temp_str;
|
|
|
string_init(temp_str);
|
|
|
uint32_t temp_data32;
|
|
|
- SubGhzSettingState loading = SubGhzSettingStateNoLoad;
|
|
|
- uint16_t i = 0;
|
|
|
+ bool temp_bool;
|
|
|
+
|
|
|
+ subghz_setting_load_default(instance);
|
|
|
|
|
|
if(file_path) {
|
|
|
do {
|
|
|
@@ -236,63 +258,60 @@ void subghz_setting_load(SubGhzSetting* instance, const char* file_path) {
|
|
|
break;
|
|
|
}
|
|
|
|
|
|
+ // Standard frequencies (optional)
|
|
|
+ temp_bool = true;
|
|
|
+ flipper_format_read_bool(fff_data_file, "add_standard_frequencies", &temp_bool, 1);
|
|
|
+ if(!temp_bool) {
|
|
|
+ FURI_LOG_I(TAG, "Removing standard frequencies");
|
|
|
+ FrequencyList_reset(instance->frequencies);
|
|
|
+ FrequencyList_reset(instance->hopper_frequencies);
|
|
|
+ } else {
|
|
|
+ FURI_LOG_I(TAG, "Keeping standard frequencies");
|
|
|
+ }
|
|
|
+
|
|
|
+ // Load frequencies
|
|
|
if(!flipper_format_rewind(fff_data_file)) {
|
|
|
FURI_LOG_E(TAG, "Rewind error");
|
|
|
break;
|
|
|
}
|
|
|
- i = 0;
|
|
|
while(flipper_format_read_uint32(
|
|
|
- fff_data_file, "Frequency", (uint32_t*)&temp_data32, 1)) {
|
|
|
+ fff_data_file, "frequency", (uint32_t*)&temp_data32, 1)) {
|
|
|
if(furi_hal_subghz_is_frequency_valid(temp_data32)) {
|
|
|
FURI_LOG_I(TAG, "Frequency loaded %lu", temp_data32);
|
|
|
- FrequenciesList_push_back(instance->frequencies, temp_data32);
|
|
|
- i++;
|
|
|
+ FrequencyList_push_back(instance->frequencies, temp_data32);
|
|
|
} else {
|
|
|
FURI_LOG_E(TAG, "Frequency not supported %lu", temp_data32);
|
|
|
}
|
|
|
}
|
|
|
- instance->frequencies_count = i;
|
|
|
|
|
|
+ // Load hopper frequencies
|
|
|
if(!flipper_format_rewind(fff_data_file)) {
|
|
|
FURI_LOG_E(TAG, "Rewind error");
|
|
|
break;
|
|
|
}
|
|
|
- i = 0;
|
|
|
while(flipper_format_read_uint32(
|
|
|
- fff_data_file, "Hopper_frequency", (uint32_t*)&temp_data32, 1)) {
|
|
|
+ fff_data_file, "hopper_frequency", (uint32_t*)&temp_data32, 1)) {
|
|
|
if(furi_hal_subghz_is_frequency_valid(temp_data32)) {
|
|
|
FURI_LOG_I(TAG, "Hopper frequency loaded %lu", temp_data32);
|
|
|
- FrequenciesList_push_back(instance->hopper_frequencies, temp_data32);
|
|
|
- i++;
|
|
|
+ FrequencyList_push_back(instance->hopper_frequencies, temp_data32);
|
|
|
} else {
|
|
|
FURI_LOG_E(TAG, "Hopper frequency not supported %lu", temp_data32);
|
|
|
}
|
|
|
}
|
|
|
- instance->hopper_frequencies_count = i;
|
|
|
|
|
|
+ // Default frequency (optional)
|
|
|
if(!flipper_format_rewind(fff_data_file)) {
|
|
|
FURI_LOG_E(TAG, "Rewind error");
|
|
|
break;
|
|
|
}
|
|
|
- if(!flipper_format_read_uint32(
|
|
|
- fff_data_file, "Frequency_default", (uint32_t*)&temp_data32, 1)) {
|
|
|
- FURI_LOG_E(TAG, "Frequency default missing");
|
|
|
- break;
|
|
|
- }
|
|
|
-
|
|
|
- for(i = 0; i < instance->frequencies_count; i++) {
|
|
|
- if(subghz_setting_get_frequency(instance, i) == temp_data32) {
|
|
|
- instance->frequency_default_index = i;
|
|
|
- FURI_LOG_I(TAG, "Frequency default index %lu", i);
|
|
|
- loading = SubGhzSettingStateLoadFrequencyDefault;
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if(loading == SubGhzSettingStateLoadFrequencyDefault) {
|
|
|
- loading = SubGhzSettingStateOkLoad;
|
|
|
- } else {
|
|
|
- FURI_LOG_E(TAG, "Frequency default index missing");
|
|
|
+ if(flipper_format_read_uint32(fff_data_file, "default_frequency", &temp_data32, 1)) {
|
|
|
+ for
|
|
|
+ M_EACH(frequency, instance->frequencies, FrequencyList_t) {
|
|
|
+ *frequency &= FREQUENCY_MASK;
|
|
|
+ if(*frequency == temp_data32) {
|
|
|
+ *frequency |= FREQUENCY_FLAG_DEFAULT;
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
} while(false);
|
|
|
}
|
|
|
@@ -301,67 +320,56 @@ void subghz_setting_load(SubGhzSetting* instance, const char* file_path) {
|
|
|
flipper_format_free(fff_data_file);
|
|
|
furi_record_close("storage");
|
|
|
|
|
|
- if(loading != SubGhzSettingStateOkLoad) {
|
|
|
- switch(furi_hal_version_get_hw_region()) {
|
|
|
- case FuriHalVersionRegionEuRu:
|
|
|
- subghz_setting_load_default(
|
|
|
- instance,
|
|
|
- subghz_frequencies_region_eu_ru,
|
|
|
- subghz_hopper_frequencies_region_eu_ru,
|
|
|
- subghz_frequency_default_index_region_eu_ru);
|
|
|
- break;
|
|
|
- case FuriHalVersionRegionUsCaAu:
|
|
|
- subghz_setting_load_default(
|
|
|
- instance,
|
|
|
- subghz_frequencies_region_us_ca_au,
|
|
|
- subghz_hopper_frequencies_region_us_ca_au,
|
|
|
- subghz_frequency_default_index_region_us_ca_au);
|
|
|
- break;
|
|
|
- case FuriHalVersionRegionJp:
|
|
|
- subghz_setting_load_default(
|
|
|
- instance,
|
|
|
- subghz_frequencies_region_jp,
|
|
|
- subghz_hopper_frequencies_region_jp,
|
|
|
- subghz_frequency_default_index_region_jp);
|
|
|
- break;
|
|
|
-
|
|
|
- default:
|
|
|
- subghz_setting_load_default(
|
|
|
- instance,
|
|
|
- subghz_frequencies,
|
|
|
- subghz_hopper_frequencies,
|
|
|
- subghz_frequency_default_index);
|
|
|
- break;
|
|
|
- }
|
|
|
+ if(!FrequencyList_size(instance->frequencies) ||
|
|
|
+ !FrequencyList_size(instance->hopper_frequencies)) {
|
|
|
+ FURI_LOG_E(TAG, "Error loading user settings, loading default settings");
|
|
|
+ subghz_setting_load_default(instance);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
size_t subghz_setting_get_frequency_count(SubGhzSetting* instance) {
|
|
|
furi_assert(instance);
|
|
|
- return instance->frequencies_count;
|
|
|
+ return FrequencyList_size(instance->frequencies);
|
|
|
}
|
|
|
|
|
|
size_t subghz_setting_get_hopper_frequency_count(SubGhzSetting* instance) {
|
|
|
furi_assert(instance);
|
|
|
- return instance->hopper_frequencies_count;
|
|
|
+ return FrequencyList_size(instance->hopper_frequencies);
|
|
|
}
|
|
|
|
|
|
uint32_t subghz_setting_get_frequency(SubGhzSetting* instance, size_t idx) {
|
|
|
furi_assert(instance);
|
|
|
- return *FrequenciesList_get(instance->frequencies, idx);
|
|
|
+ uint32_t* ret = FrequencyList_get(instance->frequencies, idx);
|
|
|
+ if(ret) {
|
|
|
+ return (*ret) & FREQUENCY_MASK;
|
|
|
+ } else {
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
uint32_t subghz_setting_get_hopper_frequency(SubGhzSetting* instance, size_t idx) {
|
|
|
furi_assert(instance);
|
|
|
- return *FrequenciesList_get(instance->hopper_frequencies, idx);
|
|
|
+ uint32_t* ret = FrequencyList_get(instance->hopper_frequencies, idx);
|
|
|
+ if(ret) {
|
|
|
+ return *ret;
|
|
|
+ } else {
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
uint32_t subghz_setting_get_frequency_default_index(SubGhzSetting* instance) {
|
|
|
furi_assert(instance);
|
|
|
- return instance->frequency_default_index;
|
|
|
+ for(size_t i = 0; i < FrequencyList_size(instance->frequencies); i++) {
|
|
|
+ uint32_t frequency = *FrequencyList_get(instance->frequencies, i);
|
|
|
+ if(frequency & FREQUENCY_FLAG_DEFAULT) {
|
|
|
+ return i;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
uint32_t subghz_setting_get_default_frequency(SubGhzSetting* instance) {
|
|
|
furi_assert(instance);
|
|
|
- return *FrequenciesList_get(instance->frequencies, instance->frequency_default_index);
|
|
|
+ return subghz_setting_get_frequency(
|
|
|
+ instance, subghz_setting_get_frequency_default_index(instance));
|
|
|
}
|