shell_cmds.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <shell/shell.h>
  7. #include "shell_utils.h"
  8. #include "shell_help.h"
  9. #include "shell_ops.h"
  10. #include "shell_vt100.h"
  11. #define SHELL_MSG_CMD_NOT_SUPPORTED "Command not supported.\n"
  12. #define SHELL_HELP_CLEAR "Clear screen."
  13. #define SHELL_HELP_BACKSPACE_MODE "Toggle backspace key mode.\n" \
  14. "Some terminals are not sending separate escape code for " \
  15. "backspace and delete button. This command forces shell to interpret" \
  16. " delete key as backspace."
  17. #define SHELL_HELP_BACKSPACE_MODE_BACKSPACE "Set different escape" \
  18. " code for backspace and delete key."
  19. #define SHELL_HELP_BACKSPACE_MODE_DELETE "Set the same escape" \
  20. " code for backspace and delete key."
  21. #define SHELL_HELP_COLORS "Toggle colored syntax."
  22. #define SHELL_HELP_COLORS_OFF "Disable colored syntax."
  23. #define SHELL_HELP_COLORS_ON "Enable colored syntax."
  24. #define SHELL_HELP_STATISTICS "Shell statistics."
  25. #define SHELL_HELP_STATISTICS_SHOW \
  26. "Get shell statistics for the Logger module."
  27. #define SHELL_HELP_STATISTICS_RESET \
  28. "Reset shell statistics for the Logger module."
  29. #define SHELL_HELP_RESIZE \
  30. "Console gets terminal screen size or assumes default in case" \
  31. " the readout fails. It must be executed after each terminal" \
  32. " width change to ensure correct text display."
  33. #define SHELL_HELP_RESIZE_DEFAULT \
  34. "Assume 80 chars screen width and send this setting " \
  35. "to the terminal."
  36. #define SHELL_HELP_HISTORY "Command history."
  37. #define SHELL_HELP_ECHO "Toggle shell echo."
  38. #define SHELL_HELP_ECHO_ON "Enable shell echo."
  39. #define SHELL_HELP_ECHO_OFF \
  40. "Disable shell echo. Editing keys and meta-keys are not handled"
  41. #define SHELL_HELP_SELECT "Selects new root command. In order for the " \
  42. "command to be selected, it must meet the criteria:\n" \
  43. " - it is a static command\n" \
  44. " - it is not preceded by a dynamic command\n" \
  45. " - it accepts arguments\n" \
  46. "Return to the main command tree is done by pressing alt+r."
  47. #define SHELL_HELP_SHELL "Useful, not Unix-like shell commands."
  48. #define SHELL_HELP_HELP "Prints help message."
  49. #define SHELL_MSG_UNKNOWN_PARAMETER " unknown parameter: "
  50. #define SHELL_MAX_TERMINAL_SIZE (250u)
  51. /* 10 == {esc, [, 2, 5, 0, ;, 2, 5, 0, '\0'} */
  52. #define SHELL_CURSOR_POSITION_BUFFER (10u)
  53. #define SHELL_DEFAULT_TERMINAL_WIDTH 80
  54. #define SHELL_DEFAULT_TERMINAL_HEIGHT 24
  55. /* Function reads cursor position from terminal. */
  56. static int cursor_position_get(const struct shell *shell, uint16_t *x, uint16_t *y)
  57. {
  58. uint16_t buff_idx = 0U;
  59. size_t cnt;
  60. char c = 0;
  61. *x = 0U;
  62. *y = 0U;
  63. memset(shell->ctx->temp_buff, 0, sizeof(shell->ctx->temp_buff));
  64. /* escape code asking terminal about its size */
  65. static char const cmd_get_terminal_size[] = "\033[6n";
  66. z_shell_raw_fprintf(shell->fprintf_ctx, cmd_get_terminal_size);
  67. /* fprintf buffer needs to be flushed to start sending prepared
  68. * escape code to the terminal.
  69. */
  70. z_transport_buffer_flush(shell);
  71. /* timeout for terminal response = ~1s */
  72. for (uint16_t i = 0; i < 1000; i++) {
  73. do {
  74. (void)shell->iface->api->read(shell->iface, &c,
  75. sizeof(c), &cnt);
  76. if (cnt == 0) {
  77. k_busy_wait(1000);
  78. break;
  79. }
  80. if ((c != SHELL_VT100_ASCII_ESC) &&
  81. (shell->ctx->temp_buff[0] !=
  82. SHELL_VT100_ASCII_ESC)) {
  83. continue;
  84. }
  85. if (c == 'R') { /* End of response from the terminal. */
  86. shell->ctx->temp_buff[buff_idx] = '\0';
  87. if (shell->ctx->temp_buff[1] != '[') {
  88. shell->ctx->temp_buff[0] = 0;
  89. return -EIO;
  90. }
  91. /* Index start position in the buffer where 'y'
  92. * is stored.
  93. */
  94. buff_idx = 2U;
  95. while (shell->ctx->temp_buff[buff_idx] != ';') {
  96. *y = *y * 10U +
  97. (shell->ctx->temp_buff[buff_idx++] -
  98. '0');
  99. if (buff_idx >=
  100. CONFIG_SHELL_CMD_BUFF_SIZE) {
  101. return -EMSGSIZE;
  102. }
  103. }
  104. if (++buff_idx >= CONFIG_SHELL_CMD_BUFF_SIZE) {
  105. return -EIO;
  106. }
  107. while (shell->ctx->temp_buff[buff_idx]
  108. != '\0') {
  109. *x = *x * 10U +
  110. (shell->ctx->temp_buff[buff_idx++] -
  111. '0');
  112. if (buff_idx >=
  113. CONFIG_SHELL_CMD_BUFF_SIZE) {
  114. return -EMSGSIZE;
  115. }
  116. }
  117. /* horizontal cursor position */
  118. if (*x > SHELL_MAX_TERMINAL_SIZE) {
  119. *x = SHELL_MAX_TERMINAL_SIZE;
  120. }
  121. /* vertical cursor position */
  122. if (*y > SHELL_MAX_TERMINAL_SIZE) {
  123. *y = SHELL_MAX_TERMINAL_SIZE;
  124. }
  125. shell->ctx->temp_buff[0] = 0;
  126. return 0;
  127. }
  128. shell->ctx->temp_buff[buff_idx] = c;
  129. if (++buff_idx > SHELL_CURSOR_POSITION_BUFFER - 1) {
  130. shell->ctx->temp_buff[0] = 0;
  131. /* data_buf[SHELL_CURSOR_POSITION_BUFFER - 1]
  132. * is reserved for '\0'
  133. */
  134. return -ENOMEM;
  135. }
  136. } while (cnt > 0);
  137. }
  138. return -ETIMEDOUT;
  139. }
  140. /* Function gets terminal width and height. */
  141. static int terminal_size_get(const struct shell *shell)
  142. {
  143. uint16_t x; /* horizontal position */
  144. uint16_t y; /* vertical position */
  145. int ret_val = 0;
  146. z_cursor_save(shell);
  147. /* Assumption: terminal width and height < 999. */
  148. /* Move to last column. */
  149. z_shell_op_cursor_vert_move(shell, -SHELL_MAX_TERMINAL_SIZE);
  150. /* Move to last row. */
  151. z_shell_op_cursor_horiz_move(shell, SHELL_MAX_TERMINAL_SIZE);
  152. if (cursor_position_get(shell, &x, &y) == 0) {
  153. shell->ctx->vt100_ctx.cons.terminal_wid = x;
  154. shell->ctx->vt100_ctx.cons.terminal_hei = y;
  155. } else {
  156. ret_val = -ENOTSUP;
  157. }
  158. z_cursor_restore(shell);
  159. return ret_val;
  160. }
  161. #ifdef CONFIG_SHELL_VT100_COMMANDS
  162. static int cmd_clear(const struct shell *shell, size_t argc, char **argv)
  163. {
  164. ARG_UNUSED(argv);
  165. Z_SHELL_VT100_CMD(shell, SHELL_VT100_CURSORHOME);
  166. Z_SHELL_VT100_CMD(shell, SHELL_VT100_CLEARSCREEN);
  167. return 0;
  168. }
  169. #endif
  170. static int cmd_bacskpace_mode_backspace(const struct shell *shell, size_t argc,
  171. char **argv)
  172. {
  173. ARG_UNUSED(argc);
  174. ARG_UNUSED(argv);
  175. z_flag_mode_delete_set(shell, false);
  176. return 0;
  177. }
  178. static int cmd_bacskpace_mode_delete(const struct shell *shell, size_t argc,
  179. char **argv)
  180. {
  181. ARG_UNUSED(argc);
  182. ARG_UNUSED(argv);
  183. z_flag_mode_delete_set(shell, true);
  184. return 0;
  185. }
  186. #ifdef CONFIG_SHELL_VT100_COMMANDS
  187. static int cmd_colors_off(const struct shell *shell, size_t argc, char **argv)
  188. {
  189. ARG_UNUSED(argc);
  190. ARG_UNUSED(argv);
  191. z_flag_use_colors_set(shell, false);
  192. return 0;
  193. }
  194. static int cmd_colors_on(const struct shell *shell, size_t argc, char **argv)
  195. {
  196. ARG_UNUSED(argv);
  197. ARG_UNUSED(argv);
  198. z_flag_use_colors_set(shell, true);
  199. return 0;
  200. }
  201. #endif
  202. static int cmd_echo_off(const struct shell *shell, size_t argc, char **argv)
  203. {
  204. ARG_UNUSED(argc);
  205. ARG_UNUSED(argv);
  206. z_flag_echo_set(shell, false);
  207. return 0;
  208. }
  209. static int cmd_echo_on(const struct shell *shell, size_t argc, char **argv)
  210. {
  211. ARG_UNUSED(argc);
  212. ARG_UNUSED(argv);
  213. z_flag_echo_set(shell, true);
  214. return 0;
  215. }
  216. static int cmd_echo(const struct shell *shell, size_t argc, char **argv)
  217. {
  218. if (argc == 2) {
  219. shell_error(shell, "%s:%s%s", argv[0],
  220. SHELL_MSG_UNKNOWN_PARAMETER, argv[1]);
  221. return -EINVAL;
  222. }
  223. shell_print(shell, "Echo status: %s",
  224. z_flag_echo_get(shell) ? "on" : "off");
  225. return 0;
  226. }
  227. static int cmd_history(const struct shell *shell, size_t argc, char **argv)
  228. {
  229. ARG_UNUSED(argc);
  230. ARG_UNUSED(argv);
  231. size_t i = 0;
  232. uint16_t len;
  233. while (1) {
  234. z_shell_history_get(shell->history, true,
  235. shell->ctx->temp_buff, &len);
  236. if (len) {
  237. shell_print(shell, "[%3d] %s",
  238. i++, shell->ctx->temp_buff);
  239. } else {
  240. break;
  241. }
  242. }
  243. shell->ctx->temp_buff[0] = '\0';
  244. return 0;
  245. }
  246. static int cmd_shell_stats_show(const struct shell *shell, size_t argc,
  247. char **argv)
  248. {
  249. ARG_UNUSED(argc);
  250. ARG_UNUSED(argv);
  251. shell_print(shell, "Lost logs: %u", shell->stats->log_lost_cnt);
  252. return 0;
  253. }
  254. static int cmd_shell_stats_reset(const struct shell *shell,
  255. size_t argc, char **argv)
  256. {
  257. ARG_UNUSED(argc);
  258. ARG_UNUSED(argv);
  259. shell->stats->log_lost_cnt = 0;
  260. return 0;
  261. }
  262. static int cmd_resize_default(const struct shell *shell,
  263. size_t argc, char **argv)
  264. {
  265. ARG_UNUSED(argc);
  266. ARG_UNUSED(argv);
  267. Z_SHELL_VT100_CMD(shell, SHELL_VT100_SETCOL_80);
  268. shell->ctx->vt100_ctx.cons.terminal_wid = SHELL_DEFAULT_TERMINAL_WIDTH;
  269. shell->ctx->vt100_ctx.cons.terminal_hei = SHELL_DEFAULT_TERMINAL_HEIGHT;
  270. return 0;
  271. }
  272. static int cmd_resize(const struct shell *shell, size_t argc, char **argv)
  273. {
  274. int err;
  275. if (argc != 1) {
  276. shell_error(shell, "%s:%s%s", argv[0],
  277. SHELL_MSG_UNKNOWN_PARAMETER, argv[1]);
  278. return -EINVAL;
  279. }
  280. err = terminal_size_get(shell);
  281. if (err != 0) {
  282. shell->ctx->vt100_ctx.cons.terminal_wid =
  283. CONFIG_SHELL_DEFAULT_TERMINAL_WIDTH;
  284. shell->ctx->vt100_ctx.cons.terminal_hei =
  285. CONFIG_SHELL_DEFAULT_TERMINAL_HEIGHT;
  286. shell_warn(shell, "No response from the terminal, assumed 80x24"
  287. " screen size");
  288. return -ENOEXEC;
  289. }
  290. return 0;
  291. }
  292. static bool no_args(const struct shell_static_entry *entry)
  293. {
  294. return (entry->args.mandatory == 1) && (entry->args.optional == 0);
  295. }
  296. static int cmd_select(const struct shell *shell, size_t argc, char **argv)
  297. {
  298. const struct shell_static_entry *candidate = NULL;
  299. struct shell_static_entry entry;
  300. size_t matching_argc;
  301. argc--;
  302. argv = argv + 1;
  303. candidate = z_shell_get_last_command(shell->ctx->selected_cmd,
  304. argc, (const char **)argv,
  305. &matching_argc, &entry, true);
  306. if ((candidate != NULL) && !no_args(candidate)
  307. && (argc == matching_argc)) {
  308. shell->ctx->selected_cmd = candidate;
  309. return 0;
  310. }
  311. shell_error(shell, "Cannot select command");
  312. return -EINVAL;
  313. }
  314. #ifdef CONFIG_SHELL_VT100_COMMANDS
  315. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_colors,
  316. SHELL_CMD_ARG(off, NULL, SHELL_HELP_COLORS_OFF, cmd_colors_off, 1, 0),
  317. SHELL_CMD_ARG(on, NULL, SHELL_HELP_COLORS_ON, cmd_colors_on, 1, 0),
  318. SHELL_SUBCMD_SET_END
  319. );
  320. #endif
  321. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_echo,
  322. SHELL_CMD_ARG(off, NULL, SHELL_HELP_ECHO_OFF, cmd_echo_off, 1, 0),
  323. SHELL_CMD_ARG(on, NULL, SHELL_HELP_ECHO_ON, cmd_echo_on, 1, 0),
  324. SHELL_SUBCMD_SET_END
  325. );
  326. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_shell_stats,
  327. SHELL_CMD_ARG(reset, NULL, SHELL_HELP_STATISTICS_RESET,
  328. cmd_shell_stats_reset, 1, 0),
  329. SHELL_CMD_ARG(show, NULL, SHELL_HELP_STATISTICS_SHOW,
  330. cmd_shell_stats_show, 1, 0),
  331. SHELL_SUBCMD_SET_END
  332. );
  333. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_backspace_mode,
  334. SHELL_CMD_ARG(backspace, NULL, SHELL_HELP_BACKSPACE_MODE_BACKSPACE,
  335. cmd_bacskpace_mode_backspace, 1, 0),
  336. SHELL_CMD_ARG(delete, NULL, SHELL_HELP_BACKSPACE_MODE_DELETE,
  337. cmd_bacskpace_mode_delete, 1, 0),
  338. SHELL_SUBCMD_SET_END
  339. );
  340. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_shell,
  341. SHELL_CMD(backspace_mode, &m_sub_backspace_mode,
  342. SHELL_HELP_BACKSPACE_MODE, NULL),
  343. SHELL_COND_CMD(CONFIG_SHELL_VT100_COMMANDS, colors, &m_sub_colors,
  344. SHELL_HELP_COLORS, NULL),
  345. SHELL_CMD_ARG(echo, &m_sub_echo, SHELL_HELP_ECHO, cmd_echo, 1, 1),
  346. SHELL_COND_CMD(CONFIG_SHELL_STATS, stats, &m_sub_shell_stats,
  347. SHELL_HELP_STATISTICS, NULL),
  348. SHELL_SUBCMD_SET_END
  349. );
  350. SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_resize,
  351. SHELL_CMD_ARG(default, NULL, SHELL_HELP_RESIZE_DEFAULT,
  352. cmd_resize_default, 1, 0),
  353. SHELL_SUBCMD_SET_END
  354. );
  355. SHELL_COND_CMD_ARG_REGISTER(CONFIG_SHELL_VT100_COMMANDS, clear, NULL,
  356. SHELL_HELP_CLEAR, cmd_clear, 1, 0);
  357. SHELL_CMD_REGISTER(shell, &m_sub_shell, SHELL_HELP_SHELL, NULL);
  358. SHELL_COND_CMD_ARG_REGISTER(CONFIG_SHELL_HISTORY, history, NULL,
  359. SHELL_HELP_HISTORY, cmd_history, 1, 0);
  360. SHELL_COND_CMD_ARG_REGISTER(CONFIG_SHELL_CMDS_RESIZE, resize, &m_sub_resize,
  361. SHELL_HELP_RESIZE, cmd_resize, 1, 1);
  362. SHELL_COND_CMD_ARG_REGISTER(CONFIG_SHELL_CMDS_SELECT, select, NULL,
  363. SHELL_HELP_SELECT, cmd_select, 2,
  364. SHELL_OPT_ARG_CHECK_SKIP);