+ vg_linear_clear( vg_mem.scratch );
+ mdl_context *msky = mdl_load_full( vg_mem.scratch, "models/rs_skydome.mdl" );
+
+ mdl_node *nupper = mdl_node_from_name( msky, "dome_complete" );
+ world.dome_upper = *mdl_node_submesh( msky, nupper, 0 );
+
+ vg_acquire_thread_sync();
+ {
+ mdl_unpack_glmesh( msky, &world.skydome );
+ }
+ vg_release_thread_sync();
+
+ /* Other systems */
+ vg_info( "Loading other world systems\n" );
+
+ vg_loader_step( world_render_init, NULL );
+ vg_loader_step( world_sfd_init, NULL );
+ vg_loader_step( world_water_init, NULL );
+ vg_loader_step( world_gates_init, NULL );
+ vg_loader_step( world_routes_init, NULL );
+
+ /* Allocate dynamic world memory arena */
+ u32 max_size = 76*1024*1024;
+ world.dynamic_vgl = vg_create_linear_allocator( vg_mem.rtmemory, max_size,
+ VG_MEMORY_SYSTEM );
+}
+
+VG_STATIC void world_audio_init(void)
+{
+ u32 size = vg_linear_remaining( vg_audio.audio_pool )
+ - sizeof(vg_linear_allocator);
+
+ world.audio_vgl = vg_create_linear_allocator( vg_audio.audio_pool,
+ size, VG_MEMORY_SYSTEM );
+}
+
+VG_STATIC void world_trigger_achievement( u32 uid )
+{
+ struct logic_achievement *ach = &world.logic_achievements[ uid ];
+
+ if( ach->achieved )
+ return;
+
+ steam_set_achievement( ach->achievement_id );
+ steam_store_achievements();
+
+ ach->achieved = 1;
+}
+
+VG_STATIC void world_run_relay( struct relay_target *rt );
+VG_STATIC void world_trigger_relay( u32 uid )
+{
+ struct logic_relay *relay = &world.logic_relays[ uid ];
+
+ for( int i=0; i<relay->target_count; i++ )