kernel: add kernel.early-console and kernel.console boot args

kernel.early-console is used to specify which output device the
kernel boot log should be written to. the first thing the kernel
does on boot after initialising the bootstrap processor is initialise
the early console, making it useful for debugging problems that
occur early in the boot process. this arg accepts a list of hard-coded
values for output devices, such as tty0 for the display or ttyS0
for the serial port. the exact values supported will depend on the
platform.

once all drivers are loaded, the kernel switches to the device specified
by kernel.console for output. unlike kernel.early-console, this arg
specifies the name of a tty device in /dev/tty. this means that, not
only are more devices supported (any device provided by a tty driver),
but the kernel can also get input from the user using this console too
(not used by the kernel itself, but will be used by the user to interact
with userspace programs, like the shell).
This commit is contained in:
2023-12-30 09:09:18 +00:00
parent fc56f906d3
commit b0c021d4e9
6 changed files with 91 additions and 18 deletions

View File

@@ -13,9 +13,9 @@
#include <socks/memblock.h> #include <socks/memblock.h>
#include <socks/vm.h> #include <socks/vm.h>
#include <socks/printk.h> #include <socks/printk.h>
#include <socks/console.h>
#include <socks/machine/cpu.h> #include <socks/machine/cpu.h>
#include <socks/libc/stdio.h> #include <socks/libc/stdio.h>
#include <arch/vgacon.h>
#ifdef KEXT_NET_DOORSTUCK_SOCKS_ACPI #ifdef KEXT_NET_DOORSTUCK_SOCKS_ACPI
#include <arch/acpi.h> #include <arch/acpi.h>
@@ -25,6 +25,10 @@
#include <socks/fbcon.h> #include <socks/fbcon.h>
#endif #endif
#ifdef KEXT_NET_DOORSTUCK_SOCKS_SERIALCON
#include <socks/serialcon.h>
#endif
#define PTR32(x) ((void *)((uintptr_t)(x))) #define PTR32(x) ((void *)((uintptr_t)(x)))
static ml_cpu_block g_bootstrap_cpu = {0}; static ml_cpu_block g_bootstrap_cpu = {0};
@@ -54,6 +58,28 @@ static void early_vm_init(void)
printk("memblock: reserved bios+kernel at [0x%016llx-0x%016llx]", 0, (uintptr_t)__pend); printk("memblock: reserved bios+kernel at [0x%016llx-0x%016llx]", 0, (uintptr_t)__pend);
} }
void early_console_init(void)
{
const char *dest = arg_value("kernel.early-console");
if (!dest) {
return;
}
#ifdef KEXT_NET_DOORSTUCK_SOCKS_FBCON
if (!strcmp(dest, "tty0")) {
early_vgacon_init();
}
#endif
#ifdef KEXT_NET_DOORSTUCK_SOCKS_SERIALCON
if (!strncmp(dest, "ttyS0", 5)) {
/* TODO allow specifying baud rate from command line */
unsigned int baud = 115200;
early_serialcon_init(baud);
}
#endif
}
static void init_bootfb(multiboot_info_t *mb) static void init_bootfb(multiboot_info_t *mb)
{ {
__bootfb_varinfo.fb_xres = mb->framebuffer_width; __bootfb_varinfo.fb_xres = mb->framebuffer_width;
@@ -109,11 +135,10 @@ int ml_init(uintptr_t arg)
init_bootfb(mb); init_bootfb(mb);
bootstrap_cpu_init(); bootstrap_cpu_init();
#ifdef KEXT_NET_DOORSTUCK_SOCKS_FBCON
early_vgacon_init();
#endif
clock_calibrate(500); clock_calibrate(500);
early_console_init();
print_kernel_banner(); print_kernel_banner();
early_vm_init(); early_vm_init();

View File

@@ -113,6 +113,9 @@ struct tty_device {
char *tty_linebuf; char *tty_linebuf;
}; };
extern void register_tty_console(void);
extern void redirect_printk_to_tty(struct device *dest);
extern kern_status_t tty_bootstrap(void); extern kern_status_t tty_bootstrap(void);
extern struct tty_ldisc *tty_default_line_discipline(void); extern struct tty_ldisc *tty_default_line_discipline(void);

View File

@@ -1,5 +1,6 @@
#include <stdint.h> #include <stdint.h>
#include <socks/init.h> #include <socks/init.h>
#include <socks/arg.h>
#include <socks/clock.h> #include <socks/clock.h>
#include <socks/input.h> #include <socks/input.h>
#include <socks/panic.h> #include <socks/panic.h>
@@ -10,6 +11,7 @@
#include <socks/kext.h> #include <socks/kext.h>
#include <socks/object.h> #include <socks/object.h>
#include <socks/sched.h> #include <socks/sched.h>
#include <socks/libc/stdio.h>
#include <socks/machine/init.h> #include <socks/machine/init.h>
#include <socks/cpu.h> #include <socks/cpu.h>
@@ -29,7 +31,7 @@ void print_kernel_banner(void)
static void hang(void) static void hang(void)
{ {
while (1) { while (1) {
//printk("tick"); printk("%d: tick", this_cpu());
milli_sleep(2000); milli_sleep(2000);
} }
} }
@@ -37,10 +39,10 @@ static void hang(void)
void background_thread(void) void background_thread(void)
{ {
printk("background_thread() running on processor %u", this_cpu()); printk("background_thread() running on processor %u", this_cpu());
milli_sleep(500); milli_sleep(1000);
while (1) { while (1) {
//printk("tock"); printk("%d: tock", this_cpu());
milli_sleep(2000); milli_sleep(2000);
} }
} }
@@ -99,6 +101,26 @@ void kernel_init(uintptr_t arg)
tty_set_foreground(cast_to_device(tty0)); tty_set_foreground(cast_to_device(tty0));
} }
const char *console_tty_name = arg_value("kernel.console");
if (!console_tty_name) {
console_tty_name = "tty0";
}
char console_tty_path[128];
snprintf(console_tty_path, sizeof console_tty_path, "/dev/tty/%s", console_tty_name);
struct object *console_tty = NULL;
status = object_get(console_tty_path, &console_tty);
if (status == KERN_OK) {
register_tty_console();
struct device *console_tty_device = cast_to_device(console_tty);
redirect_printk_to_tty(console_tty_device);
object_deref(console_tty);
} else {
printk("console tty '%s' is unavailable.", console_tty_name);
}
create_kernel_thread(background_thread); create_kernel_thread(background_thread);
struct object *kbd; struct object *kbd;

