rulimine/src/main.c

168 lines
4.2 KiB
C
Raw Normal View History

2019-05-15 07:08:56 +03:00
asm (
".section .entry\n\t"
2019-05-15 07:08:56 +03:00
"xor dh, dh\n\t"
"push edx\n\t"
"call main\n\t"
);
#include <drivers/vga_textmode.h>
#include <lib/real.h>
2020-01-22 07:02:12 +03:00
#include <lib/blib.h>
#include <lib/libc.h>
2020-04-15 14:21:44 +03:00
#include <lib/part.h>
2020-01-22 09:13:19 +03:00
#include <lib/config.h>
2020-04-14 06:20:55 +03:00
#include <fs/file.h>
2020-01-25 04:05:19 +03:00
#include <sys/interrupt.h>
2020-03-25 03:04:18 +03:00
#include <lib/elf.h>
#include <protos/stivale.h>
2020-04-19 11:14:49 +03:00
#include <protos/linux.h>
2020-01-21 13:42:17 +03:00
2020-04-21 19:27:05 +03:00
static char cmdline[128];
void boot_menu(void) {
text_disable_cursor();
int selected_entry = 0;
refresh:
text_clear();
print("qloader2\n\n");
print("Select an entry:\n\n");
size_t max_entries;
for (max_entries = 0; ; max_entries++) {
char buf[32];
if (config_get_entry_name(buf, max_entries, 32) == -1)
break;
if (max_entries == selected_entry)
print(" -> %s\n", buf);
else
print(" %s\n", buf);
}
print("\nArrows to choose, enter to select, 'e' to edit command line.");
for (;;) {
int c = getchar();
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':
config_set_entry(selected_entry);
text_enable_cursor();
text_clear();
return;
case 'e':
config_set_entry(selected_entry);
text_enable_cursor();
config_get_value(cmdline, 0, 128, "KERNEL_CMDLINE");
print("\n\n> ");
gets(cmdline, cmdline, 128);
text_clear();
return;
}
}
}
2020-01-21 13:42:17 +03:00
extern symbol bss_begin;
extern symbol bss_end;
2019-05-15 07:08:56 +03:00
void main(int boot_drive) {
2020-04-14 06:20:55 +03:00
struct file_handle f;
2020-03-15 02:12:09 +03:00
2020-01-21 13:42:17 +03:00
// Zero out .bss section
for (uint8_t *p = bss_begin; p < bss_end; p++)
*p = 0;
// Initial prompt.
init_vga_textmode();
2020-01-25 04:05:19 +03:00
init_idt();
2020-04-21 19:27:05 +03:00
print("qloader2\n\n");
2020-04-15 14:21:44 +03:00
print("Boot drive: %x\n", boot_drive);
2020-01-21 13:42:17 +03:00
2020-04-15 14:21:44 +03:00
// Look for config file.
print("Searching for config file...\n");
struct part parts[4];
for (int i = 0; ; i++) {
if (i == 4) {
panic("Config file not found.");
}
print("Checking partition %d...\n", i);
int ret = get_part(&parts[i], boot_drive, i);
if (ret) {
2020-04-15 14:21:44 +03:00
print("Partition not found.\n");
} else {
2020-04-15 14:21:44 +03:00
print("Partition found.\n");
if (!init_config(boot_drive, i)) {
print("Config file found and loaded.\n");
break;
2020-01-22 09:13:19 +03:00
}
}
}
2020-04-21 19:27:05 +03:00
int drive, part;
char path[128], proto[64], buf[32];
2020-01-22 09:13:19 +03:00
2020-04-21 19:27:05 +03:00
int timeout;
2020-04-15 14:21:44 +03:00
if (!config_get_value(buf, 0, 64, "TIMEOUT")) {
timeout = 5;
} else {
timeout = (int)strtoui(buf);
2020-01-22 09:13:19 +03:00
}
2020-04-21 19:27:05 +03:00
print("\n");
2020-03-28 04:52:59 +03:00
for (int i = timeout; i; i--) {
2020-04-21 19:27:05 +03:00
print("\rBooting in %d (press any key for boot menu, command line edit)...", i);
2020-01-25 05:06:56 +03:00
if (pit_sleep_and_quit_on_keypress(18)) {
2020-04-21 19:27:05 +03:00
boot_menu();
goto got_entry;
2020-01-25 05:06:56 +03:00
}
}
2020-04-15 14:21:44 +03:00
print("\n\n");
2020-01-25 05:06:56 +03:00
2020-04-21 19:27:05 +03:00
if (config_set_entry(0) == -1) {
panic("Invalid config entry.");
}
got_entry:
if (!config_get_value(buf, 0, 32, "KERNEL_DRIVE")) {
print("KERNEL_DRIVE not specified, using boot drive (%x)", boot_drive);
drive = boot_drive;
} else {
drive = (int)strtoui(buf);
}
config_get_value(buf, 0, 32, "KERNEL_PARTITION");
part = (int)strtoui(buf);
config_get_value(path, 0, 128, "KERNEL_PATH");
config_get_value(proto, 0, 64, "KERNEL_PROTO");
2020-04-14 06:20:55 +03:00
fopen(&f, drive, part, path);
if (!strcmp(proto, "stivale")) {
stivale_load(&f, cmdline);
2020-04-19 11:14:49 +03:00
} else if (!strcmp(proto, "linux")) {
linux_load(&f, cmdline);
} else if (!strcmp(proto, "qword")) {
2020-04-14 06:20:55 +03:00
fread(&f, (void *)0x100000, 0, f.size);
// Boot the kernel.
asm volatile (
"cli\n\t"
"jmp 0x100000\n\t"
:
: "b" (cmdline)
: "memory"
);
} else {
print("Invalid protocol specified: `%s`.\n", proto);
for (;;);
}
2020-04-15 14:21:44 +03:00
}