* 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);
 | 
						|
    }
 | 
						|
}
 |