/* * Copyright (C) 2009, 2020 Freie Universität Berlin * Copyright (C) 2013, INRIA. * Copyright (C) 2015 Kaspar Schleiser * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup sys_shell * @{ * * @file * @brief Implementation of a very simple command interpreter. * For each command (i.e. "echo"), a handler can be specified. * If the first word of a user-entered command line matches the * name of a handler, the handler will be called with the whole * command line as parameter. * * @author Kaspar Schleiser * @author René Kijewski * @author Juan Carrano * @author Hendrik van Essen * * @} */ #include #include #include #include #include #include #include #include "shell.h" #include "shell_commands.h" #define ETX '\x03' /** ASCII "End-of-Text", or ctrl-C */ #if !defined(SHELL_NO_ECHO) || !defined(SHELL_NO_PROMPT) #ifdef MODULE_NEWLIB /* use local copy of putchar, as it seems to be inlined, * enlarging code by 50% */ static void _putchar(int c) { putchar(c); } #else #define _putchar putchar #endif #endif static void flush_if_needed(void) { #ifdef MODULE_NEWLIB fflush(stdout); #endif } static shell_command_handler_t find_handler(const shell_command_t *command_list, char *command) { const shell_command_t *command_lists[] = { command_list, #ifdef MODULE_SHELL_COMMANDS _shell_command_list, #endif }; /* iterating over command_lists */ for (unsigned int i = 0; i < ARRAY_SIZE(command_lists); i++) { const shell_command_t *entry; if ((entry = command_lists[i])) { /* iterating over commands in command_lists entry */ while (entry->name != NULL) { if (strcmp(entry->name, command) == 0) { return entry->handler; } else { entry++; } } } } return NULL; } static void print_help(const shell_command_t *command_list) { printf("%-20s %s\n", "Command", "Description"); puts("---------------------------------------"); const shell_command_t *command_lists[] = { command_list, #ifdef MODULE_SHELL_COMMANDS _shell_command_list, #endif }; /* iterating over command_lists */ for (unsigned int i = 0; i < ARRAY_SIZE(command_lists); i++) { const shell_command_t *entry; if ((entry = command_lists[i])) { /* iterating over commands in command_lists entry */ while (entry->name != NULL) { printf("%-20s %s\n", entry->name, entry->desc); entry++; } } } } static void handle_input_line(const shell_command_t *command_list, char *line) { static const char *INCORRECT_QUOTING = "shell: incorrect quoting"; /* first we need to calculate the number of arguments */ unsigned argc = 0; char *pos = line; int contains_esc_seq = 0; while (1) { if ((unsigned char) *pos > ' ') { /* found an argument */ if (*pos == '"' || *pos == '\'') { /* it's a quoted argument */ const char quote_char = *pos; do { ++pos; if (!*pos) { puts(INCORRECT_QUOTING); return; } else if (*pos == '\\') { /* skip over the next character */ ++contains_esc_seq; ++pos; if (!*pos) { puts(INCORRECT_QUOTING); return; } continue; } } while (*pos != quote_char); if ((unsigned char) pos[1] > ' ') { puts(INCORRECT_QUOTING); return; } } else { /* it's an unquoted argument */ do { if (*pos == '\\') { /* skip over the next character */ ++contains_esc_seq; ++pos; if (!*pos) { puts(INCORRECT_QUOTING); return; } } ++pos; if (*pos == '"') { puts(INCORRECT_QUOTING); return; } } while ((unsigned char) *pos > ' '); } /* count the number of arguments we got */ ++argc; } /* zero out current position (space or quotation mark) and advance */ if (*pos > 0) { *pos = 0; ++pos; } else { break; } } if (!argc) { return; } /* then we fill the argv array */ char *argv[argc + 1]; argv[argc] = NULL; pos = line; for (unsigned i = 0; i < argc; ++i) { while (!*pos) { ++pos; } if (*pos == '"' || *pos == '\'') { ++pos; } argv[i] = pos; while (*pos) { ++pos; } } for (char **arg = argv; contains_esc_seq && *arg; ++arg) { for (char *c = *arg; *c; ++c) { if (*c != '\\') { continue; } for (char *d = c; *d; ++d) { *d = d[1]; } if (--contains_esc_seq == 0) { break; } } } /* then we call the appropriate handler */ shell_command_handler_t handler = find_handler(command_list, argv[0]); if (handler != NULL) { handler(argc, argv); } else { if (strcmp("help", argv[0]) == 0) { print_help(command_list); } else { printf("shell: command not found: %s\n", argv[0]); } } } /** * @brief Read a single line from standard input into a buffer. * * In addition to copying characters, this routine echoes the line back to * stdout and also supports primitive line editing. * * If the input line is too long, the input will still be consumed until the end * to prevent the next line from containing garbage. * * @param buf Buffer where the input will be placed. * @param size Size of the buffer. The maximum line length will be one less * than size, to accommodate for the null terminator. * The minimum buffer size is 1. * * @return length of the read line, excluding the terminator, if reading was * successful. * @return EOF, if the end of the input stream was reached. * @return ENOBUFS if the buffer size was exceeded. */ static int readline(char *buf, size_t size) { int curr_pos = 0; bool length_exceeded = false; assert((size_t) size > 0); while (1) { /* At the start of the loop, cur_pos should point inside of * buf. This ensures the terminator can always fit. */ assert((size_t) curr_pos < size); int c = getchar(); if (c < 0) { return EOF; } /* We allow Unix linebreaks (\n), DOS linebreaks (\r\n), and Mac * linebreaks (\r). QEMU transmits only a single '\r' == 13 on hitting * enter ("-serial stdio"). DOS newlines are handled like hitting enter * twice, but empty lines are ignored. Ctrl-C cancels the current line. */ if (c == '\r' || c == '\n' || c == ETX) { if (c == ETX) { curr_pos = 0; length_exceeded = false; } buf[curr_pos] = '\0'; #ifndef SHELL_NO_ECHO _putchar('\r'); _putchar('\n'); #endif return (length_exceeded) ? -ENOBUFS : curr_pos; } /* check for backspace: * 0x7f (DEL) when using QEMU * 0x08 (BS) for most terminals */ if (c == 0x08 || c == 0x7f) { if (curr_pos == 0) { /* ignore empty line */ continue; } /* after we dropped characters don't edit the line, yet keep the * visual effects */ if (!length_exceeded) { buf[--curr_pos] = '\0'; } /* white-tape the character */ #ifndef SHELL_NO_ECHO _putchar('\b'); _putchar(' '); _putchar('\b'); #endif } else { /* Always consume characters, but do not not always store them */ if ((size_t) curr_pos < size - 1) { buf[curr_pos++] = c; } else { length_exceeded = true; } #ifndef SHELL_NO_ECHO _putchar(c); #endif } flush_if_needed(); } } static inline void print_prompt(void) { #ifndef SHELL_NO_PROMPT _putchar('>'); _putchar(' '); #endif flush_if_needed(); } void shell_run_once(const shell_command_t *shell_commands, char *line_buf, int len) { print_prompt(); while (1) { int res = readline(line_buf, len); switch (res) { case EOF: return; case -ENOBUFS: puts("shell: maximum line length exceeded"); break; default: handle_input_line(shell_commands, line_buf); break; } print_prompt(); } }