Add logging system :)
This commit is contained in:
parent
f1756e6f58
commit
9f5fb547f7
@ -1,5 +1,6 @@
|
||||
set(SOURCES
|
||||
src/main.cpp
|
||||
src/Log.cpp
|
||||
src/video/Framebuffer.cpp
|
||||
src/video/TextConsole.cpp
|
||||
src/memory/MemoryManager.cpp
|
||||
|
115
kernel/src/Log.cpp
Normal file
115
kernel/src/Log.cpp
Normal file
@ -0,0 +1,115 @@
|
||||
#include "Log.h"
|
||||
#include "arch/Serial.h"
|
||||
#include "arch/Timer.h"
|
||||
#include "video/TextConsole.h"
|
||||
#include <Format.h>
|
||||
|
||||
static bool g_debug_enabled = true;
|
||||
static bool g_serial_enabled = true;
|
||||
static bool g_text_console_enabled = false;
|
||||
|
||||
static constexpr u32 BLACK = 0xff000000;
|
||||
static constexpr u32 WHITE = 0xffffffff;
|
||||
static constexpr u32 YELLOW = 0xffffff00;
|
||||
static constexpr u32 RED = 0xffff0000;
|
||||
|
||||
static Result<void> log_serial(LogLevel, const char* format, va_list origin)
|
||||
{
|
||||
va_list ap;
|
||||
va_copy(ap, origin);
|
||||
|
||||
TRY(Serial::printf("[%6zu.%.6zu] ", Timer::ticks(), Timer::ticks_us() - (Timer::ticks() * 1000000)));
|
||||
|
||||
TRY(cstyle_format(
|
||||
format,
|
||||
[](char c, void*) -> Result<void> {
|
||||
Serial::putchar(c);
|
||||
return {};
|
||||
},
|
||||
nullptr, ap));
|
||||
|
||||
Serial::putchar('\n');
|
||||
|
||||
va_end(ap);
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
static Result<void> log_text_console(LogLevel level, const char* format, va_list origin)
|
||||
{
|
||||
va_list ap;
|
||||
va_copy(ap, origin);
|
||||
|
||||
u32 original_foreground = TextConsole::foreground();
|
||||
u32 original_background = TextConsole::background();
|
||||
|
||||
TextConsole::set_background(BLACK);
|
||||
|
||||
if (level == LogLevel::Warn) TextConsole::set_foreground(YELLOW);
|
||||
else if (level == LogLevel::Error)
|
||||
TextConsole::set_foreground(RED);
|
||||
else
|
||||
TextConsole::set_foreground(WHITE);
|
||||
|
||||
TRY(cstyle_format(
|
||||
format,
|
||||
[](char c, void*) -> Result<void> {
|
||||
TextConsole::putchar(c);
|
||||
return {};
|
||||
},
|
||||
nullptr, ap));
|
||||
|
||||
TextConsole::putchar('\n');
|
||||
|
||||
TextConsole::set_background(original_background);
|
||||
|
||||
TextConsole::set_foreground(original_foreground);
|
||||
|
||||
va_end(ap);
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
Result<void> vlog(LogLevel level, const char* format, va_list ap)
|
||||
{
|
||||
if (!g_debug_enabled && level == LogLevel::Debug) return {};
|
||||
|
||||
if (g_serial_enabled) TRY(log_serial(level, format, ap));
|
||||
if (g_text_console_enabled) TRY(log_text_console(level, format, ap));
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
Result<void> log(LogLevel level, const char* format, ...)
|
||||
{
|
||||
va_list ap;
|
||||
va_start(ap, format);
|
||||
|
||||
TRY(vlog(level, format, ap));
|
||||
|
||||
va_end(ap);
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
void setup_log(bool enable_debug, bool enable_serial, bool enable_text_console)
|
||||
{
|
||||
g_debug_enabled = enable_debug;
|
||||
g_serial_enabled = enable_serial;
|
||||
g_text_console_enabled = enable_text_console;
|
||||
}
|
||||
|
||||
bool log_debug_enabled()
|
||||
{
|
||||
return g_debug_enabled;
|
||||
}
|
||||
|
||||
bool log_serial_enabled()
|
||||
{
|
||||
return g_serial_enabled;
|
||||
}
|
||||
|
||||
bool log_text_console_enabled()
|
||||
{
|
||||
return g_text_console_enabled;
|
||||
}
|
29
kernel/src/Log.h
Normal file
29
kernel/src/Log.h
Normal file
@ -0,0 +1,29 @@
|
||||
#pragma once
|
||||
#include <Result.h>
|
||||
#include <stdarg.h>
|
||||
|
||||
enum class LogLevel
|
||||
{
|
||||
Debug,
|
||||
Info,
|
||||
Warn,
|
||||
Error,
|
||||
};
|
||||
|
||||
Result<void> vlog(LogLevel level, const char* format, va_list ap);
|
||||
Result<void> log(LogLevel level, const char* format, ...);
|
||||
|
||||
void setup_log(bool enable_debug, bool enable_serial, bool enable_text_console);
|
||||
bool log_debug_enabled();
|
||||
bool log_serial_enabled();
|
||||
bool log_text_console_enabled();
|
||||
|
||||
#define kdbgln(...) log(LogLevel::Debug, __VA_ARGS__)
|
||||
#define kinfoln(...) log(LogLevel::Info, __VA_ARGS__)
|
||||
#define kwarnln(...) log(LogLevel::Warn, __VA_ARGS__)
|
||||
#define kerrorln(...) log(LogLevel::Error, __VA_ARGS__)
|
||||
|
||||
#define try_kdbgln(...) TRY(log(LogLevel::Debug, __VA_ARGS__))
|
||||
#define try_kinfoln(...) TRY(log(LogLevel::Info, __VA_ARGS__))
|
||||
#define try_kwarnln(...) TRY(log(LogLevel::Warn, __VA_ARGS__))
|
||||
#define try_kerrorln(...) TRY(log(LogLevel::Error, __VA_ARGS__))
|
@ -1,4 +1,5 @@
|
||||
#include "boot/Init.h"
|
||||
#include "Log.h"
|
||||
#include "arch/CPU.h"
|
||||
#include "arch/Serial.h"
|
||||
#include "boot/bootboot.h"
|
||||
@ -23,9 +24,12 @@ void Init::early_init()
|
||||
CPU::disable_interrupts();
|
||||
|
||||
Framebuffer::init();
|
||||
|
||||
setup_log(log_debug_enabled(), log_serial_enabled(), true);
|
||||
|
||||
MemoryManager::init();
|
||||
|
||||
CPU::platform_init();
|
||||
|
||||
check(MemoryManager::protect_kernel_sections().has_value());
|
||||
MemoryManager::protect_kernel_sections().release_value();
|
||||
}
|
@ -1,3 +1,4 @@
|
||||
#include "Log.h"
|
||||
#include "arch/CPU.h"
|
||||
#include "arch/MMU.h"
|
||||
#include "arch/Serial.h"
|
||||
@ -36,10 +37,14 @@ Result<void> init()
|
||||
TextConsole::set_foreground(0xff000055);
|
||||
TextConsole::set_background(0xff88ff00);
|
||||
|
||||
TextConsole::printf("Hello from Moon for the %s architecture!", CPU::platform_string());
|
||||
TextConsole::printf("Hello from Moon for the %s architecture!\n", CPU::platform_string());
|
||||
|
||||
Timer::init();
|
||||
|
||||
kinfoln("Hello, world!");
|
||||
kwarnln("THIS IS A WARNING");
|
||||
kerrorln("ERROR: Please do something.");
|
||||
|
||||
CPU::platform_finish_init();
|
||||
|
||||
CPU::enable_interrupts();
|
||||
@ -54,17 +59,17 @@ Result<void> init()
|
||||
|
||||
char buffer[64];
|
||||
to_dynamic_unit(MemoryManager::free(), buffer, sizeof(buffer));
|
||||
Serial::printf("Free memory: %s\n", buffer);
|
||||
kinfoln("Free memory: %s", buffer);
|
||||
to_dynamic_unit(MemoryManager::used(), buffer, sizeof(buffer));
|
||||
Serial::printf("Used memory: %s\n", buffer);
|
||||
kinfoln("Used memory: %s", buffer);
|
||||
to_dynamic_unit(MemoryManager::reserved(), buffer, sizeof(buffer));
|
||||
Serial::printf("Reserved memory: %s\n", buffer);
|
||||
kinfoln("Reserved memory: %s", buffer);
|
||||
|
||||
while (1)
|
||||
{
|
||||
while ((Timer::ticks_ms() - start) < 20) { CPU::wait_for_interrupt(); }
|
||||
while ((Timer::ticks_ms() - start) < 100) { CPU::wait_for_interrupt(); }
|
||||
start = Timer::ticks_ms();
|
||||
Serial::printf("%8zu milliseconds have passed!\n", start);
|
||||
kdbgln("%8zu milliseconds have passed!", start);
|
||||
}
|
||||
|
||||
return {};
|
||||
|
@ -122,6 +122,16 @@ namespace TextConsole
|
||||
g_background_color = color;
|
||||
}
|
||||
|
||||
u32 foreground()
|
||||
{
|
||||
return g_foreground_color;
|
||||
}
|
||||
|
||||
u32 background()
|
||||
{
|
||||
return g_background_color;
|
||||
}
|
||||
|
||||
void move_to(u32 x, u32 y)
|
||||
{
|
||||
g_x_position = x;
|
||||
|
@ -9,6 +9,8 @@ namespace TextConsole
|
||||
void putchar(char c);
|
||||
void set_foreground(u32 color);
|
||||
void set_background(u32 color);
|
||||
u32 foreground();
|
||||
u32 background();
|
||||
void move_to(u32 x, u32 y);
|
||||
void print(const char* str);
|
||||
void println(const char* str);
|
||||
|
Loading…
Reference in New Issue
Block a user