324 lines
13 KiB
Odin
324 lines
13 KiB
Odin
package host
|
|
|
|
//region STATIC MEMORY
|
|
// All program defined process memory here. (There will still be artifacts from the OS CRT and third-party pacakges)
|
|
host_memory: ProcessMemory
|
|
@(thread_local) thread_memory: ThreadMemory
|
|
|
|
//endregion STATIC MEMORY
|
|
|
|
//region HOST RUNTIME
|
|
|
|
load_client_api :: proc(version_id: int) -> (loaded_module: Client_API) {
|
|
profile(#procedure)
|
|
using loaded_module
|
|
// Make sure we have a dll to work with
|
|
file_io_err: OS_Error; write_time, file_io_err = file_last_write_time_by_name("sectr.dll")
|
|
if file_io_err != OS_ERROR_NONE {
|
|
panic_contextless( "Could not resolve the last write time for sectr")
|
|
}
|
|
//TODO(Ed): Lets try to minimize this...
|
|
thread_sleep( Millisecond * 25 )
|
|
// Get the live dll loaded up
|
|
file_copy_sync( Path_Sectr_Module, Path_Sectr_Live_Module, allocator = context.temp_allocator )
|
|
did_load: bool; lib, did_load = os_lib_load( Path_Sectr_Live_Module )
|
|
if ! did_load do panic( "Failed to load the sectr module.")
|
|
startup = cast( type_of( host_memory.client_api.startup)) os_lib_get_proc(lib, "startup")
|
|
shutdown = cast( type_of( host_memory.client_api.shutdown)) os_lib_get_proc(lib, "sectr_shutdown")
|
|
tick_lane_startup = cast( type_of( host_memory.client_api.tick_lane_startup)) os_lib_get_proc(lib, "tick_lane_startup")
|
|
job_worker_startup = cast( type_of( host_memory.client_api.job_worker_startup)) os_lib_get_proc(lib, "job_worker_startup")
|
|
hot_reload = cast( type_of( host_memory.client_api.hot_reload)) os_lib_get_proc(lib, "hot_reload")
|
|
tick_lane = cast( type_of( host_memory.client_api.tick_lane)) os_lib_get_proc(lib, "tick_lane")
|
|
clean_frame = cast( type_of( host_memory.client_api.clean_frame)) os_lib_get_proc(lib, "clean_frame")
|
|
jobsys_worker_tick = cast( type_of( host_memory.client_api.jobsys_worker_tick)) os_lib_get_proc(lib, "jobsys_worker_tick")
|
|
if startup == nil do panic("Failed to load sectr.startup symbol" )
|
|
if shutdown == nil do panic("Failed to load sectr.shutdown symbol" )
|
|
if tick_lane_startup == nil do panic("Failed to load sectr.tick_lane_startup symbol" )
|
|
if job_worker_startup == nil do panic("Failed to load sectr.job_worker_startup symbol" )
|
|
if hot_reload == nil do panic("Failed to load sectr.hot_reload symbol" )
|
|
if tick_lane == nil do panic("Failed to load sectr.tick_lane symbol" )
|
|
if clean_frame == nil do panic("Failed to load sectr.clean_frame symbol" )
|
|
if jobsys_worker_tick == nil do panic("Failed to laod sectr.jobsys_worker_tick")
|
|
lib_version = version_id
|
|
return
|
|
}
|
|
|
|
master_prepper_proc :: proc(thread: ^SysThread) {}
|
|
main :: proc()
|
|
{
|
|
// Setup host arenas
|
|
arena_init(& host_memory.host_persist, host_memory.host_persist_buf[:])
|
|
arena_init(& host_memory.host_scratch, host_memory.host_scratch_buf[:])
|
|
context.allocator = arena_allocator(& host_memory.host_persist)
|
|
context.temp_allocator = arena_allocator(& host_memory.host_scratch)
|
|
// Setup the "Master Prepper" thread
|
|
{
|
|
thread_memory.id = .Master_Prepper
|
|
thread_id := thread_current_id()
|
|
using thread_memory
|
|
host_memory.threads[WorkerID.Master_Prepper] = new(SysThread)
|
|
system_ctx = host_memory.threads[WorkerID.Master_Prepper]
|
|
system_ctx.creation_allocator = {}
|
|
system_ctx.procedure = master_prepper_proc
|
|
when ODIN_OS == .Windows {
|
|
// system_ctx.win32_thread = w32_get_current_thread()
|
|
// system_ctx.win32_thread_id = w32_get_current_thread_id()
|
|
system_ctx.id = cast(int) system_ctx.win32_thread_id
|
|
}
|
|
}
|
|
when SHOULD_SETUP_PROFILERS
|
|
{
|
|
// Setup main profiler
|
|
host_memory.spall_context = spall_context_create(Path_Sectr_Spall_Record)
|
|
grime_set_profiler_module_context(& host_memory.spall_context)
|
|
thread_memory.spall_buffer = spall_buffer_create(thread_memory.spall_buffer_backing[:], cast(u32) thread_memory.system_ctx.id)
|
|
grime_set_profiler_thread_buffer(& thread_memory.spall_buffer)
|
|
}
|
|
// Setup the logger
|
|
path_logger_finalized: string
|
|
{
|
|
profile("Setup the logger")
|
|
fmt_backing := make([]byte, 32 * Kilo, allocator = context.temp_allocator);
|
|
|
|
// Generating the logger's name, it will be used when the app is shutting down.
|
|
{
|
|
startup_time := time_now()
|
|
year, month, day := time_date( startup_time)
|
|
hour, min, sec := time_clock_from_time( startup_time)
|
|
|
|
if ! os_is_directory( Path_Logs ) {
|
|
os_make_directory( Path_Logs )
|
|
}
|
|
timestamp := str_pfmt_buffer( fmt_backing, "%04d-%02d-%02d_%02d-%02d-%02d", year, month, day, hour, min, sec)
|
|
host_memory.path_logger_finalized = str_pfmt_buffer( fmt_backing, "%s/sectr_%v.log", Path_Logs, timestamp)
|
|
}
|
|
logger_init( & host_memory.host_logger, "Sectr Host", str_pfmt_buffer( fmt_backing, "%s/sectr.log", Path_Logs ) )
|
|
context.logger = to_odin_logger( & host_memory.host_logger )
|
|
{
|
|
// Log System Context
|
|
builder := strbuilder_from_bytes( fmt_backing )
|
|
str_pfmt_builder( & builder, "Core Count: %v, ", os_core_count() )
|
|
str_pfmt_builder( & builder, "Page Size: %v", os_page_size() )
|
|
log_print( to_str(builder) )
|
|
}
|
|
free_all(context.temp_allocator)
|
|
}
|
|
context.logger = to_odin_logger( & host_memory.host_logger )
|
|
// Load the Enviornment API for the first-time
|
|
{
|
|
host_memory.client_api = load_client_api( 1 )
|
|
verify( host_memory.client_api.lib_version != 0, "Failed to initially load the sectr module" )
|
|
}
|
|
// Client API Startup
|
|
host_memory.client_api.startup(& host_memory, & thread_memory)
|
|
{
|
|
profile("thread_wide_startup")
|
|
assert(thread_memory.id == .Master_Prepper)
|
|
{
|
|
profile("Tick Lanes")
|
|
host_memory.tick_running = true
|
|
host_memory.tick_lanes = THREAD_TICK_LANES
|
|
barrier_init(& host_memory.lane_sync, THREAD_TICK_LANES)
|
|
when THREAD_TICK_LANES > 1 {
|
|
for id in 1 ..= (THREAD_TICK_LANES - 1) {
|
|
launch_tick_lane_thread(cast(WorkerID) id)
|
|
}
|
|
}
|
|
}
|
|
// Job System Setup
|
|
{
|
|
profile("Job System")
|
|
host_memory.job_system.running = true
|
|
host_memory.job_system.worker_num = THREAD_JOB_WORKERS
|
|
for & list in host_memory.job_system.job_lists {
|
|
list = {}
|
|
}
|
|
// Determine number of physical cores
|
|
barrier_init(& host_memory.job_hot_reload_sync, THREAD_JOB_WORKERS + 1)
|
|
for id in THREAD_JOB_WORKER_ID_START ..< THREAD_JOB_WORKER_ID_END {
|
|
log_print_fmt("Spawned job worker: %v", cast(WorkerID) id)
|
|
worker_thread := thread_create(host_job_worker_entrypoint, .Normal)
|
|
worker_thread.user_index = int(id)
|
|
host_memory.threads[worker_thread.user_index] = worker_thread
|
|
thread_start(worker_thread)
|
|
}
|
|
}
|
|
barrier_init(& host_memory.lane_job_sync, THREAD_TICK_LANES + THREAD_JOB_WORKERS)
|
|
}
|
|
host_tick_lane()
|
|
|
|
profile_begin("Host Shutdown")
|
|
if thread_memory.id == .Master_Prepper {
|
|
thread_join_multiple(.. host_memory.threads[1:THREAD_TICK_LANES + THREAD_JOB_WORKERS])
|
|
}
|
|
|
|
unload_client_api( & host_memory.client_api )
|
|
|
|
log_print("Succesfuly closed")
|
|
file_close( host_memory.host_logger.file )
|
|
file_rename( str_pfmt_tmp( "%s/sectr.log", Path_Logs), host_memory.path_logger_finalized )
|
|
profile_end()
|
|
|
|
// End profiling
|
|
spall_buffer_destroy(& host_memory.spall_context, & thread_memory.spall_buffer)
|
|
spall_context_destroy( & host_memory.spall_context )
|
|
}
|
|
launch_tick_lane_thread :: proc(id : WorkerID) {
|
|
assert_contextless(thread_memory.id == .Master_Prepper)
|
|
// TODO(Ed): We need to make our own version of this that doesn't allocate memory.
|
|
lane_thread := thread_create(host_tick_lane_entrypoint, .High)
|
|
lane_thread.user_index = int(id)
|
|
host_memory.threads[lane_thread.user_index] = lane_thread
|
|
thread_start(lane_thread)
|
|
}
|
|
|
|
host_tick_lane_entrypoint :: proc(lane_thread: ^SysThread) {
|
|
thread_memory.system_ctx = lane_thread
|
|
thread_memory.id = cast(WorkerID) lane_thread.user_index
|
|
when SHOULD_SETUP_PROFILERS
|
|
{
|
|
thread_memory.spall_buffer = spall_buffer_create(thread_memory.spall_buffer_backing[:], cast(u32) thread_memory.system_ctx.id)
|
|
host_memory.client_api.tick_lane_startup(& thread_memory)
|
|
grime_set_profiler_thread_buffer(& thread_memory.spall_buffer)
|
|
}
|
|
host_tick_lane()
|
|
}
|
|
host_tick_lane :: proc()
|
|
{
|
|
profile(#procedure)
|
|
delta_ns: Duration
|
|
host_tick := time_tick_now()
|
|
|
|
for ; sync_load(& host_memory.tick_running, .Relaxed);
|
|
{
|
|
profile("Host Tick")
|
|
leader := barrier_wait(& host_memory.lane_sync)
|
|
|
|
running: b64 = host_memory.client_api.tick_lane(duration_seconds(delta_ns), delta_ns) == false
|
|
if thread_memory.id == .Master_Prepper {
|
|
sync_store(& host_memory.tick_running, running, .Release)
|
|
}
|
|
host_memory.client_api.clean_frame()
|
|
|
|
delta_ns = time_tick_lap_time( & host_tick )
|
|
host_tick = time_tick_now()
|
|
|
|
// Lanes are synced before doing running check..
|
|
sync_client_api()
|
|
}
|
|
host_lane_shutdown()
|
|
}
|
|
host_lane_shutdown :: proc()
|
|
{
|
|
profile(#procedure)
|
|
if thread_memory.id == .Master_Prepper {
|
|
jobs_enqueued := true
|
|
if jobs_enqueued == false {
|
|
// debug_trap()
|
|
}
|
|
for ; jobs_enqueued; {
|
|
jobs_enqueued = false
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Normal].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Low].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.High].head != nil
|
|
if jobs_enqueued == false {
|
|
// debug_trap()
|
|
}
|
|
}
|
|
sync_store(& host_memory.job_system.running, false, .Release)
|
|
}
|
|
if thread_memory.id != .Master_Prepper {
|
|
spall_buffer_destroy( & host_memory.spall_context, & thread_memory.spall_buffer )
|
|
}
|
|
leader := barrier_wait(& host_memory.lane_job_sync)
|
|
}
|
|
|
|
host_job_worker_entrypoint :: proc(worker_thread: ^SysThread)
|
|
{
|
|
thread_memory.system_ctx = worker_thread
|
|
thread_memory.id = cast(WorkerID) worker_thread.user_index
|
|
when SHOULD_SETUP_PROFILERS
|
|
{
|
|
thread_memory.spall_buffer = spall_buffer_create(thread_memory.spall_buffer_backing[:], cast(u32) thread_memory.system_ctx.id)
|
|
host_memory.client_api.tick_lane_startup(& thread_memory)
|
|
grime_set_profiler_thread_buffer(& thread_memory.spall_buffer)
|
|
}
|
|
jobs_enqueued := false
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Normal].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Low].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.High].head != nil
|
|
delta_ns: Duration
|
|
host_tick := time_tick_now()
|
|
for ; jobs_enqueued || sync_load(& host_memory.job_system.running, .Relaxed);
|
|
{
|
|
profile("Host Job Tick")
|
|
|
|
host_memory.client_api.jobsys_worker_tick(duration_seconds(delta_ns), delta_ns)
|
|
|
|
delta_ns = time_tick_lap_time( & host_tick )
|
|
host_tick = time_tick_now()
|
|
|
|
jobs_enqueued = false
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Normal].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.Low].head != nil
|
|
jobs_enqueued |= host_memory.job_system.job_lists[.High].head != nil
|
|
|
|
if jobs_enqueued == false && sync_load(& host_memory.client_api_hot_reloaded, .Acquire) {
|
|
// Signals to main hread when all jobs have drained.
|
|
leader :=barrier_wait(& host_memory.job_hot_reload_sync)
|
|
// Job threads wait here until client module is back
|
|
leader =barrier_wait(& host_memory.job_hot_reload_sync)
|
|
host_memory.client_api.hot_reload(& host_memory, & thread_memory)
|
|
}
|
|
}
|
|
spall_buffer_destroy( & host_memory.spall_context, & thread_memory.spall_buffer )
|
|
// Were exiting, wait for tick lanes.
|
|
leader := barrier_wait(& host_memory.lane_job_sync)
|
|
}
|
|
|
|
@export
|
|
sync_client_api :: proc()
|
|
{
|
|
profile(#procedure)
|
|
// We don't want any lanes to be in client callstack during a hot-reload
|
|
leader := barrier_wait(& host_memory.lane_sync)
|
|
if thread_memory.id == .Master_Prepper
|
|
{
|
|
write_time, result := file_last_write_time_by_name( Path_Sectr_Module );
|
|
if result == OS_ERROR_NONE && host_memory.client_api.write_time != write_time
|
|
{
|
|
profile("Master_Prepper: Reloading client module")
|
|
sync_store(& host_memory.client_api_hot_reloaded, true, .Release)
|
|
// We nee to wait for the job queue to drain.
|
|
leader = barrier_wait(& host_memory.job_hot_reload_sync)
|
|
{
|
|
version_id := host_memory.client_api.lib_version + 1
|
|
unload_client_api( & host_memory.client_api )
|
|
// Wait for pdb to unlock (linker may still be writting)
|
|
for ; file_is_locked( Path_Sectr_Debug_Symbols ) && file_is_locked( Path_Sectr_Live_Module ); {}
|
|
|
|
thread_sleep( Millisecond * 50 )
|
|
|
|
host_memory.client_api = load_client_api( version_id )
|
|
verify( host_memory.client_api.lib_version != 0, "Failed to hot-reload the sectr module" )
|
|
}
|
|
leader = barrier_wait(& host_memory.job_hot_reload_sync)
|
|
}
|
|
}
|
|
leader = barrier_wait(& host_memory.lane_sync)
|
|
// Lanes are safe to continue.
|
|
if sync_load(& host_memory.client_api_hot_reloaded, .Acquire) {
|
|
host_memory.client_api.hot_reload(& host_memory, & thread_memory)
|
|
}
|
|
}
|
|
unload_client_api :: proc( module : ^Client_API )
|
|
{
|
|
profile(#procedure)
|
|
os_lib_unload( module.lib )
|
|
file_remove( Path_Sectr_Live_Module )
|
|
module^ = {}
|
|
log_print("Unloaded client API")
|
|
}
|
|
|
|
//endregion HOST RUNTIME
|