rpc_app.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. #include "cmsis_os2.h"
  2. #include "flipper.pb.h"
  3. #include "furi/record.h"
  4. #include "rpc_i.h"
  5. #include <furi.h>
  6. #include <loader/loader.h>
  7. #include "rpc_app.h"
  8. #define TAG "RpcSystemApp"
  9. #define APP_BUTTON_TIMEOUT 1000
  10. struct RpcAppSystem {
  11. RpcSession* session;
  12. RpcAppSystemCallback app_callback;
  13. void* app_context;
  14. osTimerId_t timer;
  15. };
  16. static void rpc_system_app_timer_callback(void* context) {
  17. furi_assert(context);
  18. RpcAppSystem* rpc_app = context;
  19. if(rpc_app->app_callback) {
  20. rpc_app->app_callback(RpcAppEventButtonRelease, NULL, rpc_app->app_context);
  21. }
  22. }
  23. static void rpc_system_app_start_process(const PB_Main* request, void* context) {
  24. furi_assert(request);
  25. furi_assert(context);
  26. furi_assert(request->which_content == PB_Main_app_start_request_tag);
  27. RpcAppSystem* rpc_app = context;
  28. RpcSession* session = rpc_app->session;
  29. furi_assert(session);
  30. char args_temp[16];
  31. FURI_LOG_D(TAG, "Start");
  32. PB_CommandStatus result = PB_CommandStatus_ERROR_APP_CANT_START;
  33. Loader* loader = furi_record_open("loader");
  34. const char* app_name = request->content.app_start_request.name;
  35. if(app_name) {
  36. const char* app_args = request->content.app_start_request.args;
  37. if(strcmp(app_args, "RPC") == 0) {
  38. // If app is being started in RPC mode - pass RPC context via args string
  39. snprintf(args_temp, 16, "RPC %08lX", (uint32_t)rpc_app);
  40. app_args = args_temp;
  41. }
  42. LoaderStatus status = loader_start(loader, app_name, app_args);
  43. if(status == LoaderStatusErrorAppStarted) {
  44. result = PB_CommandStatus_ERROR_APP_SYSTEM_LOCKED;
  45. } else if(status == LoaderStatusErrorInternal) {
  46. result = PB_CommandStatus_ERROR_APP_CANT_START;
  47. } else if(status == LoaderStatusErrorUnknownApp) {
  48. result = PB_CommandStatus_ERROR_INVALID_PARAMETERS;
  49. } else if(status == LoaderStatusOk) {
  50. result = PB_CommandStatus_OK;
  51. } else {
  52. furi_assert(0);
  53. }
  54. } else {
  55. result = PB_CommandStatus_ERROR_INVALID_PARAMETERS;
  56. }
  57. furi_record_close("loader");
  58. rpc_send_and_release_empty(session, request->command_id, result);
  59. }
  60. static void rpc_system_app_lock_status_process(const PB_Main* request, void* context) {
  61. furi_assert(request);
  62. furi_assert(context);
  63. furi_assert(request->which_content == PB_Main_app_lock_status_request_tag);
  64. RpcAppSystem* rpc_app = context;
  65. RpcSession* session = rpc_app->session;
  66. furi_assert(session);
  67. FURI_LOG_D(TAG, "LockStatus");
  68. Loader* loader = furi_record_open("loader");
  69. PB_Main response = {
  70. .has_next = false,
  71. .command_status = PB_CommandStatus_OK,
  72. .command_id = request->command_id,
  73. .which_content = PB_Main_app_lock_status_response_tag,
  74. };
  75. response.content.app_lock_status_response.locked = loader_is_locked(loader);
  76. furi_record_close("loader");
  77. rpc_send_and_release(session, &response);
  78. pb_release(&PB_Main_msg, &response);
  79. }
  80. static void rpc_system_app_exit(const PB_Main* request, void* context) {
  81. furi_assert(request);
  82. furi_assert(context);
  83. furi_assert(request->which_content == PB_Main_app_exit_request_tag);
  84. RpcAppSystem* rpc_app = context;
  85. RpcSession* session = rpc_app->session;
  86. furi_assert(session);
  87. PB_CommandStatus status;
  88. if(rpc_app->app_callback) {
  89. if(rpc_app->app_callback(RpcAppEventAppExit, NULL, rpc_app->app_context)) {
  90. status = PB_CommandStatus_OK;
  91. osTimerStop(rpc_app->timer);
  92. } else {
  93. status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
  94. }
  95. } else {
  96. status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
  97. }
  98. rpc_send_and_release_empty(session, request->command_id, status);
  99. }
  100. static void rpc_system_app_load_file(const PB_Main* request, void* context) {
  101. furi_assert(request);
  102. furi_assert(context);
  103. furi_assert(request->which_content == PB_Main_app_load_file_request_tag);
  104. RpcAppSystem* rpc_app = context;
  105. RpcSession* session = rpc_app->session;
  106. furi_assert(session);
  107. PB_CommandStatus status;
  108. if(rpc_app->app_callback) {
  109. const char* file_path = request->content.app_load_file_request.path;
  110. if(rpc_app->app_callback(RpcAppEventLoadFile, file_path, rpc_app->app_context)) {
  111. status = PB_CommandStatus_OK;
  112. } else {
  113. status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
  114. }
  115. } else {
  116. status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
  117. }
  118. rpc_send_and_release_empty(session, request->command_id, status);
  119. }
  120. static void rpc_system_app_button_press(const PB_Main* request, void* context) {
  121. furi_assert(request);
  122. furi_assert(context);
  123. furi_assert(request->which_content == PB_Main_app_button_press_request_tag);
  124. RpcAppSystem* rpc_app = context;
  125. RpcSession* session = rpc_app->session;
  126. furi_assert(session);
  127. PB_CommandStatus status;
  128. if(rpc_app->app_callback) {
  129. const char* args = request->content.app_button_press_request.args;
  130. if(rpc_app->app_callback(RpcAppEventButtonPress, args, rpc_app->app_context)) {
  131. status = PB_CommandStatus_OK;
  132. osTimerStart(rpc_app->timer, APP_BUTTON_TIMEOUT);
  133. } else {
  134. status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
  135. }
  136. } else {
  137. status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
  138. }
  139. rpc_send_and_release_empty(session, request->command_id, status);
  140. }
  141. static void rpc_system_app_button_release(const PB_Main* request, void* context) {
  142. furi_assert(request);
  143. furi_assert(request->which_content == PB_Main_app_button_release_request_tag);
  144. furi_assert(context);
  145. RpcAppSystem* rpc_app = context;
  146. RpcSession* session = rpc_app->session;
  147. furi_assert(session);
  148. PB_CommandStatus status;
  149. if(rpc_app->app_callback) {
  150. if(rpc_app->app_callback(RpcAppEventButtonRelease, NULL, rpc_app->app_context)) {
  151. status = PB_CommandStatus_OK;
  152. osTimerStop(rpc_app->timer);
  153. } else {
  154. status = PB_CommandStatus_ERROR_APP_CMD_ERROR;
  155. }
  156. } else {
  157. status = PB_CommandStatus_ERROR_APP_NOT_RUNNING;
  158. }
  159. rpc_send_and_release_empty(session, request->command_id, status);
  160. }
  161. void rpc_system_app_set_callback(RpcAppSystem* rpc_app, RpcAppSystemCallback callback, void* ctx) {
  162. furi_assert(rpc_app);
  163. rpc_app->app_callback = callback;
  164. rpc_app->app_context = ctx;
  165. }
  166. void* rpc_system_app_alloc(RpcSession* session) {
  167. furi_assert(session);
  168. RpcAppSystem* rpc_app = malloc(sizeof(RpcAppSystem));
  169. rpc_app->session = session;
  170. rpc_app->timer = osTimerNew(rpc_system_app_timer_callback, osTimerOnce, rpc_app, NULL);
  171. RpcHandler rpc_handler = {
  172. .message_handler = NULL,
  173. .decode_submessage = NULL,
  174. .context = rpc_app,
  175. };
  176. rpc_handler.message_handler = rpc_system_app_start_process;
  177. rpc_add_handler(session, PB_Main_app_start_request_tag, &rpc_handler);
  178. rpc_handler.message_handler = rpc_system_app_lock_status_process;
  179. rpc_add_handler(session, PB_Main_app_lock_status_request_tag, &rpc_handler);
  180. rpc_handler.message_handler = rpc_system_app_exit;
  181. rpc_add_handler(session, PB_Main_app_exit_request_tag, &rpc_handler);
  182. rpc_handler.message_handler = rpc_system_app_load_file;
  183. rpc_add_handler(session, PB_Main_app_load_file_request_tag, &rpc_handler);
  184. rpc_handler.message_handler = rpc_system_app_button_press;
  185. rpc_add_handler(session, PB_Main_app_button_press_request_tag, &rpc_handler);
  186. rpc_handler.message_handler = rpc_system_app_button_release;
  187. rpc_add_handler(session, PB_Main_app_button_release_request_tag, &rpc_handler);
  188. return rpc_app;
  189. }
  190. void rpc_system_app_free(void* context) {
  191. RpcAppSystem* rpc_app = context;
  192. RpcSession* session = rpc_app->session;
  193. furi_assert(session);
  194. osTimerDelete(rpc_app->timer);
  195. if(rpc_app->app_callback) {
  196. rpc_app->app_callback(RpcAppEventSessionClose, NULL, rpc_app->app_context);
  197. }
  198. free(rpc_app);
  199. }