| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 |
- #include "cmsis_os2.h"
- #include "flipper.pb.h"
- #include "furi/record.h"
- #include "rpc_i.h"
- #include <furi.h>
- #include <loader/loader.h>
- #include "rpc_app.h"
- #define TAG "RpcSystemApp"
- #define APP_BUTTON_TIMEOUT 1000
- struct RpcAppSystem {
- RpcSession* session;
- RpcAppSystemCallback app_callback;
- void* app_context;
- osTimerId_t timer;
- };
- static void rpc_system_app_timer_callback(void* context) {
- furi_assert(context);
- RpcAppSystem* rpc_app = context;
- if(rpc_app->app_callback) {
- rpc_app->app_callback(RpcAppEventButtonRelease, NULL, rpc_app->app_context);
- }
- }
- static void rpc_system_app_start_process(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(context);
- furi_assert(request->which_content == PB_Main_app_start_request_tag);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- char args_temp[16];
- FURI_LOG_D(TAG, "Start");
- PB_CommandStatus result = PB_CommandStatus_ERROR_APP_CANT_START;
- Loader* loader = furi_record_open("loader");
- const char* app_name = request->content.app_start_request.name;
- if(app_name) {
- const char* app_args = request->content.app_start_request.args;
- if(strcmp(app_args, "RPC") == 0) {
- // If app is being started in RPC mode - pass RPC context via args string
- snprintf(args_temp, 16, "RPC %08lX", (uint32_t)rpc_app);
- app_args = args_temp;
- }
- LoaderStatus status = loader_start(loader, app_name, app_args);
- if(status == LoaderStatusErrorAppStarted) {
- result = PB_CommandStatus_ERROR_APP_SYSTEM_LOCKED;
- } else if(status == LoaderStatusErrorInternal) {
- result = PB_CommandStatus_ERROR_APP_CANT_START;
- } else if(status == LoaderStatusErrorUnknownApp) {
- result = PB_CommandStatus_ERROR_INVALID_PARAMETERS;
- } else if(status == LoaderStatusOk) {
- result = PB_CommandStatus_OK;
- } else {
- furi_assert(0);
- }
- } else {
- result = PB_CommandStatus_ERROR_INVALID_PARAMETERS;
- }
- furi_record_close("loader");
- rpc_send_and_release_empty(session, request->command_id, result);
- }
- static void rpc_system_app_lock_status_process(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(context);
- furi_assert(request->which_content == PB_Main_app_lock_status_request_tag);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- FURI_LOG_D(TAG, "LockStatus");
- Loader* loader = furi_record_open("loader");
- PB_Main response = {
- .has_next = false,
- .command_status = PB_CommandStatus_OK,
- .command_id = request->command_id,
- .which_content = PB_Main_app_lock_status_response_tag,
- };
- response.content.app_lock_status_response.locked = loader_is_locked(loader);
- furi_record_close("loader");
- rpc_send_and_release(session, &response);
- pb_release(&PB_Main_msg, &response);
- }
- static void rpc_system_app_exit(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(context);
- furi_assert(request->which_content == PB_Main_app_exit_request_tag);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- PB_CommandStatus status;
- if(rpc_app->app_callback) {
- if(rpc_app->app_callback(RpcAppEventAppExit, NULL, rpc_app->app_context)) {
- status = PB_CommandStatus_OK;
- osTimerStop(rpc_app->timer);
- } else {
- status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
- }
- } else {
- status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
- }
- rpc_send_and_release_empty(session, request->command_id, status);
- }
- static void rpc_system_app_load_file(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(context);
- furi_assert(request->which_content == PB_Main_app_load_file_request_tag);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- PB_CommandStatus status;
- if(rpc_app->app_callback) {
- const char* file_path = request->content.app_load_file_request.path;
- if(rpc_app->app_callback(RpcAppEventLoadFile, file_path, rpc_app->app_context)) {
- status = PB_CommandStatus_OK;
- } else {
- status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
- }
- } else {
- status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
- }
- rpc_send_and_release_empty(session, request->command_id, status);
- }
- static void rpc_system_app_button_press(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(context);
- furi_assert(request->which_content == PB_Main_app_button_press_request_tag);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- PB_CommandStatus status;
- if(rpc_app->app_callback) {
- const char* args = request->content.app_button_press_request.args;
- if(rpc_app->app_callback(RpcAppEventButtonPress, args, rpc_app->app_context)) {
- status = PB_CommandStatus_OK;
- osTimerStart(rpc_app->timer, APP_BUTTON_TIMEOUT);
- } else {
- status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
- }
- } else {
- status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
- }
- rpc_send_and_release_empty(session, request->command_id, status);
- }
- static void rpc_system_app_button_release(const PB_Main* request, void* context) {
- furi_assert(request);
- furi_assert(request->which_content == PB_Main_app_button_release_request_tag);
- furi_assert(context);
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- PB_CommandStatus status;
- if(rpc_app->app_callback) {
- if(rpc_app->app_callback(RpcAppEventButtonRelease, NULL, rpc_app->app_context)) {
- status = PB_CommandStatus_OK;
- osTimerStop(rpc_app->timer);
- } else {
- status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
- }
- } else {
- status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
- }
- rpc_send_and_release_empty(session, request->command_id, status);
- }
- void rpc_system_app_set_callback(RpcAppSystem* rpc_app, RpcAppSystemCallback callback, void* ctx) {
- furi_assert(rpc_app);
- rpc_app->app_callback = callback;
- rpc_app->app_context = ctx;
- }
- void* rpc_system_app_alloc(RpcSession* session) {
- furi_assert(session);
- RpcAppSystem* rpc_app = malloc(sizeof(RpcAppSystem));
- rpc_app->session = session;
- rpc_app->timer = osTimerNew(rpc_system_app_timer_callback, osTimerOnce, rpc_app, NULL);
- RpcHandler rpc_handler = {
- .message_handler = NULL,
- .decode_submessage = NULL,
- .context = rpc_app,
- };
- rpc_handler.message_handler = rpc_system_app_start_process;
- rpc_add_handler(session, PB_Main_app_start_request_tag, &rpc_handler);
- rpc_handler.message_handler = rpc_system_app_lock_status_process;
- rpc_add_handler(session, PB_Main_app_lock_status_request_tag, &rpc_handler);
- rpc_handler.message_handler = rpc_system_app_exit;
- rpc_add_handler(session, PB_Main_app_exit_request_tag, &rpc_handler);
- rpc_handler.message_handler = rpc_system_app_load_file;
- rpc_add_handler(session, PB_Main_app_load_file_request_tag, &rpc_handler);
- rpc_handler.message_handler = rpc_system_app_button_press;
- rpc_add_handler(session, PB_Main_app_button_press_request_tag, &rpc_handler);
- rpc_handler.message_handler = rpc_system_app_button_release;
- rpc_add_handler(session, PB_Main_app_button_release_request_tag, &rpc_handler);
- return rpc_app;
- }
- void rpc_system_app_free(void* context) {
- RpcAppSystem* rpc_app = context;
- RpcSession* session = rpc_app->session;
- furi_assert(session);
- osTimerDelete(rpc_app->timer);
- if(rpc_app->app_callback) {
- rpc_app->app_callback(RpcAppEventSessionClose, NULL, rpc_app->app_context);
- }
- free(rpc_app);
- }
|