shell.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101
  1. /*
  2. * Copyright (c) 2015-2018 Richard Braun.
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *
  17. * Upstream site with license notes :
  18. * http://git.sceen.net/rbraun/librbraun.git/
  19. */
  20. #include <assert.h>
  21. #include <errno.h>
  22. #include <stdarg.h>
  23. #include <stddef.h>
  24. #include <stdio.h>
  25. #include <stdint.h>
  26. #include <string.h>
  27. #include <kern/bulletin.h>
  28. #include <kern/hash.h>
  29. #include <kern/init.h>
  30. #include <kern/log.h>
  31. #include <kern/macros.h>
  32. #include <kern/mutex.h>
  33. #include <kern/printf.h>
  34. #include <kern/shell.h>
  35. #include <kern/thread.h>
  36. #define SHELL_COMPLETION_MATCH_FMT "-16s"
  37. #define SHELL_COMPLETION_NR_MATCHES_PER_LINE 4
  38. /*
  39. * Escape sequence states.
  40. *
  41. * Here is an incomplete description of escape sequences :
  42. * http://en.wikipedia.org/wiki/ANSI_escape_code
  43. *
  44. * These values must be different from 0.
  45. */
  46. #define SHELL_ESC_STATE_START 1
  47. #define SHELL_ESC_STATE_CSI 2
  48. typedef void (*shell_esc_seq_fn) (struct shell *);
  49. struct shell_esc_seq
  50. {
  51. const char *str;
  52. shell_esc_seq_fn fn;
  53. };
  54. #define SHELL_SEPARATOR ' '
  55. /*
  56. * Commonly used backspace control characters.
  57. *
  58. * XXX Adjust for your needs.
  59. */
  60. #define SHELL_ERASE_BS '\b'
  61. #define SHELL_ERASE_DEL '\x7f'
  62. static struct bulletin_sub shell_log_bulletin_sub;
  63. static const char*
  64. shell_find_word (const char *str)
  65. {
  66. for ( ; ; ++str)
  67. if (!*str || *str != SHELL_SEPARATOR)
  68. return (str);
  69. }
  70. void
  71. shell_cmd_init (struct shell_cmd *cmd, const char *name,
  72. shell_fn_t fn, const char *usage,
  73. const char *short_desc, const char *long_desc)
  74. {
  75. cmd->ht_next = NULL;
  76. cmd->ls_next = NULL;
  77. cmd->name = name;
  78. cmd->fn = fn;
  79. cmd->usage = usage;
  80. cmd->short_desc = short_desc;
  81. cmd->long_desc = long_desc;
  82. }
  83. static const char*
  84. shell_cmd_name (const struct shell_cmd *cmd)
  85. {
  86. return (cmd->name);
  87. }
  88. static int
  89. shell_cmd_check_char (char c)
  90. {
  91. if ((c >= 'a' && c <= 'z') ||
  92. (c >= 'A' && c <= 'Z') ||
  93. (c >= '0' && c <= '9') ||
  94. c == '-' || c == '_')
  95. return (0);
  96. return (EINVAL);
  97. }
  98. static int
  99. shell_cmd_check (const struct shell_cmd *cmd)
  100. {
  101. size_t len = strlen (cmd->name);
  102. if (! len)
  103. return (EINVAL);
  104. for (size_t i = 0; i < len; i++)
  105. {
  106. int error = shell_cmd_check_char (cmd->name[i]);
  107. if (error)
  108. return (error);
  109. }
  110. return (0);
  111. }
  112. static const char*
  113. shell_line_str (const struct shell_line *line)
  114. {
  115. return (line->str);
  116. }
  117. static size_t
  118. shell_line_size (const struct shell_line *line)
  119. {
  120. return (line->size);
  121. }
  122. static void
  123. shell_line_reset (struct shell_line *line)
  124. {
  125. line->str[0] = '\0';
  126. line->size = 0;
  127. }
  128. static void
  129. shell_line_copy (struct shell_line *dest, const struct shell_line *src)
  130. {
  131. strcpy (dest->str, src->str);
  132. dest->size = src->size;
  133. }
  134. static int
  135. shell_line_cmp (const struct shell_line *a, const struct shell_line *b)
  136. {
  137. return (strcmp (a->str, b->str));
  138. }
  139. static int
  140. shell_line_insert (struct shell_line *line, size_t index, char c)
  141. {
  142. if (index > line->size)
  143. return (EINVAL);
  144. else if (line->size + 1 == sizeof (line->str))
  145. return (ENOMEM);
  146. size_t remaining_chars = line->size - index;
  147. if (remaining_chars)
  148. memmove (&line->str[index + 1], &line->str[index], remaining_chars);
  149. line->str[index] = c;
  150. line->str[++line->size] = '\0';
  151. return (0);
  152. }
  153. static int
  154. shell_line_erase (struct shell_line *line, size_t index)
  155. {
  156. if (index >= line->size)
  157. return (EINVAL);
  158. size_t remaining_chars = line->size - index - 1;
  159. if (remaining_chars)
  160. memmove (&line->str[index], &line->str[index + 1], remaining_chars);
  161. line->str[--line->size] = '\0';
  162. return (0);
  163. }
  164. static struct shell_line*
  165. shell_history_get (struct shell_history *history, size_t index)
  166. {
  167. return (&history->lines[index % ARRAY_SIZE (history->lines)]);
  168. }
  169. static void
  170. shell_history_init (struct shell_history *history)
  171. {
  172. for (size_t i = 0; i < ARRAY_SIZE (history->lines); i++)
  173. shell_line_reset (shell_history_get (history, i));
  174. history->newest = 0;
  175. history->oldest = 0;
  176. history->index = 0;
  177. }
  178. static struct shell_line*
  179. shell_history_get_newest (struct shell_history *history)
  180. {
  181. return (shell_history_get (history, history->newest));
  182. }
  183. static struct shell_line *
  184. shell_history_get_index (struct shell_history *history)
  185. {
  186. return (shell_history_get (history, history->index));
  187. }
  188. static void
  189. shell_history_reset_index (struct shell_history *history)
  190. {
  191. history->index = history->newest;
  192. }
  193. static int
  194. shell_history_same_newest (struct shell_history *history)
  195. {
  196. return (history->newest != history->oldest &&
  197. shell_line_cmp (shell_history_get_newest (history),
  198. shell_history_get (history, history->newest - 1))
  199. == 0);
  200. }
  201. static void
  202. shell_history_push (struct shell_history *history)
  203. {
  204. if (shell_line_size (shell_history_get_newest (history)) == 0 ||
  205. shell_history_same_newest (history))
  206. {
  207. shell_history_reset_index (history);
  208. return;
  209. }
  210. ++history->newest;
  211. shell_history_reset_index (history);
  212. // Mind integer overflows.
  213. if (history->newest - history->oldest >= ARRAY_SIZE (history->lines))
  214. history->oldest = history->newest - ARRAY_SIZE (history->lines) + 1;
  215. }
  216. static void
  217. shell_history_back (struct shell_history *history)
  218. {
  219. if (history->index == history->oldest)
  220. return;
  221. --history->index;
  222. shell_line_copy (shell_history_get_newest (history),
  223. shell_history_get_index (history));
  224. }
  225. static void
  226. shell_history_forward (struct shell_history *history)
  227. {
  228. if (history->index == history->newest)
  229. return;
  230. else if (++history->index == history->newest)
  231. shell_line_reset (shell_history_get_newest (history));
  232. else
  233. shell_line_copy (shell_history_get_newest (history),
  234. shell_history_get_index (history));
  235. }
  236. static void
  237. shell_history_print (struct shell_history *history, struct shell *shell)
  238. {
  239. // Mind integer overflows.
  240. for (size_t i = history->oldest; i != history->newest; i++)
  241. {
  242. _Auto line = shell_history_get (history, i);
  243. shell_printf (shell, "%6zu %s\n",
  244. (size_t) (i - history->oldest),
  245. shell_line_str (line));
  246. }
  247. }
  248. static void
  249. shell_cmd_set_lock (struct shell_cmd_set *cmd_set)
  250. {
  251. mutex_lock (&cmd_set->lock);
  252. }
  253. static void
  254. shell_cmd_set_unlock (struct shell_cmd_set *cmd_set)
  255. {
  256. mutex_unlock (&cmd_set->lock);
  257. }
  258. static struct shell_bucket*
  259. shell_cmd_set_get_bucket (struct shell_cmd_set *cmd_set, const char *name)
  260. {
  261. size_t index = hash_str (name, SHELL_HTABLE_BITS);
  262. assert (index < ARRAY_SIZE (cmd_set->htable));
  263. return (&cmd_set->htable[index]);
  264. }
  265. static const struct shell_cmd*
  266. shell_cmd_set_lookup (struct shell_cmd_set *cmd_set, const char *name)
  267. {
  268. shell_cmd_set_lock (cmd_set);
  269. const _Auto bucket = shell_cmd_set_get_bucket (cmd_set, name);
  270. _Auto cmd = bucket->cmd;
  271. for (; cmd != NULL; cmd = cmd->ht_next)
  272. if (strcmp (cmd->name, name) == 0)
  273. break;
  274. shell_cmd_set_unlock (cmd_set);
  275. return (cmd);
  276. }
  277. /*
  278. * Look up the first command that matches a given string.
  279. *
  280. * The input string is defined by the given string pointer and size.
  281. *
  282. * The global lock must be acquired before calling this function.
  283. */
  284. static const struct shell_cmd*
  285. shell_cmd_set_match (const struct shell_cmd_set *cmd_set, const char *str,
  286. size_t size)
  287. {
  288. const struct shell_cmd *cmd = cmd_set->cmd_list;
  289. while (cmd)
  290. {
  291. if (strncmp (cmd->name, str, size) == 0)
  292. return (cmd);
  293. cmd = cmd->ls_next;
  294. }
  295. return (NULL);
  296. }
  297. /*
  298. * Attempt command auto-completion.
  299. *
  300. * The given string is the beginning of a command, or the empty string.
  301. * The sizep parameter initially points to the size of the given string.
  302. * If the string matches any registered command, the cmdp pointer is
  303. * updated to point to the first matching command in the sorted list of
  304. * commands, and sizep is updated to the number of characters in the
  305. * command name that are common in subsequent commands. The command
  306. * pointer and the returned size can be used to print a list of commands
  307. * eligible for completion.
  308. *
  309. * If there is a single match for the given string, return 0. If there
  310. * are more than one match, return EAGAIN. If there is no match,
  311. * return EINVAL.
  312. */
  313. static int
  314. shell_cmd_set_complete (struct shell_cmd_set *cmd_set, const char *str,
  315. size_t *sizep, const struct shell_cmd **cmdp)
  316. {
  317. size_t size = *sizep;
  318. /*
  319. * Start with looking up a command that matches the given argument.
  320. * If there is no match, return an error.
  321. */
  322. const _Auto cmd = shell_cmd_set_match (cmd_set, str, size);
  323. if (! cmd)
  324. return (EINVAL);
  325. *cmdp = cmd;
  326. /*
  327. * If at least one command matches, try to complete it.
  328. * There can be two cases :
  329. * 1/ There is one and only one match, which is directly returned.
  330. * 2/ There are several matches, in which case the common length is
  331. * computed.
  332. */
  333. const struct shell_cmd *next = cmd->ls_next;
  334. if (!next || strncmp (cmd->name, next->name, size))
  335. {
  336. *sizep = strlen (cmd->name);
  337. return (0);
  338. }
  339. /*
  340. * When computing the common length, all the commands that can match
  341. * must be evaluated. Considering the current command is the first
  342. * that can match, the only other variable missing is the last
  343. * command that can match.
  344. */
  345. for (; next->ls_next; next = next->ls_next)
  346. if (strncmp (cmd->name, next->ls_next->name, size))
  347. break;
  348. if (! size)
  349. size = 1;
  350. while (cmd->name[size - 1] != '\0' &&
  351. cmd->name[size - 1] == next->name[size - 1])
  352. size++;
  353. *sizep = --size;
  354. return (EAGAIN);
  355. }
  356. struct shell_cmd_set*
  357. shell_get_cmd_set (struct shell *shell)
  358. {
  359. return (shell->cmd_set);
  360. }
  361. static struct shell_history*
  362. shell_get_history (struct shell *shell)
  363. {
  364. return (&shell->history);
  365. }
  366. static void
  367. shell_cb_help (struct shell *shell, int argc, char *argv[])
  368. {
  369. _Auto cmd_set = shell_get_cmd_set (shell);
  370. if (argc > 2)
  371. {
  372. argc = 2;
  373. argv[1] = "help";
  374. }
  375. if (argc == 2)
  376. {
  377. _Auto cmd = shell_cmd_set_lookup (cmd_set, argv[1]);
  378. if (! cmd)
  379. {
  380. shell_printf (shell, "shell: help: %s: command not found\n",
  381. argv[1]);
  382. return;
  383. }
  384. shell_printf (shell, "usage: %s\n%s\n", cmd->usage, cmd->short_desc);
  385. if (cmd->long_desc)
  386. shell_printf (shell, "\n%s\n", cmd->long_desc);
  387. return;
  388. }
  389. shell_cmd_set_lock (cmd_set);
  390. for (_Auto cmd = cmd_set->cmd_list; cmd; cmd = cmd->ls_next)
  391. shell_printf (shell, "%19s %s\n", cmd->name, cmd->short_desc);
  392. shell_cmd_set_unlock (cmd_set);
  393. }
  394. static void
  395. shell_cb_history (struct shell *shell, int argc __unused, char **argv __unused)
  396. {
  397. shell_history_print (shell_get_history (shell), shell);
  398. }
  399. static struct shell_cmd shell_default_cmds[] =
  400. {
  401. SHELL_CMD_INITIALIZER ("help", shell_cb_help,
  402. "help [command]",
  403. "obtain help about shell commands"),
  404. SHELL_CMD_INITIALIZER ("history", shell_cb_history,
  405. "history",
  406. "display history list"),
  407. };
  408. void
  409. shell_cmd_set_init (struct shell_cmd_set *cmd_set)
  410. {
  411. mutex_init (&cmd_set->lock);
  412. memset (cmd_set->htable, 0, sizeof (cmd_set->htable));
  413. cmd_set->cmd_list = NULL;
  414. SHELL_REGISTER_CMDS (shell_default_cmds, cmd_set);
  415. }
  416. static int
  417. shell_cmd_set_add_htable (struct shell_cmd_set *cmd_set, struct shell_cmd *cmd)
  418. {
  419. _Auto bucket = shell_cmd_set_get_bucket (cmd_set, cmd->name);
  420. _Auto tmp = bucket->cmd;
  421. if (! tmp)
  422. {
  423. bucket->cmd = cmd;
  424. return (0);
  425. }
  426. for ( ; ; tmp = tmp->ht_next)
  427. if (strcmp (cmd->name, tmp->name) == 0)
  428. return (EEXIST);
  429. else if (!tmp->ht_next)
  430. break;
  431. tmp->ht_next = cmd;
  432. return (0);
  433. }
  434. static void
  435. shell_cmd_set_add_list (struct shell_cmd_set *cmd_set, struct shell_cmd *cmd)
  436. {
  437. _Auto prev = cmd_set->cmd_list;
  438. if (!prev || strcmp (cmd->name, prev->name) < 0)
  439. {
  440. cmd_set->cmd_list = cmd;
  441. cmd->ls_next = prev;
  442. return;
  443. }
  444. struct shell_cmd *next;
  445. while (1)
  446. {
  447. next = prev->ls_next;
  448. if (!next || strcmp (cmd->name, next->name) < 0)
  449. break;
  450. prev = next;
  451. }
  452. prev->ls_next = cmd;
  453. cmd->ls_next = next;
  454. }
  455. static int
  456. shell_cmd_set_add (struct shell_cmd_set *cmd_set, struct shell_cmd *cmd)
  457. {
  458. int error = shell_cmd_set_add_htable (cmd_set, cmd);
  459. if (error)
  460. return (error);
  461. shell_cmd_set_add_list (cmd_set, cmd);
  462. return (0);
  463. }
  464. int
  465. shell_cmd_set_register (struct shell_cmd_set *cmd_set, struct shell_cmd *cmd)
  466. {
  467. int error = shell_cmd_check (cmd);
  468. if (error)
  469. return (error);
  470. shell_cmd_set_lock (cmd_set);
  471. error = shell_cmd_set_add (cmd_set, cmd);
  472. shell_cmd_set_unlock (cmd_set);
  473. return (error);
  474. }
  475. void
  476. shell_init (struct shell *shell, struct shell_cmd_set *cmd_set,
  477. struct stream *stream)
  478. {
  479. shell->cmd_set = cmd_set;
  480. shell->stream = stream;
  481. shell_history_init (&shell->history);
  482. shell->esc_seq_index = 0;
  483. }
  484. static void
  485. shell_prompt (struct shell *shell)
  486. {
  487. shell_printf (shell, "shell> ");
  488. }
  489. static void
  490. shell_reset (struct shell *shell)
  491. {
  492. shell_line_reset (shell_history_get_newest (&shell->history));
  493. shell->cursor = 0;
  494. shell_prompt (shell);
  495. }
  496. static void
  497. shell_erase (struct shell *shell)
  498. {
  499. _Auto current_line = shell_history_get_newest (&shell->history);
  500. size_t remaining_chars = shell_line_size (current_line);
  501. while (shell->cursor != remaining_chars)
  502. {
  503. shell_printf (shell, " ");
  504. shell->cursor++;
  505. }
  506. while (remaining_chars != 0)
  507. {
  508. shell_printf (shell, "\b \b");
  509. remaining_chars--;
  510. }
  511. shell->cursor = 0;
  512. }
  513. static void
  514. shell_restore (struct shell *shell)
  515. {
  516. _Auto current_line = shell_history_get_newest (&shell->history);
  517. shell_printf (shell, "%s", shell_line_str (current_line));
  518. shell->cursor = shell_line_size (current_line);
  519. }
  520. static int
  521. shell_is_ctrl_char (char c)
  522. {
  523. return (c < ' ' || c >= 0x7f);
  524. }
  525. static void
  526. shell_process_left (struct shell *shell)
  527. {
  528. if (!shell->cursor)
  529. return;
  530. --shell->cursor;
  531. shell_printf (shell, "\e[1D");
  532. }
  533. static int
  534. shell_process_right (struct shell *shell)
  535. {
  536. size_t size = shell_line_size (shell_history_get_newest (&shell->history));
  537. if (shell->cursor >= size)
  538. return (EAGAIN);
  539. ++shell->cursor;
  540. shell_printf (shell, "\e[1C");
  541. return (0);
  542. }
  543. static void
  544. shell_process_up (struct shell *shell)
  545. {
  546. shell_erase (shell);
  547. shell_history_back (&shell->history);
  548. shell_restore (shell);
  549. }
  550. static void
  551. shell_process_down (struct shell *shell)
  552. {
  553. shell_erase (shell);
  554. shell_history_forward (&shell->history);
  555. shell_restore (shell);
  556. }
  557. static void
  558. shell_process_backspace (struct shell *shell)
  559. {
  560. _Auto current_line = shell_history_get_newest (&shell->history);
  561. int error = shell_line_erase (current_line, shell->cursor - 1);
  562. if (error)
  563. return;
  564. --shell->cursor;
  565. shell_printf (shell, "\b%s ", shell_line_str (current_line) + shell->cursor);
  566. size_t remaining_chars = shell_line_size (current_line) - shell->cursor + 1;
  567. for (; remaining_chars; --remaining_chars)
  568. shell_printf (shell, "\b");
  569. }
  570. static int
  571. shell_process_raw_char (struct shell *shell, char c)
  572. {
  573. _Auto current_line = shell_history_get_newest (&shell->history);
  574. int error = shell_line_insert (current_line, shell->cursor, c);
  575. if (error)
  576. {
  577. shell_printf (shell, "\nshell: line too long\n");
  578. return (error);
  579. }
  580. else if (++shell->cursor == shell_line_size (current_line))
  581. {
  582. shell_printf (shell, "%c", c);
  583. return (0);
  584. }
  585. /*
  586. * This assumes that the backspace character only moves the cursor
  587. * without erasing characters.
  588. */
  589. shell_printf (shell, "%s",
  590. shell_line_str (current_line) + shell->cursor - 1);
  591. size_t remaining_chars = shell_line_size (current_line) - shell->cursor;
  592. for (; remaining_chars; --remaining_chars)
  593. shell_printf (shell, "\b");
  594. return (0);
  595. }
  596. /*
  597. * Print a list of commands eligible for completion, starting at the
  598. * given command. Other eligible commands share the same prefix, as
  599. * defined by the size argument.
  600. *
  601. * The global lock must be acquired before calling this function.
  602. */
  603. static void
  604. shell_print_cmd_matches (struct shell *shell, const struct shell_cmd *cmd,
  605. size_t size)
  606. {
  607. shell_printf (shell, "\n");
  608. const struct shell_cmd *tmp;
  609. size_t i;
  610. for (tmp = cmd, i = 1; tmp != NULL; tmp = tmp->ls_next, i++)
  611. {
  612. if (strncmp (cmd->name, tmp->name, size) != 0)
  613. break;
  614. shell_printf (shell, "%" SHELL_COMPLETION_MATCH_FMT, tmp->name);
  615. if ((i % SHELL_COMPLETION_NR_MATCHES_PER_LINE) == 0)
  616. shell_printf (shell, "\n");
  617. }
  618. if ((i % SHELL_COMPLETION_NR_MATCHES_PER_LINE) != 1)
  619. shell_printf (shell, "\n");
  620. }
  621. static int
  622. shell_process_tabulation (struct shell *shell)
  623. {
  624. const struct shell_cmd *cmd = NULL; /* GCC */
  625. _Auto cmd_set = shell->cmd_set;
  626. shell_cmd_set_lock (cmd_set);
  627. _Auto str = shell_line_str (shell_history_get_newest (&shell->history));
  628. const char *word = shell_find_word (str);
  629. size_t size = shell->cursor - (word - str),
  630. cmd_cursor = shell->cursor - size;
  631. int error = shell_cmd_set_complete (cmd_set, word, &size, &cmd);
  632. if (error && error != EAGAIN)
  633. {
  634. error = 0;
  635. goto out;
  636. }
  637. else if (error == EAGAIN)
  638. {
  639. size_t cursor = shell->cursor;
  640. shell_print_cmd_matches (shell, cmd, size);
  641. shell_prompt (shell);
  642. shell_restore (shell);
  643. // Keep existing arguments as they are.
  644. while (shell->cursor != cursor)
  645. shell_process_left (shell);
  646. }
  647. const char *name = shell_cmd_name (cmd);
  648. while (shell->cursor != cmd_cursor)
  649. shell_process_backspace (shell);
  650. for (size_t i = 0; i < size; i++)
  651. {
  652. error = shell_process_raw_char (shell, name[i]);
  653. if (error)
  654. goto out;
  655. }
  656. error = 0;
  657. out:
  658. shell_cmd_set_unlock (cmd_set);
  659. return (error);
  660. }
  661. static void
  662. shell_esc_seq_up (struct shell *shell)
  663. {
  664. shell_process_up (shell);
  665. }
  666. static void
  667. shell_esc_seq_down (struct shell *shell)
  668. {
  669. shell_process_down (shell);
  670. }
  671. static void
  672. shell_esc_seq_next (struct shell *shell)
  673. {
  674. shell_process_right (shell);
  675. }
  676. static void
  677. shell_esc_seq_prev (struct shell *shell)
  678. {
  679. shell_process_left (shell);
  680. }
  681. static void
  682. shell_esc_seq_home (struct shell *shell)
  683. {
  684. while (shell->cursor)
  685. shell_process_left (shell);
  686. }
  687. static void
  688. shell_esc_seq_del (struct shell *shell)
  689. {
  690. if (shell_process_right (shell) == 0)
  691. shell_process_backspace (shell);
  692. }
  693. static void
  694. shell_esc_seq_end (struct shell *shell)
  695. {
  696. size_t size = shell_line_size (shell_history_get_newest (&shell->history));
  697. while (shell->cursor < size)
  698. shell_process_right (shell);
  699. }
  700. static const struct shell_esc_seq shell_esc_seqs[] =
  701. {
  702. { "A", shell_esc_seq_up },
  703. { "B", shell_esc_seq_down },
  704. { "C", shell_esc_seq_next },
  705. { "D", shell_esc_seq_prev },
  706. { "H", shell_esc_seq_home },
  707. { "1~", shell_esc_seq_home },
  708. { "3~", shell_esc_seq_del },
  709. { "F", shell_esc_seq_end },
  710. { "4~", shell_esc_seq_end },
  711. };
  712. static const struct shell_esc_seq*
  713. shell_esc_seq_lookup (const char *str)
  714. {
  715. for (size_t i = 0; i < ARRAY_SIZE (shell_esc_seqs); i++)
  716. if (strcmp (shell_esc_seqs[i].str, str) == 0)
  717. return (&shell_esc_seqs[i]);
  718. return (NULL);
  719. }
  720. /*
  721. * Process a single escape sequence character.
  722. *
  723. * Return the next escape state or 0 if the sequence is complete.
  724. */
  725. static int
  726. shell_process_esc_sequence (struct shell *shell, char c)
  727. {
  728. if (shell->esc_seq_index >= (ARRAY_SIZE (shell->esc_seq) - 1))
  729. {
  730. shell_printf (shell, "shell: escape sequence too long\n");
  731. goto reset;
  732. }
  733. shell->esc_seq[shell->esc_seq_index++] = c;
  734. shell->esc_seq[shell->esc_seq_index] = '\0';
  735. if (c >= '@' && c <= '~')
  736. {
  737. _Auto seq = shell_esc_seq_lookup (shell->esc_seq);
  738. if (seq)
  739. seq->fn (shell);
  740. goto reset;
  741. }
  742. return (SHELL_ESC_STATE_CSI);
  743. reset:
  744. shell->esc_seq_index = 0;
  745. return (0);
  746. }
  747. static int
  748. shell_process_args (struct shell *shell)
  749. {
  750. snprintf (shell->tmp_line, sizeof (shell->tmp_line), "%s",
  751. shell_line_str (shell_history_get_newest (&shell->history)));
  752. char c, prev;
  753. size_t i;
  754. int j;
  755. for (i = 0, j = 0, prev = SHELL_SEPARATOR;
  756. (c = shell->tmp_line[i]) != '\0';
  757. i++, prev = c)
  758. {
  759. if (c == SHELL_SEPARATOR)
  760. {
  761. if (prev != SHELL_SEPARATOR)
  762. shell->tmp_line[i] = '\0';
  763. }
  764. else
  765. {
  766. if (prev == SHELL_SEPARATOR)
  767. {
  768. shell->argv[j] = &shell->tmp_line[i];
  769. j++;
  770. if (j == ARRAY_SIZE (shell->argv))
  771. {
  772. shell_printf (shell, "shell: too many arguments\n");
  773. return (EINVAL);
  774. }
  775. shell->argv[j] = NULL;
  776. }
  777. }
  778. }
  779. shell->argc = j;
  780. return (0);
  781. }
  782. static void
  783. shell_process_line (struct shell *shell)
  784. {
  785. const struct shell_cmd *cmd = NULL;
  786. int error = shell_process_args (shell);
  787. if (error || shell->argc == 0)
  788. goto out;
  789. cmd = shell_cmd_set_lookup (shell->cmd_set, shell->argv[0]);
  790. if (! cmd)
  791. {
  792. shell_printf (shell, "shell: %s: command not found\n", shell->argv[0]);
  793. goto out;
  794. }
  795. out:
  796. shell_history_push (&shell->history);
  797. if (cmd)
  798. cmd->fn (shell, shell->argc, shell->argv);
  799. }
  800. /*
  801. * Process a single control character.
  802. *
  803. * Return an error if the caller should reset the current line state.
  804. */
  805. static int
  806. shell_process_ctrl_char (struct shell *shell, char c)
  807. {
  808. switch (c)
  809. {
  810. case SHELL_ERASE_BS:
  811. case SHELL_ERASE_DEL:
  812. shell_process_backspace (shell);
  813. break;
  814. case '\t':
  815. return (shell_process_tabulation (shell));
  816. case '\n':
  817. case '\r':
  818. shell_printf (shell, "\n");
  819. shell_process_line (shell);
  820. return (EAGAIN);
  821. default:
  822. return (0);
  823. }
  824. return (0);
  825. }
  826. static void
  827. shell_run (struct shell *shell)
  828. {
  829. while (1)
  830. {
  831. shell_reset (shell);
  832. int error, escape = 0;
  833. while (1)
  834. {
  835. int c = stream_getc (shell->stream);
  836. if (escape)
  837. {
  838. switch (escape)
  839. {
  840. case SHELL_ESC_STATE_START:
  841. // XXX CSI and SS3 sequence processing is the same.
  842. if (c == '[' || c == 'O')
  843. escape = SHELL_ESC_STATE_CSI;
  844. else
  845. escape = 0;
  846. break;
  847. case SHELL_ESC_STATE_CSI:
  848. escape = shell_process_esc_sequence (shell, c);
  849. break;
  850. default:
  851. escape = 0;
  852. }
  853. error = 0;
  854. }
  855. else if (shell_is_ctrl_char (c))
  856. {
  857. if (c == '\e')
  858. {
  859. escape = SHELL_ESC_STATE_START;
  860. error = 0;
  861. }
  862. else
  863. {
  864. error = shell_process_ctrl_char (shell, c);
  865. if (error)
  866. break;
  867. }
  868. }
  869. else
  870. error = shell_process_raw_char (shell, c);
  871. if (error)
  872. break;
  873. }
  874. }
  875. }
  876. void
  877. shell_printf (struct shell *shell, const char *format, ...)
  878. {
  879. va_list ap;
  880. va_start (ap, format);
  881. shell_vprintf (shell, format, ap);
  882. va_end (ap);
  883. }
  884. void
  885. shell_vprintf (struct shell *shell, const char *format, va_list ap)
  886. {
  887. fmt_vxprintf (shell->stream, format, ap);
  888. }
  889. static struct shell_cmd_set shell_main_cmd_set;
  890. static struct shell shell_main;
  891. static void
  892. shell_main_run (void *arg)
  893. {
  894. shell_run (arg);
  895. }
  896. static void
  897. shell_start (uintptr_t value __unused, void *arg __unused)
  898. {
  899. struct thread_attr attr;
  900. thread_attr_init (&attr, THREAD_KERNEL_PREFIX "shell");
  901. thread_attr_set_detached (&attr);
  902. struct thread *thread;
  903. int error = thread_create (&thread, &attr, shell_main_run, &shell_main);
  904. error_check (error, "thread_create");
  905. }
  906. static int __init
  907. shell_setup (void)
  908. {
  909. shell_cmd_set_init (&shell_main_cmd_set);
  910. shell_init (&shell_main, &shell_main_cmd_set, console_stream);
  911. bulletin_subscribe (log_get_bulletin (), &shell_log_bulletin_sub,
  912. shell_start, NULL);
  913. return (0);
  914. }
  915. INIT_OP_DEFINE (shell_setup,
  916. INIT_OP_DEP (log_setup, true),
  917. INIT_OP_DEP (mutex_setup, true),
  918. INIT_OP_DEP (printf_setup, true),
  919. INIT_OP_DEP (stream_setup, true));
  920. struct shell_cmd_set* __init
  921. shell_get_main_cmd_set (void)
  922. {
  923. return (&shell_main_cmd_set);
  924. }