The numeric value for EOF is -1. This caused the shell to return the same code when EOF was encountered and when the line lenght was exceeded. Additionally, if the line length is exceeded, the correct behaviour is to consume the remaining characters until the end of the line, to prevent the following line from containing (potentially dangerous) garbage. Co-authored-by: Hendrik van Essen <hendrik.ve@fu-berlin.de>
361 lines
9.7 KiB
C
361 lines
9.7 KiB
C
/*
|
|
* Copyright (C) 2009, 2020 Freie Universität Berlin
|
|
* Copyright (C) 2013, INRIA.
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
*
|
|
* 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 <kaspar@schleiser.de>
|
|
* @author René Kijewski <rene.kijewski@fu-berlin.de>
|
|
* @author Juan Carrano <j.carrano@fu-berlin.de>
|
|
* @author Hendrik van Essen <hendrik.ve@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <stdbool.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
|
|
#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 the 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;
|
|
}
|
|
|
|
/* QEMU uses 0x7f (DEL) as backspace, while 0x08 (BS) is for most terminals */
|
|
if (c == 0x08 || c == 0x7f) {
|
|
if (curr_pos == 0) {
|
|
/* The line is empty. */
|
|
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();
|
|
}
|
|
}
|