app_switcher.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. /*
  2. * Copyright (c) 2019 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief app switch manager.
  9. */
  10. #include <os_common_api.h>
  11. #include <string.h>
  12. #include <mem_manager.h>
  13. #include <msg_manager.h>
  14. #include <app_manager.h>
  15. #include <hotplug_manager.h>
  16. #include "app_switch.h"
  17. #ifdef CONFIG_ESD_MANAGER
  18. #include "esd_manager.h"
  19. #endif
  20. #ifdef CONFIG_DVFS
  21. #include <dvfs.h>
  22. #endif
  23. #ifdef SYS_LOG_DOMAIN
  24. #undef SYS_LOG_DOMAIN
  25. #endif
  26. #define SYS_LOG_DOMAIN "app_switcher"
  27. struct app_switch_node_t {
  28. const char *app_name;
  29. uint8_t pluged;
  30. } app_switch_node_t;
  31. struct app_switch_ctx_t {
  32. uint8_t locked;
  33. uint8_t force_locked;
  34. uint8_t cur_index;
  35. uint8_t last_index;
  36. uint8_t app_num;
  37. uint8_t app_actived_num;
  38. struct app_switch_node_t app_list[MAX_APP_NUM];
  39. };
  40. OS_MUTEX_DEFINE(app_switch_mutex);
  41. static struct app_switch_ctx_t globle_switch_ctx;
  42. static inline struct app_switch_ctx_t *_app_switch_get_ctx(void)
  43. {
  44. return &globle_switch_ctx;
  45. }
  46. uint8_t *app_switch_get_app_name(uint8_t appid)
  47. {
  48. uint8_t *result = NULL;
  49. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  50. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  51. if (appid < switch_ctx->app_num) {
  52. struct app_switch_node_t *app_node = &switch_ctx->app_list[appid];
  53. result = (uint8_t *)app_node->app_name;
  54. }
  55. os_mutex_unlock(&app_switch_mutex);
  56. return result;
  57. }
  58. extern void recorder_service_app_switch_exit(void);
  59. extern void alarm_need_exit(void);
  60. int app_switch(void *app_name, uint32_t switch_mode, bool force_switch)
  61. {
  62. int ret = 0;
  63. int i = 0;
  64. #ifdef CONFIG_ESD_MANAGER
  65. uint8_t app_id = -1;
  66. #endif
  67. int target_app_index = -1;
  68. struct app_switch_node_t *app_node = NULL;
  69. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  70. #ifdef CONFIG_ACTS_DVFS_DYNAMIC_LEVEL
  71. dvfs_set_level(DVFS_LEVEL_HIGH_PERFORMANCE, "switch");
  72. #endif
  73. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  74. if (switch_ctx->force_locked) {
  75. SYS_LOG_INF("force locked %d \n", switch_ctx->force_locked);
  76. goto exit;
  77. }
  78. #ifdef CONFIG_RECORD_SERVICE
  79. recorder_service_app_switch_exit();
  80. #endif
  81. #ifdef CONFIG_ALARM_APP
  82. alarm_need_exit();
  83. #endif
  84. if (app_name
  85. && app_manager_get_current_app()
  86. && !strcmp(app_name, app_manager_get_current_app())) {
  87. SYS_LOG_INF("current app %s is you need\n", (char *)app_name);
  88. goto exit;
  89. }
  90. if (switch_ctx->locked && !force_switch) {
  91. SYS_LOG_INF("locked %d\n", switch_ctx->locked);
  92. goto exit;
  93. }
  94. if (switch_ctx->app_actived_num <= 1
  95. && switch_ctx->cur_index != 0xff
  96. && switch_ctx->app_list[switch_ctx->cur_index].pluged
  97. && !force_switch) {
  98. SYS_LOG_INF("app_actived_num %d ",
  99. switch_ctx->app_actived_num);
  100. SYS_LOG_INF("cur_index %d pluged %d",
  101. switch_ctx->cur_index,
  102. switch_ctx->app_list[switch_ctx->cur_index].pluged);
  103. goto exit;
  104. }
  105. next:
  106. if (!app_name) {
  107. if (switch_mode == APP_SWITCH_NEXT) {
  108. target_app_index = (switch_ctx->cur_index + 1)
  109. % switch_ctx->app_num;
  110. } else if (switch_mode == APP_SWITCH_PREV) {
  111. target_app_index = (switch_ctx->cur_index - 1 + switch_ctx->app_num)
  112. % switch_ctx->app_num;
  113. } else {
  114. target_app_index = switch_ctx->last_index;
  115. }
  116. } else {
  117. for (i = 0; i < switch_ctx->app_num; i++) {
  118. app_node = &switch_ctx->app_list[i];
  119. if (!memcmp(app_node->app_name, app_name, strlen(app_name))
  120. && (app_node->pluged || force_switch)) {
  121. target_app_index = i;
  122. }
  123. }
  124. }
  125. if (target_app_index == switch_ctx->cur_index
  126. || target_app_index < 0) {
  127. if (switch_ctx->cur_index < switch_ctx->app_num) {
  128. SYS_LOG_INF("currren app is %s",
  129. switch_ctx->app_list[switch_ctx->cur_index].app_name);
  130. }
  131. ret = 0xff;
  132. goto exit;
  133. }
  134. switch_ctx->last_index = switch_ctx->cur_index;
  135. switch_ctx->cur_index = target_app_index;
  136. app_node = &switch_ctx->app_list[switch_ctx->cur_index];
  137. if (!app_node->pluged && !force_switch) {
  138. /**if last is not exit ,go to prev */
  139. if (switch_mode == APP_SWITCH_LAST) {
  140. switch_mode = APP_SWITCH_PREV;
  141. }
  142. goto next;
  143. }
  144. SYS_LOG_INF("name:%s", app_node->app_name);
  145. ret = app_manager_active_app((char *)app_node->app_name);
  146. #ifdef CONFIG_ESD_MANAGER
  147. app_id = target_app_index;
  148. esd_manager_save_scene(TAG_APP_ID, &app_id, 1);
  149. #endif
  150. exit:
  151. os_mutex_unlock(&app_switch_mutex);
  152. #ifdef CONFIG_ACTS_DVFS_DYNAMIC_LEVEL
  153. dvfs_unset_level(DVFS_LEVEL_HIGH_PERFORMANCE, "switch");
  154. #endif
  155. return ret;
  156. }
  157. void app_switch_add_app(const char *app_name)
  158. {
  159. int i = 0;
  160. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  161. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  162. for (i = 0; i < switch_ctx->app_num; i++) {
  163. struct app_switch_node_t *app_node = &switch_ctx->app_list[i];
  164. if (!memcmp(app_node->app_name, app_name, strlen(app_name))
  165. && !app_node->pluged) {
  166. app_node->pluged = 1;
  167. switch_ctx->app_actived_num++;
  168. SYS_LOG_INF("name:%s plug in", app_node->app_name);
  169. }
  170. }
  171. os_mutex_unlock(&app_switch_mutex);
  172. }
  173. void app_switch_remove_app(const char *app_name)
  174. {
  175. int i = 0;
  176. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  177. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  178. for (i = 0; i < switch_ctx->app_num; i++) {
  179. struct app_switch_node_t *app_node = &switch_ctx->app_list[i];
  180. if (!memcmp(app_node->app_name, app_name, strlen(app_name))
  181. && app_node->pluged) {
  182. app_node->pluged = 0;
  183. switch_ctx->app_actived_num--;
  184. SYS_LOG_INF("name:%s plug out ", app_node->app_name);
  185. }
  186. }
  187. os_mutex_unlock(&app_switch_mutex);
  188. }
  189. void app_switch_lock(uint8_t reason)
  190. {
  191. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  192. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  193. switch_ctx->locked |= reason;
  194. os_mutex_unlock(&app_switch_mutex);
  195. }
  196. void app_switch_unlock(uint8_t reason)
  197. {
  198. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  199. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  200. switch_ctx->locked &= ~reason;
  201. os_mutex_unlock(&app_switch_mutex);
  202. }
  203. void app_switch_force_lock(uint8_t reason)
  204. {
  205. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  206. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  207. switch_ctx->force_locked |= reason;
  208. os_mutex_unlock(&app_switch_mutex);
  209. }
  210. void app_switch_force_unlock(uint8_t reason)
  211. {
  212. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  213. os_mutex_lock(&app_switch_mutex, OS_FOREVER);
  214. switch_ctx->force_locked &= ~reason;
  215. os_mutex_unlock(&app_switch_mutex);
  216. }
  217. int app_switch_init(const char **app_id_switch_list, int app_num)
  218. {
  219. int i = 0;
  220. struct app_switch_ctx_t *switch_ctx = _app_switch_get_ctx();
  221. memset(switch_ctx, 0, sizeof(struct app_switch_ctx_t));
  222. for (i = 0; i < app_num; i++) {
  223. struct app_switch_node_t *app_node = &switch_ctx->app_list[i];
  224. app_node->app_name = app_id_switch_list[i];
  225. app_node->pluged = 0;
  226. }
  227. #ifdef CONFIG_ASSERT
  228. __ASSERT(app_num <= MAX_APP_NUM,"app_num %d is large than MAX_APP_NUM %d ", app_num, MAX_APP_NUM);
  229. #endif
  230. switch_ctx->cur_index = 0xff;
  231. switch_ctx->last_index = 0xff;
  232. switch_ctx->app_num = app_num;
  233. switch_ctx->app_actived_num = 0;
  234. switch_ctx->locked = 0;
  235. switch_ctx->force_locked = 0;
  236. return 0;
  237. }