2024-09-25 16:25:35 -04:00
|
|
|
#include <stdint.h>
|
|
|
|
|
2024-09-21 19:57:27 -04:00
|
|
|
#include "alloc.h"
|
2024-09-25 15:31:44 -04:00
|
|
|
#include "commands.h"
|
2024-09-10 14:03:33 -04:00
|
|
|
#include "kprintf.h"
|
|
|
|
#include "shell.h"
|
|
|
|
#include "string.h"
|
|
|
|
#include "terminal.h"
|
|
|
|
|
2024-09-25 15:31:44 -04:00
|
|
|
#define BORDER "==========================================================="
|
|
|
|
#define HEADER "Welcome to bozOShell - Available Commands"
|
2024-09-10 14:03:33 -04:00
|
|
|
|
2024-09-18 18:05:21 -04:00
|
|
|
struct shell_command {
|
|
|
|
char name[16];
|
|
|
|
char description[256];
|
2024-09-25 15:31:44 -04:00
|
|
|
void (*fn)(char *arg);
|
2024-09-25 16:25:35 -04:00
|
|
|
uint8_t name_len;
|
2024-09-18 18:05:21 -04:00
|
|
|
};
|
|
|
|
|
2024-09-25 16:25:35 -04:00
|
|
|
#define ARRAY_SIZE(array) sizeof(array) / sizeof(array[0])
|
|
|
|
#define COMMAND_ADDER(name, description, function) \
|
|
|
|
{name, description, function, ARRAY_SIZE(name) - 1}
|
|
|
|
|
2024-09-25 15:31:44 -04:00
|
|
|
const struct shell_command commands[] = {
|
2024-09-25 16:25:35 -04:00
|
|
|
COMMAND_ADDER("help", "Print this help menu", help_cmd),
|
|
|
|
COMMAND_ADDER("reboot", "Reboot the system", reboot_cmd),
|
|
|
|
COMMAND_ADDER("poweroff", "Shut down the system", poweroff_cmd),
|
|
|
|
COMMAND_ADDER("halt", "Stop all CPU functions", halt_cmd),
|
|
|
|
COMMAND_ADDER("stack", "Print the stack trace", stack_cmd),
|
|
|
|
COMMAND_ADDER("heap", "Print the heaps", heap_cmd),
|
|
|
|
COMMAND_ADDER("clear", "Clear the current terminal", clear_cmd),
|
|
|
|
COMMAND_ADDER("date", "Display the current time and date", date_cmd),
|
|
|
|
COMMAND_ADDER("merdella", "Surprise", merdella_cmd),
|
|
|
|
COMMAND_ADDER("color", "Change the screen color", color_cmd),
|
2024-09-18 18:05:21 -04:00
|
|
|
};
|
|
|
|
|
2024-09-25 15:31:44 -04:00
|
|
|
#define NB_CMDS (sizeof(commands) / sizeof(commands[0]))
|
|
|
|
|
|
|
|
extern struct screen *screen;
|
|
|
|
|
|
|
|
void help_cmd(char *arg)
|
2024-09-10 14:03:33 -04:00
|
|
|
{
|
2024-09-18 18:05:21 -04:00
|
|
|
const size_t padding = 15;
|
2024-09-10 14:03:33 -04:00
|
|
|
|
2024-09-25 15:31:44 -04:00
|
|
|
(void)arg;
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf("%s\n", BORDER);
|
|
|
|
kprintf(" %s\n", HEADER);
|
|
|
|
kprintf("%s\n", BORDER);
|
2024-09-18 18:05:21 -04:00
|
|
|
|
2024-09-21 19:57:27 -04:00
|
|
|
for (size_t i = 0; i < NB_CMDS; i++) {
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf(" %s", commands[i].name);
|
2024-09-18 18:05:21 -04:00
|
|
|
for (size_t j = 0; j < (padding - strlen(commands[i].name));
|
|
|
|
j++)
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf(" ");
|
|
|
|
kprintf(": %s\n", commands[i].description);
|
2024-09-18 18:05:21 -04:00
|
|
|
}
|
|
|
|
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf("%s\n", BORDER);
|
2024-09-18 18:05:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void auto_complete(void)
|
|
|
|
{
|
|
|
|
const size_t len = strlen(screen->line);
|
|
|
|
int nb_matches = 0;
|
|
|
|
const char *last_match;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < NB_CMDS; i++) {
|
|
|
|
if (!strncmp(screen->line, commands[i].name, len)) {
|
|
|
|
nb_matches++;
|
|
|
|
last_match = commands[i].name;
|
2024-09-10 14:03:33 -04:00
|
|
|
}
|
|
|
|
}
|
2024-09-18 18:05:21 -04:00
|
|
|
if (nb_matches != 1)
|
|
|
|
return;
|
|
|
|
for (size_t i = len; last_match[i]; i++) {
|
|
|
|
screen->line[i] = last_match[i];
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf("%c", screen->line[i]);
|
2024-09-18 18:05:21 -04:00
|
|
|
}
|
2024-09-10 14:03:33 -04:00
|
|
|
}
|
|
|
|
|
2024-09-11 09:51:40 -04:00
|
|
|
static void read_line(void)
|
2024-09-10 14:03:33 -04:00
|
|
|
{
|
|
|
|
size_t i = 0;
|
|
|
|
struct key_event ev;
|
|
|
|
uint8_t prev_scan_code = 0;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
ev = terminal_getkey();
|
|
|
|
if (ev.c == '\n')
|
|
|
|
break;
|
2024-09-11 09:51:40 -04:00
|
|
|
char *buf = screen->line;
|
|
|
|
i = strlen(screen->line);
|
|
|
|
const size_t size = sizeof(screen->line);
|
2024-09-10 14:03:33 -04:00
|
|
|
if (prev_scan_code != ev.scan_code && ev.scan_code) {
|
|
|
|
if (ev.scan_code == KEY_BACKSPACE && i) {
|
|
|
|
buf[--i] = '\0';
|
|
|
|
move_cursor(LEFT);
|
|
|
|
terminal_putchar(' ');
|
|
|
|
move_cursor(LEFT);
|
2024-09-18 18:05:21 -04:00
|
|
|
} else if (ev.scan_code == KEY_TAB && i) {
|
|
|
|
auto_complete();
|
|
|
|
} else if (ev.c && i < size - 1) {
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf("%c", ev.c);
|
2024-09-10 14:03:33 -04:00
|
|
|
buf[i++] = ev.c;
|
|
|
|
}
|
|
|
|
if (i >= size)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
prev_scan_code = ev.scan_code;
|
|
|
|
}
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf("\n");
|
2024-09-11 09:51:40 -04:00
|
|
|
screen->line[i] = '\0';
|
2024-09-10 14:03:33 -04:00
|
|
|
}
|
|
|
|
|
2024-09-25 15:31:44 -04:00
|
|
|
void shell_init()
|
2024-09-10 14:03:33 -04:00
|
|
|
{
|
|
|
|
while (1) {
|
2024-09-20 06:40:36 -04:00
|
|
|
kprintf(PROMPT);
|
2024-09-11 09:51:40 -04:00
|
|
|
read_line();
|
2024-09-18 18:05:21 -04:00
|
|
|
bool invalid = true;
|
2024-09-21 19:57:27 -04:00
|
|
|
for (unsigned i = 0; i < NB_CMDS; i++) {
|
2024-09-25 16:25:35 -04:00
|
|
|
if (!strncmp(commands[i].name, screen->line,
|
|
|
|
commands[i].name_len)) {
|
|
|
|
commands[i].fn(
|
|
|
|
screen->line + commands[i].name_len +
|
|
|
|
(strlen(screen->line) >
|
|
|
|
commands[i]
|
|
|
|
.name_len)); // "reboot " => arg = ""
|
|
|
|
// not " " and "reboot" =>
|
|
|
|
// "" without invalid read
|
2024-09-18 18:05:21 -04:00
|
|
|
invalid = false;
|
|
|
|
break;
|
|
|
|
}
|
2024-09-10 14:03:33 -04:00
|
|
|
}
|
2024-09-18 18:05:21 -04:00
|
|
|
if (invalid && screen->line[0])
|
2024-09-21 06:17:27 -04:00
|
|
|
kprintf(KERN_WARNING "invalid command: %s\n",
|
|
|
|
screen->line);
|
2024-09-11 09:51:40 -04:00
|
|
|
memset(screen->line, '\0', sizeof(screen->line));
|
2024-09-10 14:03:33 -04:00
|
|
|
}
|
|
|
|
}
|