shell_ops.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <ctype.h>
  7. #include "shell_ops.h"
  8. void z_shell_op_cursor_vert_move(const struct shell *shell, int32_t delta)
  9. {
  10. if (delta != 0) {
  11. z_shell_raw_fprintf(shell->fprintf_ctx, "\033[%d%c",
  12. delta > 0 ? delta : -delta,
  13. delta > 0 ? 'A' : 'B');
  14. }
  15. }
  16. void z_shell_op_cursor_horiz_move(const struct shell *shell, int32_t delta)
  17. {
  18. if (delta != 0) {
  19. z_shell_raw_fprintf(shell->fprintf_ctx, "\033[%d%c",
  20. delta > 0 ? delta : -delta,
  21. delta > 0 ? 'C' : 'D');
  22. }
  23. }
  24. /* Function returns true if command length is equal to multiplicity of terminal
  25. * width.
  26. */
  27. static inline bool full_line_cmd(const struct shell *shell)
  28. {
  29. return ((shell->ctx->cmd_buff_len + z_shell_strlen(shell->ctx->prompt))
  30. % shell->ctx->vt100_ctx.cons.terminal_wid == 0U);
  31. }
  32. /* Function returns true if cursor is at beginning of an empty line. */
  33. bool z_shell_cursor_in_empty_line(const struct shell *shell)
  34. {
  35. return ((shell->ctx->cmd_buff_pos + z_shell_strlen(shell->ctx->prompt))
  36. % shell->ctx->vt100_ctx.cons.terminal_wid == 0U);
  37. }
  38. void z_shell_op_cond_next_line(const struct shell *shell)
  39. {
  40. if (z_shell_cursor_in_empty_line(shell) || full_line_cmd(shell)) {
  41. z_cursor_next_line_move(shell);
  42. }
  43. }
  44. void z_shell_op_cursor_position_synchronize(const struct shell *shell)
  45. {
  46. struct shell_multiline_cons *cons = &shell->ctx->vt100_ctx.cons;
  47. bool last_line;
  48. z_shell_multiline_data_calc(cons, shell->ctx->cmd_buff_pos,
  49. shell->ctx->cmd_buff_len);
  50. last_line = (cons->cur_y == cons->cur_y_end);
  51. /* In case cursor reaches the bottom line of a terminal, it will
  52. * be moved to the next line.
  53. */
  54. if (full_line_cmd(shell)) {
  55. z_cursor_next_line_move(shell);
  56. }
  57. if (last_line) {
  58. z_shell_op_cursor_horiz_move(shell, cons->cur_x -
  59. cons->cur_x_end);
  60. } else {
  61. z_shell_op_cursor_vert_move(shell, cons->cur_y_end - cons->cur_y);
  62. z_shell_op_cursor_horiz_move(shell, cons->cur_x -
  63. cons->cur_x_end);
  64. }
  65. }
  66. void z_shell_op_cursor_move(const struct shell *shell, int16_t val)
  67. {
  68. struct shell_multiline_cons *cons = &shell->ctx->vt100_ctx.cons;
  69. uint16_t new_pos = shell->ctx->cmd_buff_pos + val;
  70. int32_t row_span;
  71. int32_t col_span;
  72. z_shell_multiline_data_calc(cons, shell->ctx->cmd_buff_pos,
  73. shell->ctx->cmd_buff_len);
  74. /* Calculate the new cursor. */
  75. row_span = z_row_span_with_buffer_offsets_get(
  76. &shell->ctx->vt100_ctx.cons,
  77. shell->ctx->cmd_buff_pos,
  78. new_pos);
  79. col_span = z_column_span_with_buffer_offsets_get(
  80. &shell->ctx->vt100_ctx.cons,
  81. shell->ctx->cmd_buff_pos,
  82. new_pos);
  83. z_shell_op_cursor_vert_move(shell, -row_span);
  84. z_shell_op_cursor_horiz_move(shell, col_span);
  85. shell->ctx->cmd_buff_pos = new_pos;
  86. }
  87. static uint16_t shift_calc(const char *str, uint16_t pos, uint16_t len, int16_t sign)
  88. {
  89. bool found = false;
  90. uint16_t ret = 0U;
  91. uint16_t idx;
  92. while (1) {
  93. idx = pos + ret * sign;
  94. if (((idx == 0U) && (sign < 0)) ||
  95. ((idx == len) && (sign > 0))) {
  96. break;
  97. }
  98. if (isalnum((int)str[idx]) != 0) {
  99. found = true;
  100. } else {
  101. if (found) {
  102. break;
  103. }
  104. }
  105. ret++;
  106. }
  107. return ret;
  108. }
  109. void z_shell_op_cursor_word_move(const struct shell *shell, int16_t val)
  110. {
  111. int16_t shift;
  112. int16_t sign;
  113. if (val < 0) {
  114. val = -val;
  115. sign = -1;
  116. } else {
  117. sign = 1;
  118. }
  119. while (val--) {
  120. shift = shift_calc(shell->ctx->cmd_buff,
  121. shell->ctx->cmd_buff_pos,
  122. shell->ctx->cmd_buff_len, sign);
  123. z_shell_op_cursor_move(shell, sign * shift);
  124. }
  125. }
  126. void z_shell_op_word_remove(const struct shell *shell)
  127. {
  128. char *str = &shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos - 1];
  129. char *str_start = &shell->ctx->cmd_buff[0];
  130. uint16_t chars_to_delete;
  131. /* Line must not be empty and cursor must not be at 0 to continue. */
  132. if ((shell->ctx->cmd_buff_len == 0) ||
  133. (shell->ctx->cmd_buff_pos == 0)) {
  134. return;
  135. }
  136. /* Start at the current position. */
  137. chars_to_delete = 0U;
  138. /* Look back for all spaces then for non-spaces. */
  139. while ((str >= str_start) && (*str == ' ')) {
  140. ++chars_to_delete;
  141. --str;
  142. }
  143. while ((str >= str_start) && (*str != ' ')) {
  144. ++chars_to_delete;
  145. --str;
  146. }
  147. /* Manage the buffer. */
  148. memmove(str + 1, str + 1 + chars_to_delete,
  149. shell->ctx->cmd_buff_len - chars_to_delete);
  150. shell->ctx->cmd_buff_len -= chars_to_delete;
  151. shell->ctx->cmd_buff[shell->ctx->cmd_buff_len] = '\0';
  152. /* Update display. */
  153. z_shell_op_cursor_move(shell, -chars_to_delete);
  154. z_cursor_save(shell);
  155. z_shell_fprintf(shell, SHELL_NORMAL, "%s", str + 1);
  156. z_clear_eos(shell);
  157. z_cursor_restore(shell);
  158. }
  159. void z_shell_op_cursor_home_move(const struct shell *shell)
  160. {
  161. z_shell_op_cursor_move(shell, -shell->ctx->cmd_buff_pos);
  162. }
  163. void z_shell_op_cursor_end_move(const struct shell *shell)
  164. {
  165. z_shell_op_cursor_move(shell, shell->ctx->cmd_buff_len -
  166. shell->ctx->cmd_buff_pos);
  167. }
  168. void z_shell_op_left_arrow(const struct shell *shell)
  169. {
  170. if (shell->ctx->cmd_buff_pos > 0) {
  171. z_shell_op_cursor_move(shell, -1);
  172. }
  173. }
  174. void z_shell_op_right_arrow(const struct shell *shell)
  175. {
  176. if (shell->ctx->cmd_buff_pos < shell->ctx->cmd_buff_len) {
  177. z_shell_op_cursor_move(shell, 1);
  178. }
  179. }
  180. static void reprint_from_cursor(const struct shell *shell, uint16_t diff,
  181. bool data_removed)
  182. {
  183. /* Clear eos is needed only when newly printed command is shorter than
  184. * previously printed command. This can happen when delete or backspace
  185. * was called.
  186. *
  187. * Such condition is useful for Bluetooth devices to save number of
  188. * bytes transmitted between terminal and device.
  189. */
  190. if (data_removed) {
  191. z_clear_eos(shell);
  192. }
  193. if (z_flag_obscure_get(shell)) {
  194. int len = strlen(&shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos]);
  195. while (len--) {
  196. z_shell_raw_fprintf(shell->fprintf_ctx, "*");
  197. }
  198. } else {
  199. z_shell_fprintf(shell, SHELL_NORMAL, "%s",
  200. &shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos]);
  201. }
  202. shell->ctx->cmd_buff_pos = shell->ctx->cmd_buff_len;
  203. if (full_line_cmd(shell)) {
  204. if (((data_removed) && (diff > 0)) || (!data_removed)) {
  205. z_cursor_next_line_move(shell);
  206. }
  207. }
  208. z_shell_op_cursor_move(shell, -diff);
  209. }
  210. static void data_insert(const struct shell *shell, const char *data, uint16_t len)
  211. {
  212. uint16_t after = shell->ctx->cmd_buff_len - shell->ctx->cmd_buff_pos;
  213. char *curr_pos = &shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos];
  214. if ((shell->ctx->cmd_buff_len + len) >= CONFIG_SHELL_CMD_BUFF_SIZE) {
  215. return;
  216. }
  217. memmove(curr_pos + len, curr_pos, after);
  218. memcpy(curr_pos, data, len);
  219. shell->ctx->cmd_buff_len += len;
  220. shell->ctx->cmd_buff[shell->ctx->cmd_buff_len] = '\0';
  221. if (!z_flag_echo_get(shell)) {
  222. shell->ctx->cmd_buff_pos += len;
  223. return;
  224. }
  225. reprint_from_cursor(shell, after, false);
  226. }
  227. static void char_replace(const struct shell *shell, char data)
  228. {
  229. shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos++] = data;
  230. if (!z_flag_echo_get(shell)) {
  231. return;
  232. }
  233. if (z_flag_obscure_get(shell)) {
  234. data = '*';
  235. }
  236. z_shell_raw_fprintf(shell->fprintf_ctx, "%c", data);
  237. if (z_shell_cursor_in_empty_line(shell)) {
  238. z_cursor_next_line_move(shell);
  239. }
  240. }
  241. void z_shell_op_char_insert(const struct shell *shell, char data)
  242. {
  243. if (shell->ctx->internal.flags.insert_mode &&
  244. (shell->ctx->cmd_buff_len != shell->ctx->cmd_buff_pos)) {
  245. char_replace(shell, data);
  246. } else {
  247. data_insert(shell, &data, 1);
  248. }
  249. }
  250. void z_shell_op_char_backspace(const struct shell *shell)
  251. {
  252. if ((shell->ctx->cmd_buff_len == 0) ||
  253. (shell->ctx->cmd_buff_pos == 0)) {
  254. return;
  255. }
  256. z_shell_op_cursor_move(shell, -1);
  257. z_shell_op_char_delete(shell);
  258. }
  259. void z_shell_op_char_delete(const struct shell *shell)
  260. {
  261. uint16_t diff = shell->ctx->cmd_buff_len - shell->ctx->cmd_buff_pos;
  262. char *str = &shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos];
  263. if (diff == 0U) {
  264. return;
  265. }
  266. memmove(str, str + 1, diff);
  267. --shell->ctx->cmd_buff_len;
  268. reprint_from_cursor(shell, --diff, true);
  269. }
  270. void z_shell_op_delete_from_cursor(const struct shell *shell)
  271. {
  272. shell->ctx->cmd_buff_len = shell->ctx->cmd_buff_pos;
  273. shell->ctx->cmd_buff[shell->ctx->cmd_buff_pos] = '\0';
  274. z_clear_eos(shell);
  275. }
  276. void z_shell_op_completion_insert(const struct shell *shell,
  277. const char *compl,
  278. uint16_t compl_len)
  279. {
  280. data_insert(shell, compl, compl_len);
  281. }
  282. void z_shell_cmd_line_erase(const struct shell *shell)
  283. {
  284. z_shell_multiline_data_calc(&shell->ctx->vt100_ctx.cons,
  285. shell->ctx->cmd_buff_pos,
  286. shell->ctx->cmd_buff_len);
  287. z_shell_op_cursor_horiz_move(shell,
  288. -(shell->ctx->vt100_ctx.cons.cur_x - 1));
  289. z_shell_op_cursor_vert_move(shell, shell->ctx->vt100_ctx.cons.cur_y - 1);
  290. z_clear_eos(shell);
  291. }
  292. static void print_prompt(const struct shell *shell)
  293. {
  294. z_shell_fprintf(shell, SHELL_INFO, "%s", shell->ctx->prompt);
  295. }
  296. void z_shell_print_cmd(const struct shell *shell)
  297. {
  298. z_shell_raw_fprintf(shell->fprintf_ctx, "%s", shell->ctx->cmd_buff);
  299. }
  300. void z_shell_print_prompt_and_cmd(const struct shell *shell)
  301. {
  302. print_prompt(shell);
  303. if (z_flag_echo_get(shell)) {
  304. z_shell_print_cmd(shell);
  305. z_shell_op_cursor_position_synchronize(shell);
  306. }
  307. }
  308. static void shell_pend_on_txdone(const struct shell *shell)
  309. {
  310. if (IS_ENABLED(CONFIG_MULTITHREADING) &&
  311. (shell->ctx->state < SHELL_STATE_PANIC_MODE_ACTIVE)) {
  312. struct k_poll_event event;
  313. k_poll_event_init(&event,
  314. K_POLL_TYPE_SIGNAL,
  315. K_POLL_MODE_NOTIFY_ONLY,
  316. &shell->ctx->signals[SHELL_SIGNAL_TXDONE]);
  317. k_poll(&event, 1, K_FOREVER);
  318. k_poll_signal_reset(&shell->ctx->signals[SHELL_SIGNAL_TXDONE]);
  319. } else {
  320. /* Blocking wait in case of bare metal. */
  321. while (!z_flag_tx_rdy_get(shell)) {
  322. }
  323. z_flag_tx_rdy_set(shell, false);
  324. }
  325. }
  326. void z_shell_write(const struct shell *shell, const void *data,
  327. size_t length)
  328. {
  329. __ASSERT_NO_MSG(shell && data);
  330. size_t offset = 0;
  331. size_t tmp_cnt;
  332. while (length) {
  333. int err = shell->iface->api->write(shell->iface,
  334. &((const uint8_t *) data)[offset], length,
  335. &tmp_cnt);
  336. (void)err;
  337. __ASSERT_NO_MSG(err == 0);
  338. __ASSERT_NO_MSG(length >= tmp_cnt);
  339. offset += tmp_cnt;
  340. length -= tmp_cnt;
  341. if (tmp_cnt == 0 &&
  342. (shell->ctx->state != SHELL_STATE_PANIC_MODE_ACTIVE)) {
  343. shell_pend_on_txdone(shell);
  344. }
  345. }
  346. }
  347. /* Function shall be only used by the fprintf module. */
  348. void z_shell_print_stream(const void *user_ctx, const char *data, size_t len)
  349. {
  350. z_shell_write((const struct shell *) user_ctx, data, len);
  351. }
  352. static void vt100_bgcolor_set(const struct shell *shell,
  353. enum shell_vt100_color bgcolor)
  354. {
  355. if ((bgcolor == SHELL_NORMAL) ||
  356. (shell->ctx->vt100_ctx.col.bgcol == bgcolor)) {
  357. return;
  358. }
  359. /* -1 because default value is first in enum */
  360. uint8_t cmd[] = SHELL_VT100_BGCOLOR(bgcolor - 1);
  361. shell->ctx->vt100_ctx.col.bgcol = bgcolor;
  362. z_shell_raw_fprintf(shell->fprintf_ctx, "%s", cmd);
  363. }
  364. void z_shell_vt100_color_set(const struct shell *shell,
  365. enum shell_vt100_color color)
  366. {
  367. if (shell->ctx->vt100_ctx.col.col == color) {
  368. return;
  369. }
  370. shell->ctx->vt100_ctx.col.col = color;
  371. if (color != SHELL_NORMAL) {
  372. uint8_t cmd[] = SHELL_VT100_COLOR(color - 1);
  373. z_shell_raw_fprintf(shell->fprintf_ctx, "%s", cmd);
  374. } else {
  375. static const uint8_t cmd[] = SHELL_VT100_MODESOFF;
  376. z_shell_raw_fprintf(shell->fprintf_ctx, "%s", cmd);
  377. }
  378. }
  379. void z_shell_vt100_colors_restore(const struct shell *shell,
  380. const struct shell_vt100_colors *color)
  381. {
  382. z_shell_vt100_color_set(shell, color->col);
  383. vt100_bgcolor_set(shell, color->bgcol);
  384. }
  385. void z_shell_vfprintf(const struct shell *shell, enum shell_vt100_color color,
  386. const char *fmt, va_list args)
  387. {
  388. if (IS_ENABLED(CONFIG_SHELL_VT100_COLORS) &&
  389. shell->ctx->internal.flags.use_colors &&
  390. (color != shell->ctx->vt100_ctx.col.col)) {
  391. struct shell_vt100_colors col;
  392. z_shell_vt100_colors_store(shell, &col);
  393. z_shell_vt100_color_set(shell, color);
  394. z_shell_fprintf_fmt(shell->fprintf_ctx, fmt, args);
  395. z_shell_vt100_colors_restore(shell, &col);
  396. } else {
  397. z_shell_fprintf_fmt(shell->fprintf_ctx, fmt, args);
  398. }
  399. }
  400. void z_shell_fprintf(const struct shell *shell,
  401. enum shell_vt100_color color,
  402. const char *fmt, ...)
  403. {
  404. __ASSERT_NO_MSG(shell);
  405. __ASSERT(!k_is_in_isr(), "Thread context required.");
  406. __ASSERT_NO_MSG(shell->ctx);
  407. __ASSERT_NO_MSG(shell->fprintf_ctx);
  408. __ASSERT_NO_MSG(fmt);
  409. va_list args;
  410. va_start(args, fmt);
  411. z_shell_vfprintf(shell, color, fmt, args);
  412. va_end(args);
  413. }