View File

@@ -9,10 +9,13 @@ static struct char_device_ops tty_ops = {
.write = tty_write, .write = tty_write,
}; };
static spin_lock_t foreground_lock; static spin_lock_t foreground_lock = SPIN_LOCK_INIT;
static struct device *foreground = NULL; static struct device *foreground = NULL;
static struct device *foreground_input = NULL; static struct device *foreground_input = NULL;
static spin_lock_t kernel_console_tty_lock = SPIN_LOCK_INIT;
static struct device *kernel_console_tty = NULL;
static void tty_input_hook_callback(struct device *dev, struct input_event *ev, enum input_event_hook_flags *flags, void *arg) static void tty_input_hook_callback(struct device *dev, struct input_event *ev, enum input_event_hook_flags *flags, void *arg)
{ {
struct device *fg_tty = foreground; struct device *fg_tty = foreground;
@@ -31,9 +34,9 @@ static struct input_event_hook foreground_input_hook = {
static void tty_console_write(struct console *con, const char *s, unsigned int len) static void tty_console_write(struct console *con, const char *s, unsigned int len)
{ {
if (foreground) { if (kernel_console_tty) {
size_t nr_written; size_t nr_written;
tty_write(foreground, s, 0, len, &nr_written, 0); tty_write(kernel_console_tty, s, 0, len, &nr_written, 0);
} }
} }
@@ -42,6 +45,26 @@ static struct console tty_console = {
.c_write = tty_console_write, .c_write = tty_console_write,
}; };
void register_tty_console(void)
{
console_register(&tty_console);
}
void redirect_printk_to_tty(struct device *dest)
{
unsigned long flags;
spin_lock_irqsave(&kernel_console_tty_lock, &flags);
if (kernel_console_tty) {
device_deref(kernel_console_tty);
kernel_console_tty = NULL;
}
kernel_console_tty = device_ref(dest);
spin_unlock_irqrestore(&kernel_console_tty_lock, flags);
}
struct device *tty_device_create(void) struct device *tty_device_create(void)
{ {
struct char_device *cdev = char_device_create(); struct char_device *cdev = char_device_create();
@@ -94,17 +117,17 @@ kern_status_t tty_device_register(struct device *dev, struct tty_driver *owner,
void tty_set_foreground(struct device *tty) void tty_set_foreground(struct device *tty)
{ {
bool console_init = false; unsigned long flags;
spin_lock_irqsave(&foreground_lock, &flags);
if (foreground) { if (foreground) {
console_init = true;
device_deref(foreground); device_deref(foreground);
foreground = NULL;
} }
foreground = device_ref(tty); foreground = device_ref(tty);
if (!console_init) { spin_unlock_irqrestore(&foreground_lock, flags);
console_register(&tty_console);
}
} }
kern_status_t tty_connect_foreground_input_device(struct device *input) kern_status_t tty_connect_foreground_input_device(struct device *input)

View File

@@ -18,7 +18,7 @@ extern char serial_recv_byte(int device);
extern int serial_rcvd(int device); extern int serial_rcvd(int device);
extern void serialcon_init(int baud); extern void early_serialcon_init(int baud);
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@@ -5,7 +5,7 @@
#include <socks/libc/stdio.h> #include <socks/libc/stdio.h>
#include <arch/ports.h> #include <arch/ports.h>
#include <arch/irq.h> #include <arch/irq.h>
#include "serial.h" #include <socks/serialcon.h>
#define COM1 0x3F8 #define COM1 0x3F8
#define COM2 0x2F8 #define COM2 0x2F8
@@ -155,7 +155,7 @@ static struct irq_hook irq1_hook = {
.irq_callback = serial_irq1, .irq_callback = serial_irq1,
}; };
void serialcon_init(int baud) void early_serialcon_init(int baud)
{ {
hook_irq(IRQ4, &irq1_hook); hook_irq(IRQ4, &irq1_hook);
init_serial_port(COM1, baud); init_serial_port(COM1, baud);