167 lines
5.1 KiB
Odin
167 lines
5.1 KiB
Odin
package grime
|
|
|
|
import "base:runtime"
|
|
import "core:fmt"
|
|
import "core:mem"
|
|
import "core:os"
|
|
import str "core:strings"
|
|
import "core:time"
|
|
import core_log "core:log"
|
|
|
|
Max_Logger_Message_Width :: 160
|
|
|
|
LogLevel :: core_log.Level
|
|
|
|
LoggerEntry :: struct {
|
|
text : string,
|
|
timestamp : string,
|
|
level : string,
|
|
location : string,
|
|
}
|
|
|
|
Logger :: struct {
|
|
file_path : string,
|
|
file : os.Handle,
|
|
id : string,
|
|
|
|
varena : VArena,
|
|
entries : Array(LoggerEntry),
|
|
}
|
|
|
|
to_odin_logger :: proc( logger : ^ Logger ) -> core_log.Logger {
|
|
return { logger_interface, logger, core_log.Level.Debug, core_log.Default_File_Logger_Opts }
|
|
}
|
|
|
|
logger_init :: proc( logger : ^ Logger, id : string, file_path : string, file := os.INVALID_HANDLE )
|
|
{
|
|
if file == os.INVALID_HANDLE
|
|
{
|
|
logger_file, result_code := file_open( file_path, os.O_RDWR | os.O_CREATE )
|
|
assert( result_code == os.ERROR_NONE, "Log failures are fatal and must never occur at runtime (there is no logging)" )
|
|
logger.file = logger_file
|
|
os.truncate( file_path, 0 )
|
|
}
|
|
else {
|
|
logger.file = file
|
|
}
|
|
logger.file_path = file_path
|
|
logger.id = id
|
|
|
|
LOGGER_VARENA_BASE_ADDRESS : uintptr = 2 * Terabyte
|
|
@static vmem_init_counter : uintptr = 0
|
|
|
|
alloc_error : AllocatorError
|
|
// logger.varena, alloc_error = varena_init(
|
|
// LOGGER_VARENA_BASE_ADDRESS + vmem_init_counter * 250 * Megabyte,
|
|
// 1 * Megabyte,
|
|
// 128 * Kilobyte,
|
|
// growth_policy = nil,
|
|
// allow_any_resize = true,
|
|
// dbg_name = "logger varena",
|
|
// enable_mem_tracking = false )
|
|
// verify( alloc_error == .None, "Failed to allocate logger's virtual arena")
|
|
vmem_init_counter += 1
|
|
|
|
// TODO(Ed): Figure out another solution here...
|
|
logger.entries, alloc_error = array_init(Array(LoggerEntry), 8192, runtime.heap_allocator())
|
|
verify( alloc_error == .None, "Failed to allocate logger's entries array")
|
|
|
|
context.logger = { logger_interface, logger, core_log.Level.Debug, core_log.Default_File_Logger_Opts }
|
|
log("Initialized Logger")
|
|
when false {
|
|
log("This sentence is over 80 characters long on purpose to test the ability of this logger to properfly wrap long as logs with a new line and then at the end of that pad it with the appropraite signature.")
|
|
}
|
|
}
|
|
|
|
logger_interface :: proc(
|
|
logger_data : rawptr,
|
|
level : core_log.Level,
|
|
text : string,
|
|
options : core_log.Options,
|
|
location := #caller_location )
|
|
{
|
|
logger := cast(^ Logger) logger_data
|
|
|
|
@static builder_backing : [16 * Kilobyte] byte; {
|
|
mem.set( raw_data( builder_backing[:] ), 0, len(builder_backing) )
|
|
}
|
|
builder := str.builder_from_bytes( builder_backing[:] )
|
|
|
|
first_line_length := len(text) > Max_Logger_Message_Width ? Max_Logger_Message_Width : len(text)
|
|
first_line := transmute(string) text[ 0 : first_line_length ]
|
|
|
|
str_fmt_builder( & builder, "%s ", first_line )
|
|
|
|
// str_fmt_builder( & builder, "%-s ", first_line )
|
|
|
|
// Signature
|
|
{
|
|
when time.IS_SUPPORTED
|
|
{
|
|
if core_log.Full_Timestamp_Opts & options != nil {
|
|
str_fmt_builder( & builder, "[")
|
|
|
|
t := time.now()
|
|
year, month, day := time.date(t)
|
|
hour, minute, second := time.clock(t)
|
|
|
|
if .Date in options {
|
|
str_fmt_builder( & builder, "%d-%02d-%02d ", year, month, day )
|
|
}
|
|
if .Time in options {
|
|
str_fmt_builder( & builder, "%02d:%02d:%02d", hour, minute, second)
|
|
}
|
|
|
|
str_fmt_builder( & builder, "] ")
|
|
}
|
|
}
|
|
|
|
core_log.do_level_header( options, & builder, level )
|
|
|
|
if logger.id != "" {
|
|
str_fmt_builder( & builder, "[%s] ", logger.id )
|
|
}
|
|
core_log.do_location_header( options, & builder, location )
|
|
}
|
|
|
|
// Oversized message handling
|
|
if len(text) > Max_Logger_Message_Width
|
|
{
|
|
offset := Max_Logger_Message_Width
|
|
bytes := transmute( []u8 ) text
|
|
for left := len(bytes) - Max_Logger_Message_Width; left > 0; left -= Max_Logger_Message_Width
|
|
{
|
|
str_fmt_builder( & builder, "\n" )
|
|
subset_length := len(text) - offset
|
|
if subset_length > Max_Logger_Message_Width {
|
|
subset_length = Max_Logger_Message_Width
|
|
}
|
|
subset := slice_ptr( ptr_offset( raw_data(bytes), offset), subset_length )
|
|
str_fmt_builder( & builder, "%s", transmute(string) subset )
|
|
offset += Max_Logger_Message_Width
|
|
}
|
|
}
|
|
|
|
str_to_file_ln( logger.file, to_string(builder) )
|
|
}
|
|
|
|
// This buffer is used below excluisvely to prevent any allocator recusion when verbose logging from allocators.
|
|
// This means a single line is limited to 32k buffer (increase naturally if this SOMEHOW becomes a bottleneck...)
|
|
Logger_Allocator_Buffer : [32 * Kilobyte]u8
|
|
|
|
log :: proc( msg : string, level := LogLevel.Info, loc := #caller_location ) {
|
|
temp_arena : Arena; arena_init(& temp_arena, Logger_Allocator_Buffer[:])
|
|
context.allocator = arena_allocator(& temp_arena)
|
|
context.temp_allocator = arena_allocator(& temp_arena)
|
|
|
|
core_log.log( level, msg, location = loc )
|
|
}
|
|
|
|
log_fmt :: proc( fmt : string, args : ..any, level := LogLevel.Info, loc := #caller_location ) {
|
|
temp_arena : Arena; arena_init(& temp_arena, Logger_Allocator_Buffer[:])
|
|
context.allocator = arena_allocator(& temp_arena)
|
|
context.temp_allocator = arena_allocator(& temp_arena)
|
|
|
|
core_log.logf( level, fmt, ..args, location = loc )
|
|
}
|