limine/stage2/menu.c

203 lines
5.8 KiB
C
Raw Normal View History

2020-06-05 21:27:52 +03:00
#include <stdint.h>
#include <stddef.h>
#include <stdbool.h>
2020-08-29 21:02:16 +03:00
#include <limine.h>
2020-06-05 21:27:52 +03:00
#include <menu.h>
#include <lib/print.h>
#include <lib/blib.h>
#include <lib/libc.h>
#include <lib/config.h>
2020-09-02 10:55:56 +03:00
#include <lib/term.h>
2020-10-27 10:09:27 +03:00
#include <lib/readline.h>
2020-09-20 13:03:44 +03:00
#include <mm/pmm.h>
#include <drivers/vbe.h>
2020-06-05 21:27:52 +03:00
static char *cmdline;
#define CMDLINE_MAX 1024
static char config_entry_name[1024];
2020-09-26 04:56:44 +03:00
char *menu(int boot_drive) {
cmdline = conv_mem_alloc(CMDLINE_MAX);
2020-09-26 04:56:44 +03:00
char buf[16];
int selected_entry = 0;
if (config_get_value(buf, 0, 16, "DEFAULT_ENTRY")) {
selected_entry = (int)strtoui(buf);
}
int timeout = 5;
if (config_get_value(buf, 0, 16, "TIMEOUT")) {
timeout = (int)strtoui(buf);
}
if (!timeout)
goto autoboot;
// If there is GRAPHICS config key and the value is "yes", enable graphics
if (config_get_value(buf, 0, 16, "GRAPHICS") && !strcmp(buf, "yes")) {
2020-09-26 04:56:44 +03:00
// default scheme
int margin = 64;
2020-09-30 18:29:07 +03:00
int margin_gradient = 20;
2020-09-26 04:56:44 +03:00
uint32_t colourscheme[] = {
0x00000000, // black
0x00aa0000, // red
0x0000aa00, // green
0x00aa5500, // brown
0x000000aa, // blue
0x00aa00aa, // magenta
0x0000aaaa, // cyan
0x00aaaaaa // grey
};
if (config_get_value(buf, 0, 16, "THEME_BLACK")) {
colourscheme[0] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_RED")) {
colourscheme[1] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_GREEN")) {
colourscheme[2] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_BROWN")) {
colourscheme[3] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_BLUE")) {
colourscheme[4] = (int)strtoui16(buf);
}
2020-09-25 00:18:51 +03:00
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_MAGENTA")) {
colourscheme[5] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_CYAN")) {
colourscheme[6] = (int)strtoui16(buf);
}
2020-06-05 21:27:52 +03:00
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_GREY")) {
colourscheme[7] = (int)strtoui16(buf);
}
2020-09-26 04:56:44 +03:00
if (config_get_value(buf, 0, 16, "THEME_MARGIN")) {
margin = (int)strtoui(buf);
}
2020-09-30 18:29:07 +03:00
if (config_get_value(buf, 0, 16, "THEME_MARGIN_GRADIENT")) {
margin_gradient = (int)strtoui(buf);
}
int bg_drive;
if (!config_get_value(buf, 0, 16, "BACKGROUND_DRIVE")) {
bg_drive = boot_drive;
} else {
bg_drive = (int)strtoui(buf);
}
int bg_part;
if (!config_get_value(buf, 0, 16, "BACKGROUND_PARTITION")) {
goto nobg;
} else {
bg_part = (int)strtoui(buf);
}
if (!config_get_value(cmdline, 0, CMDLINE_MAX, "BACKGROUND_PATH"))
goto nobg;
struct file_handle *bg_file = conv_mem_alloc(sizeof(struct file_handle));
if (fopen(bg_file, bg_drive, bg_part, cmdline))
goto nobg;
struct image *bg = conv_mem_alloc(sizeof(struct image));
if (open_image(bg, bg_file))
goto nobg;
2020-09-30 18:29:07 +03:00
term_vbe(colourscheme, margin, margin_gradient, bg);
goto yesbg;
nobg:
2020-09-30 18:29:07 +03:00
term_vbe(colourscheme, margin, margin_gradient, NULL);
yesbg:;
2020-06-05 21:27:52 +03:00
}
2020-09-02 10:55:56 +03:00
disable_cursor();
2020-06-05 21:27:52 +03:00
bool skip_timeout = false;
refresh:
2020-09-02 10:55:56 +03:00
clear(true);
2020-08-29 21:02:16 +03:00
print("\n\n \e[36m Limine " LIMINE_VERSION " \e[37m\n\n\n");
2020-06-05 21:27:52 +03:00
print("Select an entry:\n\n");
int max_entries;
for (max_entries = 0; ; max_entries++) {
if (config_get_entry_name(config_entry_name, max_entries, 1024) == -1)
break;
if (max_entries == selected_entry)
print(" \e[47m\e[30m %s \e[40m\e[37m\n", config_entry_name);
else
print(" %s\n", config_entry_name);
}
if (max_entries == 0)
panic("Config contains no entries.");
print("\nArrows to choose, enter to select, 'e' to edit command line.");
int c;
2020-06-05 21:27:52 +03:00
if (skip_timeout == false) {
print("\n\n");
2020-06-05 21:27:52 +03:00
for (int i = timeout; i; i--) {
print("\rBooting automatically in %u, press any key to stop the countdown...", i);
if ((c = pit_sleep_and_quit_on_keypress(18))) {
2020-06-05 21:27:52 +03:00
skip_timeout = true;
print("\e[2K\r\e[2A");
goto timeout_aborted;
2020-06-05 21:27:52 +03:00
}
}
goto autoboot;
}
for (;;) {
c = getchar();
timeout_aborted:
2020-06-05 21:27:52 +03:00
switch (c) {
case GETCHAR_CURSOR_UP:
if (--selected_entry == -1)
selected_entry = max_entries - 1;
goto refresh;
case GETCHAR_CURSOR_DOWN:
if (++selected_entry == max_entries)
selected_entry = 0;
goto refresh;
case '\r':
autoboot:
config_set_entry(selected_entry);
2020-09-02 10:55:56 +03:00
enable_cursor();
2020-06-05 21:27:52 +03:00
if (!config_get_value(cmdline, 0, CMDLINE_MAX, "KERNEL_CMDLINE")) {
if (!config_get_value(cmdline, 0, CMDLINE_MAX, "CMDLINE")) {
cmdline[0] = '\0';
}
}
2020-09-02 10:55:56 +03:00
clear(true);
2020-06-05 21:27:52 +03:00
return cmdline;
case 'e':
config_set_entry(selected_entry);
2020-09-02 10:55:56 +03:00
enable_cursor();
2020-06-05 21:27:52 +03:00
if (!config_get_value(cmdline, 0, CMDLINE_MAX, "KERNEL_CMDLINE")) {
if (!config_get_value(cmdline, 0, CMDLINE_MAX, "CMDLINE")) {
cmdline[0] = '\0';
}
}
print("\n\n> ");
2020-10-27 10:09:27 +03:00
readline(cmdline, cmdline, CMDLINE_MAX);
2020-09-02 10:55:56 +03:00
clear(true);
2020-06-05 21:27:52 +03:00
return cmdline;
}
}
}