mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2024-12-30 16:56:41 +03:00
53435579b3
* fbt, faploader: minimal app module implementation * faploader, libs: moved API hashtable core to flipper_application * example: compound api * lib: flipper_application: naming fixes, doxygen comments * fbt: changed `requires` manifest field behavior for app extensions * examples: refactored plugin apps; faploader: changed new API naming; fbt: changed PLUGIN app type meaning * loader: dropped support for debug apps & plugin menus * moved applications/plugins -> applications/external * Restored x bit on chiplist_convert.py * git: fixed free-dap submodule path * pvs: updated submodule paths * examples: example_advanced_plugins.c: removed potential memory leak on errors * examples: example_plugins: refined requires * fbt: not deploying app modules for debug/sample apps; extra validation for .PLUGIN-type apps * apps: removed cdefines for external apps * fbt: moved ext app path definition * fbt: reworked fap_dist handling; f18: synced api_symbols.csv * fbt: removed resources_paths for extapps * scripts: reworked storage * scripts: reworked runfap.py & selfupdate.py to use new api * wip: fal runner * fbt: moved file packaging into separate module * scripts: storage: fixes * scripts: storage: minor fixes for new api * fbt: changed internal artifact storage details for external apps * scripts: storage: additional fixes and better error reporting; examples: using APP_DATA_PATH() * fbt, scripts: reworked launch_app to deploy plugins; moved old runfap.py to distfap.py * fbt: extra check for plugins descriptors * fbt: additional checks in emitter * fbt: better info message on SDK rebuild * scripts: removed requirements.txt * loader: removed remnants of plugins & debug menus * post-review fixes
157 lines
5.0 KiB
C
157 lines
5.0 KiB
C
#include "weather_station_app_i.h"
|
|
|
|
#define TAG "WeatherStation"
|
|
#include <flipper_format/flipper_format_i.h>
|
|
|
|
void ws_preset_init(
|
|
void* context,
|
|
const char* preset_name,
|
|
uint32_t frequency,
|
|
uint8_t* preset_data,
|
|
size_t preset_data_size) {
|
|
furi_assert(context);
|
|
WeatherStationApp* app = context;
|
|
furi_string_set(app->txrx->preset->name, preset_name);
|
|
app->txrx->preset->frequency = frequency;
|
|
app->txrx->preset->data = preset_data;
|
|
app->txrx->preset->data_size = preset_data_size;
|
|
}
|
|
|
|
bool ws_set_preset(WeatherStationApp* app, const char* preset) {
|
|
if(!strcmp(preset, "FuriHalSubGhzPresetOok270Async")) {
|
|
furi_string_set(app->txrx->preset->name, "AM270");
|
|
} else if(!strcmp(preset, "FuriHalSubGhzPresetOok650Async")) {
|
|
furi_string_set(app->txrx->preset->name, "AM650");
|
|
} else if(!strcmp(preset, "FuriHalSubGhzPreset2FSKDev238Async")) {
|
|
furi_string_set(app->txrx->preset->name, "FM238");
|
|
} else if(!strcmp(preset, "FuriHalSubGhzPreset2FSKDev476Async")) {
|
|
furi_string_set(app->txrx->preset->name, "FM476");
|
|
} else if(!strcmp(preset, "FuriHalSubGhzPresetCustom")) {
|
|
furi_string_set(app->txrx->preset->name, "CUSTOM");
|
|
} else {
|
|
FURI_LOG_E(TAG, "Unknown preset");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void ws_get_frequency_modulation(
|
|
WeatherStationApp* app,
|
|
FuriString* frequency,
|
|
FuriString* modulation) {
|
|
furi_assert(app);
|
|
if(frequency != NULL) {
|
|
furi_string_printf(
|
|
frequency,
|
|
"%03ld.%02ld",
|
|
app->txrx->preset->frequency / 1000000 % 1000,
|
|
app->txrx->preset->frequency / 10000 % 100);
|
|
}
|
|
if(modulation != NULL) {
|
|
furi_string_printf(modulation, "%.2s", furi_string_get_cstr(app->txrx->preset->name));
|
|
}
|
|
}
|
|
|
|
void ws_begin(WeatherStationApp* app, uint8_t* preset_data) {
|
|
furi_assert(app);
|
|
UNUSED(preset_data);
|
|
furi_hal_subghz_reset();
|
|
furi_hal_subghz_idle();
|
|
furi_hal_subghz_load_custom_preset(preset_data);
|
|
furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
app->txrx->txrx_state = WSTxRxStateIDLE;
|
|
}
|
|
|
|
uint32_t ws_rx(WeatherStationApp* app, uint32_t frequency) {
|
|
furi_assert(app);
|
|
if(!furi_hal_subghz_is_frequency_valid(frequency)) {
|
|
furi_crash("WeatherStation: Incorrect RX frequency.");
|
|
}
|
|
furi_assert(
|
|
app->txrx->txrx_state != WSTxRxStateRx && app->txrx->txrx_state != WSTxRxStateSleep);
|
|
|
|
furi_hal_subghz_idle();
|
|
uint32_t value = furi_hal_subghz_set_frequency_and_path(frequency);
|
|
furi_hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
|
furi_hal_subghz_flush_rx();
|
|
furi_hal_subghz_rx();
|
|
|
|
furi_hal_subghz_start_async_rx(subghz_worker_rx_callback, app->txrx->worker);
|
|
subghz_worker_start(app->txrx->worker);
|
|
app->txrx->txrx_state = WSTxRxStateRx;
|
|
return value;
|
|
}
|
|
|
|
void ws_idle(WeatherStationApp* app) {
|
|
furi_assert(app);
|
|
furi_assert(app->txrx->txrx_state != WSTxRxStateSleep);
|
|
furi_hal_subghz_idle();
|
|
app->txrx->txrx_state = WSTxRxStateIDLE;
|
|
}
|
|
|
|
void ws_rx_end(WeatherStationApp* app) {
|
|
furi_assert(app);
|
|
furi_assert(app->txrx->txrx_state == WSTxRxStateRx);
|
|
if(subghz_worker_is_running(app->txrx->worker)) {
|
|
subghz_worker_stop(app->txrx->worker);
|
|
furi_hal_subghz_stop_async_rx();
|
|
}
|
|
furi_hal_subghz_idle();
|
|
app->txrx->txrx_state = WSTxRxStateIDLE;
|
|
}
|
|
|
|
void ws_sleep(WeatherStationApp* app) {
|
|
furi_assert(app);
|
|
furi_hal_subghz_sleep();
|
|
app->txrx->txrx_state = WSTxRxStateSleep;
|
|
}
|
|
|
|
void ws_hopper_update(WeatherStationApp* app) {
|
|
furi_assert(app);
|
|
|
|
switch(app->txrx->hopper_state) {
|
|
case WSHopperStateOFF:
|
|
case WSHopperStatePause:
|
|
return;
|
|
case WSHopperStateRSSITimeOut:
|
|
if(app->txrx->hopper_timeout != 0) {
|
|
app->txrx->hopper_timeout--;
|
|
return;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
float rssi = -127.0f;
|
|
if(app->txrx->hopper_state != WSHopperStateRSSITimeOut) {
|
|
// See RSSI Calculation timings in CC1101 17.3 RSSI
|
|
rssi = furi_hal_subghz_get_rssi();
|
|
|
|
// Stay if RSSI is high enough
|
|
if(rssi > -90.0f) {
|
|
app->txrx->hopper_timeout = 10;
|
|
app->txrx->hopper_state = WSHopperStateRSSITimeOut;
|
|
return;
|
|
}
|
|
} else {
|
|
app->txrx->hopper_state = WSHopperStateRunnig;
|
|
}
|
|
// Select next frequency
|
|
if(app->txrx->hopper_idx_frequency <
|
|
subghz_setting_get_hopper_frequency_count(app->setting) - 1) {
|
|
app->txrx->hopper_idx_frequency++;
|
|
} else {
|
|
app->txrx->hopper_idx_frequency = 0;
|
|
}
|
|
|
|
if(app->txrx->txrx_state == WSTxRxStateRx) {
|
|
ws_rx_end(app);
|
|
};
|
|
if(app->txrx->txrx_state == WSTxRxStateIDLE) {
|
|
subghz_receiver_reset(app->txrx->receiver);
|
|
app->txrx->preset->frequency =
|
|
subghz_setting_get_hopper_frequency(app->setting, app->txrx->hopper_idx_frequency);
|
|
ws_rx(app, app->txrx->preset->frequency);
|
|
}
|
|
}
|