/* * Copyright (c) 2023-2024 Ian Marco Moffett and the Osmora Team. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of Hyra nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ /* Maintenance shell */ #include <sys/reboot.h> #include <sys/auxv.h> #include <string.h> #include <stdio.h> #include <fcntl.h> #include <unistd.h> #include <termios.h> #include "mshell.h" #define INPUT_SIZE 32 #define MAX_FILE_SIZE 1024 #define TTY_DEV "/dev/tty1" #define PROMPT "mshell> " struct mshell_state { uint8_t running : 1; char input[INPUT_SIZE]; }; static void help(void) { printf( "MSHELL COMMANDS\n" "\thelp - show this message\n" "\treboot - reboot the system\n" "\ttty - show the current TTY\n" "\tpagesize - get the current page size\n" "\tkversion - get the kernel version\n" "\tmemstat - get info about memory\n" "\tintr - get interrupt information\n" "\texit - exit the shell\n" ); } static void print_file(const char *path) { char buf[MAX_FILE_SIZE]; int fd = open(path, O_RDONLY); int len; if (fd < 0) { printf("Failed to open %s\n", path); return; } len = read(fd, buf, sizeof(buf)); if (len > 0) { buf[len] = '\0'; printf("%s", buf); } close(fd); } static void parse_input(struct mshell_state *state) { char cmd[INPUT_SIZE]; char *p = &state->input[0]; size_t cmd_idx = 0; /* Skip leading whitespace */ while (*p && *p == ' ') ++p; /* Push the command */ while (*p && *p != '\n') cmd[cmd_idx++] = *p++; cmd[cmd_idx] = '\0'; /* Ignore empty commands */ if (cmd_idx == 0) return; if (strcmp(cmd, "reboot") == 0) { reboot(REBOOT_DEFAULT); } else if (strcmp(cmd, "pagesize") == 0) { printf("%d\n", auxv_entry(AT_PAGESIZE)); } else if (strcmp(cmd, "tty") == 0) { printf("%s\n", TTY_DEV); } else if (strcmp(cmd, "help") == 0) { help(); } else if (strcmp(cmd, "exit") == 0) { state->running = 0; } else if (strcmp(cmd, "kversion") == 0) { print_file("/proc/version"); } else if (strcmp(cmd, "memstat") == 0) { print_file("/proc/memstat"); } else if (strcmp(cmd, "intr") == 0) { print_file("/proc/interrupts"); } else { printf("Unknown command '%s'\n", cmd); printf("Use 'help' for help\n"); } } int mshell_enter(void) { int fd, tmp; size_t input_idx; struct termios tm_old, tm; struct mshell_state state = { .running = 1 }; /* Try opening the TTY */ fd = open(TTY_DEV, O_RDONLY); if (fd < 0) return -1; /* Set raw mode */ tcgetattr(fd, &tm_old); tm = tm_old; tm.c_lflag &= ~(ICANON | ECHO); tcsetattr(fd, 0, &tm); printf("%s", PROMPT); input_idx = 0; while (state.running) { if (read(fd, &tmp, 1) <= 0) continue; switch (tmp) { case '\n': /* * Parse the input, the newline is so that any printed * text that is the result of a command can have * its own lines. */ printf("\n"); parse_input(&state); /* Reset the buffer */ input_idx = 0; state.input[0] = '\0'; printf("%s", PROMPT); fflush(stdout); break; case '\b': if (input_idx > 0) { /* * Replace the last char with a '\0' and rewrite * the prompt. */ state.input[--input_idx] = '\0'; printf("\b"); fflush(stdout); continue; } break; default: if (input_idx < INPUT_SIZE) { state.input[input_idx++] = tmp; state.input[input_idx] = '\0'; printf("%c", tmp); fflush(stdout); } break; } } /* Restore TTY state */ tcsetattr(fd, 0, &tm_old); close(fd); return 0; }