shell.c 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <ctype.h>
  7. #include <stdlib.h>
  8. #include <sys/atomic.h>
  9. #include <shell/shell.h>
  10. #if defined(CONFIG_SHELL_BACKEND_DUMMY)
  11. #include <shell/shell_dummy.h>
  12. #endif
  13. #include "shell_ops.h"
  14. #include "shell_help.h"
  15. #include "shell_utils.h"
  16. #include "shell_vt100.h"
  17. #include "shell_wildcard.h"
  18. /* 2 == 1 char for cmd + 1 char for '\0' */
  19. #if (CONFIG_SHELL_CMD_BUFF_SIZE < 2)
  20. #error too small CONFIG_SHELL_CMD_BUFF_SIZE
  21. #endif
  22. #if (CONFIG_SHELL_PRINTF_BUFF_SIZE < 1)
  23. #error too small SHELL_PRINTF_BUFF_SIZE
  24. #endif
  25. #define SHELL_MSG_CMD_NOT_FOUND ": command not found"
  26. #define SHELL_MSG_BACKEND_NOT_ACTIVE \
  27. "WARNING: A print request was detected on not active shell backend.\n"
  28. #define SHELL_MSG_TOO_MANY_ARGS "Too many arguments in the command.\n"
  29. #define SHELL_INIT_OPTION_PRINTER (NULL)
  30. #define SHELL_THREAD_PRIORITY \
  31. COND_CODE_1(CONFIG_SHELL_THREAD_PRIORITY_OVERRIDE, \
  32. (CONFIG_SHELL_THREAD_PRIORITY), (K_LOWEST_APPLICATION_THREAD_PRIO))
  33. BUILD_ASSERT(SHELL_THREAD_PRIORITY >=
  34. K_HIGHEST_APPLICATION_THREAD_PRIO
  35. && SHELL_THREAD_PRIORITY <= K_LOWEST_APPLICATION_THREAD_PRIO,
  36. "Invalid range for thread priority");
  37. static inline void receive_state_change(const struct shell *shell,
  38. enum shell_receive_state state)
  39. {
  40. shell->ctx->receive_state = state;
  41. }
  42. static void cmd_buffer_clear(const struct shell *shell)
  43. {
  44. shell->ctx->cmd_buff[0] = '\0'; /* clear command buffer */
  45. shell->ctx->cmd_buff_pos = 0;
  46. shell->ctx->cmd_buff_len = 0;
  47. }
  48. static void shell_internal_help_print(const struct shell *shell)
  49. {
  50. if (!IS_ENABLED(CONFIG_SHELL_HELP)) {
  51. return;
  52. }
  53. z_shell_help_cmd_print(shell, &shell->ctx->active_cmd);
  54. z_shell_help_subcmd_print(shell, &shell->ctx->active_cmd,
  55. "Subcommands:\n");
  56. }
  57. /**
  58. * @brief Prints error message on wrong argument count.
  59. * Optionally, printing help on wrong argument count.
  60. *
  61. * @param[in] shell Pointer to the shell instance.
  62. * @param[in] arg_cnt_ok Flag indicating valid number of arguments.
  63. *
  64. * @return 0 if check passed
  65. * @return -EINVAL if wrong argument count
  66. */
  67. static int cmd_precheck(const struct shell *shell,
  68. bool arg_cnt_ok)
  69. {
  70. if (!arg_cnt_ok) {
  71. z_shell_fprintf(shell, SHELL_ERROR,
  72. "%s: wrong parameter count\n",
  73. shell->ctx->active_cmd.syntax);
  74. if (IS_ENABLED(CONFIG_SHELL_HELP_ON_WRONG_ARGUMENT_COUNT)) {
  75. shell_internal_help_print(shell);
  76. }
  77. return -EINVAL;
  78. }
  79. return 0;
  80. }
  81. static inline void state_set(const struct shell *shell, enum shell_state state)
  82. {
  83. shell->ctx->state = state;
  84. if (state == SHELL_STATE_ACTIVE && !shell->ctx->bypass) {
  85. cmd_buffer_clear(shell);
  86. if (z_flag_print_noinit_get(shell)) {
  87. z_shell_fprintf(shell, SHELL_WARNING, "%s",
  88. SHELL_MSG_BACKEND_NOT_ACTIVE);
  89. z_flag_print_noinit_set(shell, false);
  90. }
  91. z_shell_print_prompt_and_cmd(shell);
  92. }
  93. }
  94. static inline enum shell_state state_get(const struct shell *shell)
  95. {
  96. return shell->ctx->state;
  97. }
  98. static inline const struct shell_static_entry *
  99. selected_cmd_get(const struct shell *shell)
  100. {
  101. if (IS_ENABLED(CONFIG_SHELL_CMDS_SELECT)
  102. || (CONFIG_SHELL_CMD_ROOT[0] != 0)) {
  103. return shell->ctx->selected_cmd;
  104. }
  105. return NULL;
  106. }
  107. static void tab_item_print(const struct shell *shell, const char *option,
  108. uint16_t longest_option)
  109. {
  110. static const char *tab = " ";
  111. uint16_t columns;
  112. uint16_t diff;
  113. /* Function initialization has been requested. */
  114. if (option == NULL) {
  115. shell->ctx->vt100_ctx.printed_cmd = 0;
  116. return;
  117. }
  118. longest_option += z_shell_strlen(tab);
  119. columns = (shell->ctx->vt100_ctx.cons.terminal_wid
  120. - z_shell_strlen(tab)) / longest_option;
  121. diff = longest_option - z_shell_strlen(option);
  122. if (shell->ctx->vt100_ctx.printed_cmd++ % columns == 0U) {
  123. z_shell_fprintf(shell, SHELL_OPTION, "\n%s%s", tab, option);
  124. } else {
  125. z_shell_fprintf(shell, SHELL_OPTION, "%s", option);
  126. }
  127. z_shell_op_cursor_horiz_move(shell, diff);
  128. }
  129. static void history_init(const struct shell *shell)
  130. {
  131. if (!IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  132. return;
  133. }
  134. z_shell_history_init(shell->history);
  135. }
  136. static void history_purge(const struct shell *shell)
  137. {
  138. if (!IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  139. return;
  140. }
  141. z_shell_history_purge(shell->history);
  142. }
  143. static void history_mode_exit(const struct shell *shell)
  144. {
  145. if (!IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  146. return;
  147. }
  148. z_flag_history_exit_set(shell, false);
  149. z_shell_history_mode_exit(shell->history);
  150. }
  151. static void history_put(const struct shell *shell, uint8_t *line, size_t length)
  152. {
  153. if (!IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  154. return;
  155. }
  156. z_shell_history_put(shell->history, line, length);
  157. }
  158. static void history_handle(const struct shell *shell, bool up)
  159. {
  160. bool history_mode;
  161. uint16_t len;
  162. /*optional feature */
  163. if (!IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  164. return;
  165. }
  166. /* Checking if history process has been stopped */
  167. if (z_flag_history_exit_get(shell)) {
  168. z_flag_history_exit_set(shell, false);
  169. z_shell_history_mode_exit(shell->history);
  170. }
  171. /* Backup command if history is entered */
  172. if (!z_shell_history_active(shell->history)) {
  173. if (up) {
  174. uint16_t cmd_len = z_shell_strlen(shell->ctx->cmd_buff);
  175. if (cmd_len) {
  176. strcpy(shell->ctx->temp_buff,
  177. shell->ctx->cmd_buff);
  178. } else {
  179. shell->ctx->temp_buff[0] = '\0';
  180. }
  181. } else {
  182. /* Pressing 'down' not in history mode has no effect. */
  183. return;
  184. }
  185. }
  186. /* Start by checking if history is not empty. */
  187. history_mode = z_shell_history_get(shell->history, up,
  188. shell->ctx->cmd_buff, &len);
  189. /* On exiting history mode print backed up command. */
  190. if (!history_mode) {
  191. strcpy(shell->ctx->cmd_buff, shell->ctx->temp_buff);
  192. len = z_shell_strlen(shell->ctx->cmd_buff);
  193. }
  194. z_shell_op_cursor_home_move(shell);
  195. z_clear_eos(shell);
  196. z_shell_print_cmd(shell);
  197. shell->ctx->cmd_buff_pos = len;
  198. shell->ctx->cmd_buff_len = len;
  199. z_shell_op_cond_next_line(shell);
  200. }
  201. static inline uint16_t completion_space_get(const struct shell *shell)
  202. {
  203. uint16_t space = (CONFIG_SHELL_CMD_BUFF_SIZE - 1) -
  204. shell->ctx->cmd_buff_len;
  205. return space;
  206. }
  207. /* Prepare arguments and return number of space available for completion. */
  208. static bool tab_prepare(const struct shell *shell,
  209. const struct shell_static_entry **cmd,
  210. const char ***argv, size_t *argc,
  211. size_t *complete_arg_idx,
  212. struct shell_static_entry *d_entry)
  213. {
  214. uint16_t compl_space = completion_space_get(shell);
  215. size_t search_argc;
  216. if (compl_space == 0U) {
  217. return false;
  218. }
  219. /* Copy command from its beginning to cursor position. */
  220. memcpy(shell->ctx->temp_buff, shell->ctx->cmd_buff,
  221. shell->ctx->cmd_buff_pos);
  222. shell->ctx->temp_buff[shell->ctx->cmd_buff_pos] = '\0';
  223. /* Create argument list. */
  224. (void)z_shell_make_argv(argc, *argv, shell->ctx->temp_buff,
  225. CONFIG_SHELL_ARGC_MAX);
  226. if (*argc > CONFIG_SHELL_ARGC_MAX) {
  227. return false;
  228. }
  229. /* terminate arguments with NULL */
  230. (*argv)[*argc] = NULL;
  231. if ((IS_ENABLED(CONFIG_SHELL_CMDS_SELECT) || (CONFIG_SHELL_CMD_ROOT[0] != 0))
  232. && (*argc > 0) &&
  233. (strcmp("select", (*argv)[0]) == 0) &&
  234. !z_shell_in_select_mode(shell)) {
  235. *argv = *argv + 1;
  236. *argc = *argc - 1;
  237. }
  238. /* If last command is not completed (followed by space) it is treated
  239. * as uncompleted one.
  240. */
  241. int space = isspace((int)shell->ctx->cmd_buff[
  242. shell->ctx->cmd_buff_pos - 1]);
  243. /* root command completion */
  244. if ((*argc == 0) || ((space == 0) && (*argc == 1))) {
  245. *complete_arg_idx = Z_SHELL_CMD_ROOT_LVL;
  246. *cmd = selected_cmd_get(shell);
  247. return true;
  248. }
  249. search_argc = space ? *argc : *argc - 1;
  250. *cmd = z_shell_get_last_command(selected_cmd_get(shell), search_argc,
  251. *argv, complete_arg_idx, d_entry,
  252. false);
  253. /* if search_argc == 0 (empty command line) shell_get_last_command will
  254. * return NULL tab is allowed, otherwise not.
  255. */
  256. if ((*cmd == NULL) && (search_argc != 0)) {
  257. return false;
  258. }
  259. return true;
  260. }
  261. static inline bool is_completion_candidate(const char *candidate,
  262. const char *str, size_t len)
  263. {
  264. return (strncmp(candidate, str, len) == 0) ? true : false;
  265. }
  266. static void find_completion_candidates(const struct shell *shell,
  267. const struct shell_static_entry *cmd,
  268. const char *incompl_cmd,
  269. size_t *first_idx, size_t *cnt,
  270. uint16_t *longest)
  271. {
  272. const struct shell_static_entry *candidate;
  273. struct shell_static_entry dloc;
  274. size_t incompl_cmd_len;
  275. size_t idx = 0;
  276. incompl_cmd_len = z_shell_strlen(incompl_cmd);
  277. *longest = 0U;
  278. *cnt = 0;
  279. while ((candidate = z_shell_cmd_get(cmd, idx, &dloc)) != NULL) {
  280. bool is_candidate;
  281. is_candidate = is_completion_candidate(candidate->syntax,
  282. incompl_cmd, incompl_cmd_len);
  283. if (is_candidate) {
  284. *longest = Z_MAX(strlen(candidate->syntax), *longest);
  285. if (*cnt == 0) {
  286. *first_idx = idx;
  287. }
  288. (*cnt)++;
  289. }
  290. idx++;
  291. }
  292. }
  293. static void autocomplete(const struct shell *shell,
  294. const struct shell_static_entry *cmd,
  295. const char *arg,
  296. size_t subcmd_idx)
  297. {
  298. const struct shell_static_entry *match;
  299. uint16_t cmd_len;
  300. uint16_t arg_len = z_shell_strlen(arg);
  301. /* shell->ctx->active_cmd can be safely used outside of command context
  302. * to save stack
  303. */
  304. match = z_shell_cmd_get(cmd, subcmd_idx, &shell->ctx->active_cmd);
  305. __ASSERT_NO_MSG(match != NULL);
  306. cmd_len = z_shell_strlen(match->syntax);
  307. if (!IS_ENABLED(CONFIG_SHELL_TAB_AUTOCOMPLETION)) {
  308. /* Add a space if the Tab button is pressed when command is
  309. * complete.
  310. */
  311. if (cmd_len == arg_len) {
  312. z_shell_op_char_insert(shell, ' ');
  313. }
  314. return;
  315. }
  316. /* no exact match found */
  317. if (cmd_len != arg_len) {
  318. z_shell_op_completion_insert(shell,
  319. match->syntax + arg_len,
  320. cmd_len - arg_len);
  321. }
  322. /* Next character in the buffer is not 'space'. */
  323. if (!isspace((int) shell->ctx->cmd_buff[
  324. shell->ctx->cmd_buff_pos])) {
  325. if (z_flag_insert_mode_get(shell)) {
  326. z_flag_insert_mode_set(shell, false);
  327. z_shell_op_char_insert(shell, ' ');
  328. z_flag_insert_mode_set(shell, true);
  329. } else {
  330. z_shell_op_char_insert(shell, ' ');
  331. }
  332. } else {
  333. /* case:
  334. * | | -> cursor
  335. * cons_name $: valid_cmd valid_sub_cmd| |argument <tab>
  336. */
  337. z_shell_op_cursor_move(shell, 1);
  338. /* result:
  339. * cons_name $: valid_cmd valid_sub_cmd |a|rgument
  340. */
  341. }
  342. }
  343. static size_t str_common(const char *s1, const char *s2, size_t n)
  344. {
  345. size_t common = 0;
  346. while ((n > 0) && (*s1 == *s2) && (*s1 != '\0')) {
  347. s1++;
  348. s2++;
  349. n--;
  350. common++;
  351. }
  352. return common;
  353. }
  354. static void tab_options_print(const struct shell *shell,
  355. const struct shell_static_entry *cmd,
  356. const char *str, size_t first, size_t cnt,
  357. uint16_t longest)
  358. {
  359. const struct shell_static_entry *match;
  360. size_t str_len = z_shell_strlen(str);
  361. size_t idx = first;
  362. /* Printing all matching commands (options). */
  363. tab_item_print(shell, SHELL_INIT_OPTION_PRINTER, longest);
  364. while (cnt) {
  365. /* shell->ctx->active_cmd can be safely used outside of command
  366. * context to save stack
  367. */
  368. match = z_shell_cmd_get(cmd, idx, &shell->ctx->active_cmd);
  369. __ASSERT_NO_MSG(match != NULL);
  370. idx++;
  371. if (str && match->syntax &&
  372. !is_completion_candidate(match->syntax, str, str_len)) {
  373. continue;
  374. }
  375. tab_item_print(shell, match->syntax, longest);
  376. cnt--;
  377. }
  378. z_cursor_next_line_move(shell);
  379. z_shell_print_prompt_and_cmd(shell);
  380. }
  381. static uint16_t common_beginning_find(const struct shell *shell,
  382. const struct shell_static_entry *cmd,
  383. const char **str,
  384. size_t first, size_t cnt, uint16_t arg_len)
  385. {
  386. struct shell_static_entry dynamic_entry;
  387. const struct shell_static_entry *match;
  388. uint16_t common = UINT16_MAX;
  389. size_t idx = first + 1;
  390. __ASSERT_NO_MSG(cnt > 1);
  391. match = z_shell_cmd_get(cmd, first, &dynamic_entry);
  392. __ASSERT_NO_MSG(match);
  393. strncpy(shell->ctx->temp_buff, match->syntax,
  394. sizeof(shell->ctx->temp_buff) - 1);
  395. *str = match->syntax;
  396. while (cnt > 1) {
  397. struct shell_static_entry dynamic_entry2;
  398. const struct shell_static_entry *match2;
  399. int curr_common;
  400. match2 = z_shell_cmd_get(cmd, idx++, &dynamic_entry2);
  401. if (match2 == NULL) {
  402. break;
  403. }
  404. curr_common = str_common(shell->ctx->temp_buff, match2->syntax,
  405. UINT16_MAX);
  406. if ((arg_len == 0U) || (curr_common >= arg_len)) {
  407. --cnt;
  408. common = (curr_common < common) ? curr_common : common;
  409. }
  410. }
  411. return common;
  412. }
  413. static void partial_autocomplete(const struct shell *shell,
  414. const struct shell_static_entry *cmd,
  415. const char *arg,
  416. size_t first, size_t cnt)
  417. {
  418. const char *completion;
  419. uint16_t arg_len = z_shell_strlen(arg);
  420. uint16_t common = common_beginning_find(shell, cmd, &completion, first,
  421. cnt, arg_len);
  422. if (!IS_ENABLED(CONFIG_SHELL_TAB_AUTOCOMPLETION)) {
  423. return;
  424. }
  425. if (common) {
  426. z_shell_op_completion_insert(shell, &completion[arg_len],
  427. common - arg_len);
  428. }
  429. }
  430. static int exec_cmd(const struct shell *shell, size_t argc, const char **argv,
  431. const struct shell_static_entry *help_entry)
  432. {
  433. int ret_val = 0;
  434. if (shell->ctx->active_cmd.handler == NULL) {
  435. if ((help_entry != NULL) && IS_ENABLED(CONFIG_SHELL_HELP)) {
  436. if (help_entry->help == NULL) {
  437. return -ENOEXEC;
  438. }
  439. if (help_entry->help != shell->ctx->active_cmd.help) {
  440. shell->ctx->active_cmd = *help_entry;
  441. }
  442. shell_internal_help_print(shell);
  443. return SHELL_CMD_HELP_PRINTED;
  444. } else {
  445. z_shell_fprintf(shell, SHELL_ERROR,
  446. SHELL_MSG_SPECIFY_SUBCOMMAND);
  447. return -ENOEXEC;
  448. }
  449. }
  450. if (shell->ctx->active_cmd.args.mandatory) {
  451. uint32_t mand = shell->ctx->active_cmd.args.mandatory;
  452. uint8_t opt8 = shell->ctx->active_cmd.args.optional;
  453. uint32_t opt = (opt8 == SHELL_OPT_ARG_CHECK_SKIP) ?
  454. UINT16_MAX : opt8;
  455. bool in_range = (argc >= mand) && (argc <= (mand + opt));
  456. /* Check if argc is within allowed range */
  457. ret_val = cmd_precheck(shell, in_range);
  458. }
  459. if (!ret_val) {
  460. #if CONFIG_SHELL_GETOPT
  461. z_shell_getopt_init(&shell->ctx->getopt_state);
  462. #endif
  463. z_flag_cmd_ctx_set(shell, true);
  464. /* Unlock thread mutex in case command would like to borrow
  465. * shell context to other thread to avoid mutex deadlock.
  466. */
  467. k_mutex_unlock(&shell->ctx->wr_mtx);
  468. ret_val = shell->ctx->active_cmd.handler(shell, argc,
  469. (char **)argv);
  470. /* Bring back mutex to shell thread. */
  471. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  472. z_flag_cmd_ctx_set(shell, false);
  473. }
  474. return ret_val;
  475. }
  476. static void active_cmd_prepare(const struct shell_static_entry *entry,
  477. struct shell_static_entry *active_cmd,
  478. struct shell_static_entry *help_entry,
  479. size_t *lvl, size_t *handler_lvl,
  480. size_t *args_left)
  481. {
  482. if (entry->handler) {
  483. *handler_lvl = *lvl;
  484. *active_cmd = *entry;
  485. if ((entry->subcmd == NULL)
  486. && entry->args.optional == SHELL_OPT_ARG_RAW) {
  487. *args_left = entry->args.mandatory - 1;
  488. *lvl = *lvl + 1;
  489. }
  490. }
  491. if (entry->help) {
  492. *help_entry = *entry;
  493. }
  494. }
  495. static bool wildcard_check_report(const struct shell *shell, bool found,
  496. const struct shell_static_entry *entry)
  497. {
  498. /* An error occurred, fnmatch argument cannot be followed by argument
  499. * with a handler to avoid multiple function calls.
  500. */
  501. if (IS_ENABLED(CONFIG_SHELL_WILDCARD) && found && entry->handler) {
  502. z_shell_op_cursor_end_move(shell);
  503. z_shell_op_cond_next_line(shell);
  504. z_shell_fprintf(shell, SHELL_ERROR,
  505. "Error: requested multiple function executions\n");
  506. return false;
  507. }
  508. return true;
  509. }
  510. /* Function is analyzing the command buffer to find matching commands. Next, it
  511. * invokes the last recognized command which has a handler and passes the rest
  512. * of command buffer as arguments.
  513. *
  514. * By default command buffer is parsed and spaces are treated by arguments
  515. * separators. Complex arguments are provided in quotation marks with quotation
  516. * marks escaped within the argument. Argument parser is removing quotation
  517. * marks at argument boundary as well as escape characters within the argument.
  518. * However, it is possible to indicate that command shall treat remaining part
  519. * of command buffer as the last argument without parsing. This can be used for
  520. * commands which expects whole command buffer to be passed directly to
  521. * the command handler without any preprocessing.
  522. * Because of that feature, command buffer is processed argument by argument and
  523. * decision on further processing is based on currently processed command.
  524. */
  525. static int execute(const struct shell *shell)
  526. {
  527. struct shell_static_entry dloc; /* Memory for dynamic commands. */
  528. const char *argv[CONFIG_SHELL_ARGC_MAX + 1]; /* +1 reserved for NULL */
  529. const struct shell_static_entry *parent = selected_cmd_get(shell);
  530. const struct shell_static_entry *entry = NULL;
  531. struct shell_static_entry help_entry;
  532. size_t cmd_lvl = 0;
  533. size_t cmd_with_handler_lvl = 0;
  534. bool wildcard_found = false;
  535. size_t argc = 0, args_left = SIZE_MAX;
  536. char quote;
  537. const char **argvp;
  538. char *cmd_buf = shell->ctx->cmd_buff;
  539. bool has_last_handler = false;
  540. z_shell_op_cursor_end_move(shell);
  541. if (!z_shell_cursor_in_empty_line(shell)) {
  542. z_cursor_next_line_move(shell);
  543. }
  544. memset(&shell->ctx->active_cmd, 0, sizeof(shell->ctx->active_cmd));
  545. if (IS_ENABLED(CONFIG_SHELL_HISTORY)) {
  546. z_shell_cmd_trim(shell);
  547. history_put(shell, shell->ctx->cmd_buff,
  548. shell->ctx->cmd_buff_len);
  549. }
  550. if (IS_ENABLED(CONFIG_SHELL_WILDCARD)) {
  551. z_shell_wildcard_prepare(shell);
  552. }
  553. /* Parent present means we are in select mode. */
  554. if (parent != NULL) {
  555. argv[0] = parent->syntax;
  556. argv[1] = cmd_buf;
  557. argvp = &argv[1];
  558. active_cmd_prepare(parent, &shell->ctx->active_cmd, &help_entry,
  559. &cmd_lvl, &cmd_with_handler_lvl, &args_left);
  560. cmd_lvl++;
  561. } else {
  562. help_entry.help = NULL;
  563. argvp = &argv[0];
  564. }
  565. /* Below loop is analyzing subcommands of found root command. */
  566. while ((argc != 1) && (cmd_lvl < CONFIG_SHELL_ARGC_MAX)
  567. && args_left > 0) {
  568. quote = z_shell_make_argv(&argc, argvp, cmd_buf, 2);
  569. cmd_buf = (char *)argvp[1];
  570. if (argc == 0) {
  571. return -ENOEXEC;
  572. } else if ((argc == 1) && (quote != 0)) {
  573. z_shell_fprintf(shell, SHELL_ERROR,
  574. "not terminated: %c\n", quote);
  575. return -ENOEXEC;
  576. }
  577. if (IS_ENABLED(CONFIG_SHELL_HELP) && (cmd_lvl > 0) &&
  578. z_shell_help_request(argvp[0])) {
  579. /* Command called with help option so it makes no sense
  580. * to search deeper commands.
  581. */
  582. if (help_entry.help) {
  583. shell->ctx->active_cmd = help_entry;
  584. shell_internal_help_print(shell);
  585. return SHELL_CMD_HELP_PRINTED;
  586. }
  587. z_shell_fprintf(shell, SHELL_ERROR,
  588. SHELL_MSG_SPECIFY_SUBCOMMAND);
  589. return -ENOEXEC;
  590. }
  591. if (IS_ENABLED(CONFIG_SHELL_WILDCARD) && (cmd_lvl > 0)) {
  592. enum shell_wildcard_status status;
  593. status = z_shell_wildcard_process(shell, entry,
  594. argvp[0]);
  595. /* Wildcard character found but there is no matching
  596. * command.
  597. */
  598. if (status == SHELL_WILDCARD_CMD_NO_MATCH_FOUND) {
  599. break;
  600. }
  601. /* Wildcard character was not found function can process
  602. * argument.
  603. */
  604. if (status != SHELL_WILDCARD_NOT_FOUND) {
  605. ++cmd_lvl;
  606. wildcard_found = true;
  607. continue;
  608. }
  609. }
  610. if (has_last_handler == false) {
  611. entry = z_shell_find_cmd(parent, argvp[0], &dloc);
  612. }
  613. argvp++;
  614. args_left--;
  615. if (entry) {
  616. if (wildcard_check_report(shell, wildcard_found, entry)
  617. == false) {
  618. return -ENOEXEC;
  619. }
  620. active_cmd_prepare(entry, &shell->ctx->active_cmd,
  621. &help_entry, &cmd_lvl,
  622. &cmd_with_handler_lvl, &args_left);
  623. parent = entry;
  624. } else {
  625. if (cmd_lvl == 0 &&
  626. (!z_shell_in_select_mode(shell) ||
  627. shell->ctx->selected_cmd->handler == NULL)) {
  628. z_shell_fprintf(shell, SHELL_ERROR,
  629. "%s%s\n", argv[0],
  630. SHELL_MSG_CMD_NOT_FOUND);
  631. }
  632. /* last handler found - no need to search commands in
  633. * the next iteration.
  634. */
  635. has_last_handler = true;
  636. }
  637. if (args_left || (argc == 2)) {
  638. cmd_lvl++;
  639. }
  640. }
  641. if ((cmd_lvl >= CONFIG_SHELL_ARGC_MAX) && (argc == 2)) {
  642. /* argc == 2 indicates that when command string was parsed
  643. * there was more characters remaining. It means that number of
  644. * arguments exceeds the limit.
  645. */
  646. z_shell_fprintf(shell, SHELL_ERROR, "%s\n",
  647. SHELL_MSG_TOO_MANY_ARGS);
  648. return -ENOEXEC;
  649. }
  650. if (IS_ENABLED(CONFIG_SHELL_WILDCARD) && wildcard_found) {
  651. z_shell_wildcard_finalize(shell);
  652. /* cmd_buffer has been overwritten by function finalize function
  653. * with all expanded commands. Hence shell_make_argv needs to
  654. * be called again.
  655. */
  656. (void)z_shell_make_argv(&cmd_lvl,
  657. &argv[selected_cmd_get(shell) ? 1 : 0],
  658. shell->ctx->cmd_buff,
  659. CONFIG_SHELL_ARGC_MAX);
  660. if (selected_cmd_get(shell)) {
  661. /* Apart from what is in the command buffer, there is
  662. * a selected command.
  663. */
  664. cmd_lvl++;
  665. }
  666. }
  667. /* terminate arguments with NULL */
  668. argv[cmd_lvl] = NULL;
  669. /* Executing the deepest found handler. */
  670. return exec_cmd(shell, cmd_lvl - cmd_with_handler_lvl,
  671. &argv[cmd_with_handler_lvl], &help_entry);
  672. }
  673. static void tab_handle(const struct shell *shell)
  674. {
  675. const char *__argv[CONFIG_SHELL_ARGC_MAX + 1];
  676. /* d_entry - placeholder for dynamic command */
  677. struct shell_static_entry d_entry;
  678. const struct shell_static_entry *cmd;
  679. const char **argv = __argv;
  680. size_t first = 0;
  681. size_t arg_idx;
  682. uint16_t longest;
  683. size_t argc;
  684. size_t cnt;
  685. bool tab_possible = tab_prepare(shell, &cmd, &argv, &argc, &arg_idx,
  686. &d_entry);
  687. if (tab_possible == false) {
  688. return;
  689. }
  690. find_completion_candidates(shell, cmd, argv[arg_idx], &first, &cnt,
  691. &longest);
  692. if (cnt == 1) {
  693. /* Autocompletion.*/
  694. autocomplete(shell, cmd, argv[arg_idx], first);
  695. } else if (cnt > 1) {
  696. tab_options_print(shell, cmd, argv[arg_idx], first, cnt,
  697. longest);
  698. partial_autocomplete(shell, cmd, argv[arg_idx], first, cnt);
  699. }
  700. }
  701. static void alt_metakeys_handle(const struct shell *shell, char data)
  702. {
  703. /* Optional feature */
  704. if (!IS_ENABLED(CONFIG_SHELL_METAKEYS)) {
  705. return;
  706. }
  707. if (data == SHELL_VT100_ASCII_ALT_B) {
  708. z_shell_op_cursor_word_move(shell, -1);
  709. } else if (data == SHELL_VT100_ASCII_ALT_F) {
  710. z_shell_op_cursor_word_move(shell, 1);
  711. } else if (data == SHELL_VT100_ASCII_ALT_R &&
  712. IS_ENABLED(CONFIG_SHELL_CMDS_SELECT)) {
  713. if (selected_cmd_get(shell) != NULL) {
  714. z_shell_cmd_line_erase(shell);
  715. z_shell_fprintf(shell, SHELL_WARNING,
  716. "Restored default root commands\n");
  717. if (CONFIG_SHELL_CMD_ROOT[0]) {
  718. shell->ctx->selected_cmd = root_cmd_find(CONFIG_SHELL_CMD_ROOT);
  719. } else {
  720. shell->ctx->selected_cmd = NULL;
  721. }
  722. z_shell_print_prompt_and_cmd(shell);
  723. }
  724. }
  725. }
  726. static void ctrl_metakeys_handle(const struct shell *shell, char data)
  727. {
  728. /* Optional feature */
  729. if (!IS_ENABLED(CONFIG_SHELL_METAKEYS)) {
  730. return;
  731. }
  732. switch (data) {
  733. case SHELL_VT100_ASCII_CTRL_A: /* CTRL + A */
  734. z_shell_op_cursor_home_move(shell);
  735. break;
  736. case SHELL_VT100_ASCII_CTRL_B: /* CTRL + B */
  737. z_shell_op_left_arrow(shell);
  738. break;
  739. case SHELL_VT100_ASCII_CTRL_C: /* CTRL + C */
  740. z_shell_op_cursor_end_move(shell);
  741. if (!z_shell_cursor_in_empty_line(shell)) {
  742. z_cursor_next_line_move(shell);
  743. }
  744. z_flag_history_exit_set(shell, true);
  745. state_set(shell, SHELL_STATE_ACTIVE);
  746. break;
  747. case SHELL_VT100_ASCII_CTRL_D: /* CTRL + D */
  748. z_shell_op_char_delete(shell);
  749. break;
  750. case SHELL_VT100_ASCII_CTRL_E: /* CTRL + E */
  751. z_shell_op_cursor_end_move(shell);
  752. break;
  753. case SHELL_VT100_ASCII_CTRL_F: /* CTRL + F */
  754. z_shell_op_right_arrow(shell);
  755. break;
  756. case SHELL_VT100_ASCII_CTRL_K: /* CTRL + K */
  757. z_shell_op_delete_from_cursor(shell);
  758. break;
  759. case SHELL_VT100_ASCII_CTRL_L: /* CTRL + L */
  760. Z_SHELL_VT100_CMD(shell, SHELL_VT100_CURSORHOME);
  761. Z_SHELL_VT100_CMD(shell, SHELL_VT100_CLEARSCREEN);
  762. z_shell_print_prompt_and_cmd(shell);
  763. break;
  764. case SHELL_VT100_ASCII_CTRL_N: /* CTRL + N */
  765. history_handle(shell, false);
  766. break;
  767. case SHELL_VT100_ASCII_CTRL_P: /* CTRL + P */
  768. history_handle(shell, true);
  769. break;
  770. case SHELL_VT100_ASCII_CTRL_U: /* CTRL + U */
  771. z_shell_op_cursor_home_move(shell);
  772. cmd_buffer_clear(shell);
  773. z_flag_history_exit_set(shell, true);
  774. z_clear_eos(shell);
  775. break;
  776. case SHELL_VT100_ASCII_CTRL_W: /* CTRL + W */
  777. z_shell_op_word_remove(shell);
  778. z_flag_history_exit_set(shell, true);
  779. break;
  780. default:
  781. break;
  782. }
  783. }
  784. /* Functions returns true if new line character shall be processed */
  785. static bool process_nl(const struct shell *shell, uint8_t data)
  786. {
  787. if ((data != '\r') && (data != '\n')) {
  788. z_flag_last_nl_set(shell, 0);
  789. return false;
  790. }
  791. if ((z_flag_last_nl_get(shell) == 0U) ||
  792. (data == z_flag_last_nl_get(shell))) {
  793. z_flag_last_nl_set(shell, data);
  794. return true;
  795. }
  796. return false;
  797. }
  798. #define SHELL_ASCII_MAX_CHAR (127u)
  799. static inline int ascii_filter(const char data)
  800. {
  801. return (uint8_t) data > SHELL_ASCII_MAX_CHAR ? -EINVAL : 0;
  802. }
  803. static void state_collect(const struct shell *shell)
  804. {
  805. size_t count = 0;
  806. char data;
  807. while (true) {
  808. shell_bypass_cb_t bypass = shell->ctx->bypass;
  809. if (bypass) {
  810. uint8_t buf[16];
  811. (void)shell->iface->api->read(shell->iface, buf,
  812. sizeof(buf), &count);
  813. if (count) {
  814. bypass(shell, buf, count);
  815. /* Check if bypass mode ended. */
  816. if (!(volatile shell_bypass_cb_t *)shell->ctx->bypass) {
  817. state_set(shell, SHELL_STATE_ACTIVE);
  818. } else {
  819. continue;
  820. }
  821. }
  822. return;
  823. }
  824. (void)shell->iface->api->read(shell->iface, &data,
  825. sizeof(data), &count);
  826. if (count == 0) {
  827. return;
  828. }
  829. if (ascii_filter(data) != 0) {
  830. continue;
  831. }
  832. switch (shell->ctx->receive_state) {
  833. case SHELL_RECEIVE_DEFAULT:
  834. if (process_nl(shell, data)) {
  835. if (!shell->ctx->cmd_buff_len) {
  836. history_mode_exit(shell);
  837. z_cursor_next_line_move(shell);
  838. } else {
  839. /* Command execution */
  840. (void)execute(shell);
  841. }
  842. /* Function responsible for printing prompt
  843. * on received NL.
  844. */
  845. state_set(shell, SHELL_STATE_ACTIVE);
  846. continue;
  847. }
  848. switch (data) {
  849. case SHELL_VT100_ASCII_ESC: /* ESCAPE */
  850. receive_state_change(shell, SHELL_RECEIVE_ESC);
  851. break;
  852. case '\0':
  853. break;
  854. case '\t': /* TAB */
  855. if (z_flag_echo_get(shell) &&
  856. IS_ENABLED(CONFIG_SHELL_TAB)) {
  857. /* If the Tab key is pressed, "history
  858. * mode" must be terminated because
  859. * tab and history handlers are sharing
  860. * the same array: temp_buff.
  861. */
  862. z_flag_history_exit_set(shell, true);
  863. tab_handle(shell);
  864. }
  865. break;
  866. case SHELL_VT100_ASCII_BSPACE: /* BACKSPACE */
  867. if (z_flag_echo_get(shell)) {
  868. z_flag_history_exit_set(shell, true);
  869. z_shell_op_char_backspace(shell);
  870. }
  871. break;
  872. case SHELL_VT100_ASCII_DEL: /* DELETE */
  873. if (z_flag_echo_get(shell)) {
  874. z_flag_history_exit_set(shell, true);
  875. if (z_flag_mode_delete_get(shell)) {
  876. z_shell_op_char_backspace(shell);
  877. } else {
  878. z_shell_op_char_delete(shell);
  879. }
  880. }
  881. break;
  882. default:
  883. if (isprint((int) data)) {
  884. z_flag_history_exit_set(shell, true);
  885. z_shell_op_char_insert(shell, data);
  886. } else if (z_flag_echo_get(shell)) {
  887. ctrl_metakeys_handle(shell, data);
  888. }
  889. break;
  890. }
  891. break;
  892. case SHELL_RECEIVE_ESC:
  893. if (data == '[') {
  894. receive_state_change(shell,
  895. SHELL_RECEIVE_ESC_SEQ);
  896. break;
  897. } else if (z_flag_echo_get(shell)) {
  898. alt_metakeys_handle(shell, data);
  899. }
  900. receive_state_change(shell, SHELL_RECEIVE_DEFAULT);
  901. break;
  902. case SHELL_RECEIVE_ESC_SEQ:
  903. receive_state_change(shell, SHELL_RECEIVE_DEFAULT);
  904. if (!z_flag_echo_get(shell)) {
  905. continue;
  906. }
  907. switch (data) {
  908. case 'A': /* UP arrow */
  909. history_handle(shell, true);
  910. break;
  911. case 'B': /* DOWN arrow */
  912. history_handle(shell, false);
  913. break;
  914. case 'C': /* RIGHT arrow */
  915. z_shell_op_right_arrow(shell);
  916. break;
  917. case 'D': /* LEFT arrow */
  918. z_shell_op_left_arrow(shell);
  919. break;
  920. case '4': /* END Button in ESC[n~ mode */
  921. receive_state_change(shell,
  922. SHELL_RECEIVE_TILDE_EXP);
  923. __fallthrough;
  924. case 'F': /* END Button in VT100 mode */
  925. z_shell_op_cursor_end_move(shell);
  926. break;
  927. case '1': /* HOME Button in ESC[n~ mode */
  928. receive_state_change(shell,
  929. SHELL_RECEIVE_TILDE_EXP);
  930. __fallthrough;
  931. case 'H': /* HOME Button in VT100 mode */
  932. z_shell_op_cursor_home_move(shell);
  933. break;
  934. case '2': /* INSERT Button in ESC[n~ mode */
  935. receive_state_change(shell,
  936. SHELL_RECEIVE_TILDE_EXP);
  937. __fallthrough;
  938. case 'L': {/* INSERT Button in VT100 mode */
  939. bool status = z_flag_insert_mode_get(shell);
  940. z_flag_insert_mode_set(shell, !status);
  941. break;
  942. }
  943. case '3':/* DELETE Button in ESC[n~ mode */
  944. receive_state_change(shell,
  945. SHELL_RECEIVE_TILDE_EXP);
  946. if (z_flag_echo_get(shell)) {
  947. z_shell_op_char_delete(shell);
  948. }
  949. break;
  950. default:
  951. break;
  952. }
  953. break;
  954. case SHELL_RECEIVE_TILDE_EXP:
  955. receive_state_change(shell, SHELL_RECEIVE_DEFAULT);
  956. break;
  957. default:
  958. receive_state_change(shell, SHELL_RECEIVE_DEFAULT);
  959. break;
  960. }
  961. }
  962. z_transport_buffer_flush(shell);
  963. }
  964. static void transport_evt_handler(enum shell_transport_evt evt_type, void *ctx)
  965. {
  966. struct shell *shell = (struct shell *)ctx;
  967. struct k_poll_signal *signal;
  968. signal = (evt_type == SHELL_TRANSPORT_EVT_RX_RDY) ?
  969. &shell->ctx->signals[SHELL_SIGNAL_RXRDY] :
  970. &shell->ctx->signals[SHELL_SIGNAL_TXDONE];
  971. k_poll_signal_raise(signal, 0);
  972. }
  973. static void shell_log_process(const struct shell *shell)
  974. {
  975. bool processed = false;
  976. int signaled = 0;
  977. int result;
  978. do {
  979. if (!IS_ENABLED(CONFIG_LOG_IMMEDIATE)) {
  980. z_shell_cmd_line_erase(shell);
  981. processed = z_shell_log_backend_process(
  982. shell->log_backend);
  983. }
  984. struct k_poll_signal *signal =
  985. &shell->ctx->signals[SHELL_SIGNAL_RXRDY];
  986. z_shell_print_prompt_and_cmd(shell);
  987. /* Arbitrary delay added to ensure that prompt is
  988. * readable and can be used to enter further commands.
  989. */
  990. if (shell->ctx->cmd_buff_len) {
  991. k_sleep(K_MSEC(15));
  992. }
  993. k_poll_signal_check(signal, &signaled, &result);
  994. } while (processed && !signaled);
  995. }
  996. static int instance_init(const struct shell *shell, const void *p_config,
  997. bool use_colors)
  998. {
  999. __ASSERT_NO_MSG((shell->shell_flag == SHELL_FLAG_CRLF_DEFAULT) ||
  1000. (shell->shell_flag == SHELL_FLAG_OLF_CRLF));
  1001. memset(shell->ctx, 0, sizeof(*shell->ctx));
  1002. shell->ctx->prompt = shell->default_prompt;
  1003. if (CONFIG_SHELL_CMD_ROOT[0]) {
  1004. shell->ctx->selected_cmd = root_cmd_find(CONFIG_SHELL_CMD_ROOT);
  1005. }
  1006. history_init(shell);
  1007. k_mutex_init(&shell->ctx->wr_mtx);
  1008. for (int i = 0; i < SHELL_SIGNALS; i++) {
  1009. k_poll_signal_init(&shell->ctx->signals[i]);
  1010. k_poll_event_init(&shell->ctx->events[i],
  1011. K_POLL_TYPE_SIGNAL,
  1012. K_POLL_MODE_NOTIFY_ONLY,
  1013. &shell->ctx->signals[i]);
  1014. }
  1015. if (IS_ENABLED(CONFIG_SHELL_STATS)) {
  1016. shell->stats->log_lost_cnt = 0;
  1017. }
  1018. z_flag_tx_rdy_set(shell, true);
  1019. z_flag_echo_set(shell, IS_ENABLED(CONFIG_SHELL_ECHO_STATUS));
  1020. z_flag_obscure_set(shell, IS_ENABLED(CONFIG_SHELL_START_OBSCURED));
  1021. z_flag_mode_delete_set(shell,
  1022. IS_ENABLED(CONFIG_SHELL_BACKSPACE_MODE_DELETE));
  1023. shell->ctx->vt100_ctx.cons.terminal_wid =
  1024. CONFIG_SHELL_DEFAULT_TERMINAL_WIDTH;
  1025. shell->ctx->vt100_ctx.cons.terminal_hei =
  1026. CONFIG_SHELL_DEFAULT_TERMINAL_HEIGHT;
  1027. shell->ctx->vt100_ctx.cons.name_len = z_shell_strlen(shell->ctx->prompt);
  1028. z_flag_use_colors_set(shell, IS_ENABLED(CONFIG_SHELL_VT100_COLORS) && use_colors);
  1029. int ret = shell->iface->api->init(shell->iface, p_config,
  1030. transport_evt_handler,
  1031. (void *)shell);
  1032. if (ret == 0) {
  1033. state_set(shell, SHELL_STATE_INITIALIZED);
  1034. }
  1035. return ret;
  1036. }
  1037. static int instance_uninit(const struct shell *shell)
  1038. {
  1039. __ASSERT_NO_MSG(shell);
  1040. __ASSERT_NO_MSG(shell->ctx && shell->iface);
  1041. int err;
  1042. if (z_flag_processing_get(shell)) {
  1043. return -EBUSY;
  1044. }
  1045. if (IS_ENABLED(CONFIG_SHELL_LOG_BACKEND)) {
  1046. /* todo purge log queue */
  1047. z_shell_log_backend_disable(shell->log_backend);
  1048. }
  1049. err = shell->iface->api->uninit(shell->iface);
  1050. if (err != 0) {
  1051. return err;
  1052. }
  1053. history_purge(shell);
  1054. state_set(shell, SHELL_STATE_UNINITIALIZED);
  1055. return 0;
  1056. }
  1057. typedef void (*shell_signal_handler_t)(const struct shell *shell);
  1058. static void shell_signal_handle(const struct shell *shell,
  1059. enum shell_signal sig_idx,
  1060. shell_signal_handler_t handler)
  1061. {
  1062. struct k_poll_signal *signal = &shell->ctx->signals[sig_idx];
  1063. int set;
  1064. int res;
  1065. k_poll_signal_check(signal, &set, &res);
  1066. if (set) {
  1067. k_poll_signal_reset(signal);
  1068. handler(shell);
  1069. }
  1070. }
  1071. static void kill_handler(const struct shell *shell)
  1072. {
  1073. int err = instance_uninit(shell);
  1074. if (shell->ctx->uninit_cb) {
  1075. shell->ctx->uninit_cb(shell, err);
  1076. }
  1077. shell->ctx->tid = NULL;
  1078. k_thread_abort(k_current_get());
  1079. }
  1080. void shell_thread(void *shell_handle, void *arg_log_backend,
  1081. void *arg_log_level)
  1082. {
  1083. struct shell *shell = shell_handle;
  1084. bool log_backend = (bool)arg_log_backend;
  1085. uint32_t log_level = POINTER_TO_UINT(arg_log_level);
  1086. int err;
  1087. err = shell->iface->api->enable(shell->iface, false);
  1088. if (err != 0) {
  1089. return;
  1090. }
  1091. if (IS_ENABLED(CONFIG_SHELL_LOG_BACKEND) && log_backend
  1092. && !IS_ENABLED(CONFIG_SHELL_START_OBSCURED)) {
  1093. z_shell_log_backend_enable(shell->log_backend, (void *)shell,
  1094. log_level);
  1095. }
  1096. /* Enable shell and print prompt. */
  1097. err = shell_start(shell);
  1098. if (err != 0) {
  1099. return;
  1100. }
  1101. while (true) {
  1102. /* waiting for all signals except SHELL_SIGNAL_TXDONE */
  1103. err = k_poll(shell->ctx->events, SHELL_SIGNAL_TXDONE,
  1104. K_FOREVER);
  1105. if (err != 0) {
  1106. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1107. z_shell_fprintf(shell, SHELL_ERROR,
  1108. "Shell thread error: %d", err);
  1109. k_mutex_unlock(&shell->ctx->wr_mtx);
  1110. return;
  1111. }
  1112. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1113. if (shell->iface->api->update) {
  1114. shell->iface->api->update(shell->iface);
  1115. }
  1116. shell_signal_handle(shell, SHELL_SIGNAL_KILL, kill_handler);
  1117. shell_signal_handle(shell, SHELL_SIGNAL_RXRDY, shell_process);
  1118. if (IS_ENABLED(CONFIG_SHELL_LOG_BACKEND)) {
  1119. shell_signal_handle(shell, SHELL_SIGNAL_LOG_MSG,
  1120. shell_log_process);
  1121. }
  1122. k_mutex_unlock(&shell->ctx->wr_mtx);
  1123. }
  1124. }
  1125. int shell_init(const struct shell *shell, const void *transport_config,
  1126. bool use_colors, bool log_backend, uint32_t init_log_level)
  1127. {
  1128. __ASSERT_NO_MSG(shell);
  1129. __ASSERT_NO_MSG(shell->ctx && shell->iface && shell->default_prompt);
  1130. if (shell->ctx->tid) {
  1131. return -EALREADY;
  1132. }
  1133. int err = instance_init(shell, transport_config, use_colors);
  1134. if (err != 0) {
  1135. return err;
  1136. }
  1137. k_tid_t tid = k_thread_create(shell->thread,
  1138. shell->stack, CONFIG_SHELL_STACK_SIZE,
  1139. shell_thread, (void *)shell, (void *)log_backend,
  1140. UINT_TO_POINTER(init_log_level),
  1141. SHELL_THREAD_PRIORITY, 0, K_NO_WAIT);
  1142. shell->ctx->tid = tid;
  1143. k_thread_name_set(tid, shell->thread_name);
  1144. return 0;
  1145. }
  1146. void shell_uninit(const struct shell *shell, shell_uninit_cb_t cb)
  1147. {
  1148. __ASSERT_NO_MSG(shell);
  1149. if (IS_ENABLED(CONFIG_MULTITHREADING)) {
  1150. struct k_poll_signal *signal =
  1151. &shell->ctx->signals[SHELL_SIGNAL_KILL];
  1152. shell->ctx->uninit_cb = cb;
  1153. /* signal kill message */
  1154. (void)k_poll_signal_raise(signal, 0);
  1155. return;
  1156. }
  1157. int err = instance_uninit(shell);
  1158. if (cb) {
  1159. cb(shell, err);
  1160. } else {
  1161. __ASSERT_NO_MSG(0);
  1162. }
  1163. }
  1164. int shell_start(const struct shell *shell)
  1165. {
  1166. __ASSERT_NO_MSG(shell);
  1167. __ASSERT_NO_MSG(shell->ctx && shell->iface && shell->default_prompt);
  1168. if (state_get(shell) != SHELL_STATE_INITIALIZED) {
  1169. return -ENOTSUP;
  1170. }
  1171. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1172. if (IS_ENABLED(CONFIG_SHELL_VT100_COLORS)) {
  1173. z_shell_vt100_color_set(shell, SHELL_NORMAL);
  1174. }
  1175. z_shell_raw_fprintf(shell->fprintf_ctx, "\n\n");
  1176. state_set(shell, SHELL_STATE_ACTIVE);
  1177. k_mutex_unlock(&shell->ctx->wr_mtx);
  1178. return 0;
  1179. }
  1180. int shell_stop(const struct shell *shell)
  1181. {
  1182. __ASSERT_NO_MSG(shell);
  1183. __ASSERT_NO_MSG(shell->ctx);
  1184. enum shell_state state = state_get(shell);
  1185. if ((state == SHELL_STATE_INITIALIZED) ||
  1186. (state == SHELL_STATE_UNINITIALIZED)) {
  1187. return -ENOTSUP;
  1188. }
  1189. state_set(shell, SHELL_STATE_INITIALIZED);
  1190. return 0;
  1191. }
  1192. void shell_process(const struct shell *shell)
  1193. {
  1194. __ASSERT_NO_MSG(shell);
  1195. __ASSERT_NO_MSG(shell->ctx);
  1196. /* atomically set the processing flag */
  1197. z_flag_processing_set(shell, true);
  1198. switch (shell->ctx->state) {
  1199. case SHELL_STATE_UNINITIALIZED:
  1200. case SHELL_STATE_INITIALIZED:
  1201. /* Console initialized but not started. */
  1202. break;
  1203. case SHELL_STATE_ACTIVE:
  1204. state_collect(shell);
  1205. break;
  1206. default:
  1207. break;
  1208. }
  1209. /* atomically clear the processing flag */
  1210. z_flag_processing_set(shell, false);
  1211. }
  1212. /* This function mustn't be used from shell context to avoid deadlock.
  1213. * However it can be used in shell command handlers.
  1214. */
  1215. void shell_vfprintf(const struct shell *shell, enum shell_vt100_color color,
  1216. const char *fmt, va_list args)
  1217. {
  1218. __ASSERT_NO_MSG(shell);
  1219. __ASSERT(!k_is_in_isr(), "Thread context required.");
  1220. __ASSERT_NO_MSG(shell->ctx);
  1221. __ASSERT_NO_MSG((shell->ctx->internal.flags.cmd_ctx == 1) ||
  1222. (k_current_get() != shell->ctx->tid));
  1223. __ASSERT_NO_MSG(shell->fprintf_ctx);
  1224. __ASSERT_NO_MSG(fmt);
  1225. /* Sending a message to a non-active shell leads to a dead lock. */
  1226. if (state_get(shell) != SHELL_STATE_ACTIVE) {
  1227. z_flag_print_noinit_set(shell, true);
  1228. return;
  1229. }
  1230. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1231. if (!z_flag_cmd_ctx_get(shell) && !shell->ctx->bypass) {
  1232. z_shell_cmd_line_erase(shell);
  1233. }
  1234. z_shell_vfprintf(shell, color, fmt, args);
  1235. if (!z_flag_cmd_ctx_get(shell) && !shell->ctx->bypass) {
  1236. z_shell_print_prompt_and_cmd(shell);
  1237. }
  1238. z_transport_buffer_flush(shell);
  1239. k_mutex_unlock(&shell->ctx->wr_mtx);
  1240. }
  1241. /* This function mustn't be used from shell context to avoid deadlock.
  1242. * However it can be used in shell command handlers.
  1243. */
  1244. void shell_fprintf(const struct shell *shell, enum shell_vt100_color color,
  1245. const char *fmt, ...)
  1246. {
  1247. va_list args;
  1248. va_start(args, fmt);
  1249. shell_vfprintf(shell, color, fmt, args);
  1250. va_end(args);
  1251. }
  1252. void shell_hexdump_line(const struct shell *shell, unsigned int offset,
  1253. const uint8_t *data, size_t len)
  1254. {
  1255. int i;
  1256. shell_fprintf(shell, SHELL_NORMAL, "%08X: ", offset);
  1257. for (i = 0; i < SHELL_HEXDUMP_BYTES_IN_LINE; i++) {
  1258. if (i > 0 && !(i % 8)) {
  1259. shell_fprintf(shell, SHELL_NORMAL, " ");
  1260. }
  1261. if (i < len) {
  1262. shell_fprintf(shell, SHELL_NORMAL, "%02x ",
  1263. data[i] & 0xFF);
  1264. } else {
  1265. shell_fprintf(shell, SHELL_NORMAL, " ");
  1266. }
  1267. }
  1268. shell_fprintf(shell, SHELL_NORMAL, "|");
  1269. for (i = 0; i < SHELL_HEXDUMP_BYTES_IN_LINE; i++) {
  1270. if (i > 0 && !(i % 8)) {
  1271. shell_fprintf(shell, SHELL_NORMAL, " ");
  1272. }
  1273. if (i < len) {
  1274. char c = data[i];
  1275. shell_fprintf(shell, SHELL_NORMAL, "%c",
  1276. isprint((int)c) ? c : '.');
  1277. } else {
  1278. shell_fprintf(shell, SHELL_NORMAL, " ");
  1279. }
  1280. }
  1281. shell_print(shell, "|");
  1282. }
  1283. void shell_hexdump(const struct shell *shell, const uint8_t *data, size_t len)
  1284. {
  1285. const uint8_t *p = data;
  1286. size_t line_len;
  1287. while (len) {
  1288. line_len = MIN(len, SHELL_HEXDUMP_BYTES_IN_LINE);
  1289. shell_hexdump_line(shell, p - data, p, line_len);
  1290. len -= line_len;
  1291. p += line_len;
  1292. }
  1293. }
  1294. int shell_prompt_change(const struct shell *shell, const char *prompt)
  1295. {
  1296. __ASSERT_NO_MSG(shell);
  1297. if (prompt == NULL) {
  1298. return -EINVAL;
  1299. }
  1300. shell->ctx->prompt = prompt;
  1301. shell->ctx->vt100_ctx.cons.name_len = z_shell_strlen(prompt);
  1302. return 0;
  1303. }
  1304. void shell_help(const struct shell *shell)
  1305. {
  1306. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1307. shell_internal_help_print(shell);
  1308. k_mutex_unlock(&shell->ctx->wr_mtx);
  1309. }
  1310. int shell_execute_cmd(const struct shell *shell, const char *cmd)
  1311. {
  1312. uint16_t cmd_len = z_shell_strlen(cmd);
  1313. int ret_val;
  1314. if (cmd == NULL) {
  1315. return -ENOEXEC;
  1316. }
  1317. if (cmd_len > (CONFIG_SHELL_CMD_BUFF_SIZE - 1)) {
  1318. return -ENOMEM;
  1319. }
  1320. if (shell == NULL) {
  1321. #if defined(CONFIG_SHELL_BACKEND_DUMMY)
  1322. shell = shell_backend_dummy_get_ptr();
  1323. #else
  1324. return -EINVAL;
  1325. #endif
  1326. }
  1327. __ASSERT(shell->ctx->internal.flags.cmd_ctx == 0,
  1328. "Function cannot be called"
  1329. " from command context");
  1330. strcpy(shell->ctx->cmd_buff, cmd);
  1331. shell->ctx->cmd_buff_len = cmd_len;
  1332. shell->ctx->cmd_buff_pos = cmd_len;
  1333. k_mutex_lock(&shell->ctx->wr_mtx, K_FOREVER);
  1334. ret_val = execute(shell);
  1335. k_mutex_unlock(&shell->ctx->wr_mtx);
  1336. cmd_buffer_clear(shell);
  1337. return ret_val;
  1338. }
  1339. int shell_insert_mode_set(const struct shell *shell, bool val)
  1340. {
  1341. if (shell == NULL) {
  1342. return -EINVAL;
  1343. }
  1344. return (int)z_flag_insert_mode_set(shell, val);
  1345. }
  1346. int shell_use_colors_set(const struct shell *shell, bool val)
  1347. {
  1348. if (shell == NULL) {
  1349. return -EINVAL;
  1350. }
  1351. return (int)z_flag_use_colors_set(shell, val);
  1352. }
  1353. int shell_echo_set(const struct shell *shell, bool val)
  1354. {
  1355. if (shell == NULL) {
  1356. return -EINVAL;
  1357. }
  1358. return (int)z_flag_echo_set(shell, val);
  1359. }
  1360. int shell_obscure_set(const struct shell *shell, bool val)
  1361. {
  1362. if (shell == NULL) {
  1363. return -EINVAL;
  1364. }
  1365. return (int)z_flag_obscure_set(shell, val);
  1366. }
  1367. int shell_mode_delete_set(const struct shell *shell, bool val)
  1368. {
  1369. if (shell == NULL) {
  1370. return -EINVAL;
  1371. }
  1372. return (int)z_flag_mode_delete_set(shell, val);
  1373. }
  1374. void shell_set_bypass(const struct shell *sh, shell_bypass_cb_t bypass)
  1375. {
  1376. sh->ctx->bypass = bypass;
  1377. }
  1378. static int cmd_help(const struct shell *shell, size_t argc, char **argv)
  1379. {
  1380. ARG_UNUSED(argc);
  1381. ARG_UNUSED(argv);
  1382. #if defined(CONFIG_SHELL_TAB)
  1383. shell_print(shell, "Please press the <Tab> button to see all available "
  1384. "commands.");
  1385. #endif
  1386. #if defined(CONFIG_SHELL_TAB_AUTOCOMPLETION)
  1387. shell_print(shell,
  1388. "You can also use the <Tab> button to prompt or auto-complete"
  1389. " all commands or its subcommands.");
  1390. #endif
  1391. #if defined(CONFIG_SHELL_HELP)
  1392. shell_print(shell,
  1393. "You can try to call commands with <-h> or <--help> parameter"
  1394. " for more information.");
  1395. #endif
  1396. #if defined(CONFIG_SHELL_METAKEYS)
  1397. shell_print(shell,
  1398. "\nShell supports following meta-keys:\n"
  1399. " Ctrl + (a key from: abcdefklnpuw)\n"
  1400. " Alt + (a key from: bf)\n"
  1401. "Please refer to shell documentation for more details.");
  1402. #endif
  1403. if (IS_ENABLED(CONFIG_SHELL_HELP)) {
  1404. /* For NULL argument function will print all root commands */
  1405. z_shell_help_subcmd_print(shell, NULL,
  1406. "\nAvailable commands:\n");
  1407. } else {
  1408. const struct shell_static_entry *entry;
  1409. size_t idx = 0;
  1410. shell_print(shell, "\nAvailable commands:");
  1411. while ((entry = z_shell_cmd_get(NULL, idx++, NULL)) != NULL) {
  1412. shell_print(shell, " %s", entry->syntax);
  1413. }
  1414. }
  1415. return 0;
  1416. }
  1417. SHELL_CMD_ARG_REGISTER(help, NULL, "Prints the help message.", cmd_help, 1, 0);