HandmadeHero/project/platform/win32_platform.cpp

1815 lines
57 KiB
C++
Raw Normal View History

2023-09-15 18:35:27 -07:00
/*
TODO : This is not a final platform layer
- Saved game locations
- Getting a handle to our own executable file
- Asset loading path
- Threading (launch a thread)
- Raw Input (support for multiple keyboards)
- Sleep / timeBeginPeriod
- ClipCursor() (for multimonitor support)
- Fullscreen support
- WM_SETCURSOR (control cursor visibility)
- QueryCancelAutoplay
- WM_ACTIVATEAPP (for when not active)
- Blit speed improvemnts (BitBlt)
- Hardware acceleration ( OpenGL or Direct3D or both )
- GetKeyboardLayout (for French keyboards, international WASD support)
*/
// Platform Layer headers
#include "platform.hpp"
#include "jsl.hpp" // Using this to get dualsense controllers
#include "win32.hpp"
2023-09-15 18:35:27 -07:00
// Engine layer headers
2023-09-26 14:26:35 -07:00
#include "engine/engine.hpp"
#include "engine/engine_to_platform_api.hpp"
#include "gen/engine_symbol_table.hpp"
2023-09-23 18:03:33 -07:00
#if 1
2023-09-24 19:37:05 -07:00
// TODO(Ed): Redo these macros properly later.
2023-09-09 20:58:28 -07:00
#define congrats( message ) do { \
JslSetLightColour( 0, (255 << 16) | (215 << 8) ); \
MessageBoxA( 0, message, "Congratulations!", MB_OK | MB_ICONEXCLAMATION ); \
JslSetLightColour( 0, (255 << 8 ) ); \
} while (0)
2023-09-10 07:40:22 -07:00
#define ensure( condition, message ) ensure_impl( condition, message )
inline bool
ensure_impl( bool condition, char const* message ) {
if ( ! condition ) {
JslSetLightColour( 0, (255 << 16) );
MessageBoxA( 0, message, "Ensure Failure", MB_OK | MB_ICONASTERISK );
JslSetLightColour( 0, ( 255 << 8 ) );
}
return condition;
}
2023-09-09 20:58:28 -07:00
#define fatal(message) do { \
JslSetLightColour( 0, (255 << 16) ); \
MessageBoxA( 0, message, "Fatal Error", MB_OK | MB_ICONERROR ); \
JslSetLightColour( 0, (255 << 8 ) ); \
} while (0)
2023-09-23 18:03:33 -07:00
#endif
2023-09-09 20:58:28 -07:00
2023-09-20 11:43:55 -07:00
NS_PLATFORM_BEGIN
using namespace win32;
2023-09-08 18:08:57 -07:00
2023-09-24 19:37:05 -07:00
// This is the "backbuffer" data related to the windowing surface provided by the operating system.
2023-09-09 09:47:06 -07:00
struct OffscreenBuffer
{
BITMAPINFO info;
2023-09-20 21:26:23 -07:00
char _PAD_[4];
void* memory; // Lets use directly mess with the "pixel's memory buffer"
s32 width;
s32 height;
s32 pitch;
s32 bytes_per_pixel;
2023-09-09 09:47:06 -07:00
};
struct WinDimensions
{
u32 width;
u32 height;
2023-09-09 09:47:06 -07:00
};
2023-09-17 18:20:11 -07:00
// TODO : This will def need to be looked over.
2023-09-23 18:03:33 -07:00
struct DirectSoundBuffer
2023-09-17 18:20:11 -07:00
{
LPDIRECTSOUNDBUFFER secondary_buffer;
s16* samples;
u32 secondary_buffer_size;
u32 samples_per_second;
u32 bytes_per_sample;
2023-09-23 18:03:33 -07:00
2023-09-24 19:37:05 -07:00
// TODO(Ed) : Makes math easier...
u32 bytes_per_second;
u32 guard_sample_bytes;
2023-09-24 19:37:05 -07:00
DWORD is_playing;
u32 running_sample_index;
2023-09-24 19:37:05 -07:00
// TODO(Ed) : Should this be in bytes?
u32 latency_sample_count;
2023-09-17 18:20:11 -07:00
};
2023-09-23 18:03:33 -07:00
#pragma region Static Data
2023-09-29 12:58:18 -07:00
global StrPath Path_Root;
global StrPath Path_Binaries;
global StrPath Path_Scratch;
2023-09-26 22:16:41 -07:00
2023-09-23 18:03:33 -07:00
// TODO(Ed) : This is a global for now.
2023-09-24 19:37:05 -07:00
global b32 Running = false;
global b32 Pause_Rendering = false;
2023-09-10 16:56:09 -07:00
2023-09-23 18:03:33 -07:00
global WinDimensions Window_Dimensions;
global OffscreenBuffer Surface_Back_Buffer;
using DirectSoundCreateFn = HRESULT WINAPI (LPGUID lpGuid, LPDIRECTSOUND* ppDS, LPUNKNOWN pUnkOuter );
global DirectSoundCreateFn* direct_sound_create;
2023-09-16 15:41:07 -07:00
2023-09-22 12:13:18 -07:00
constexpr u64 Tick_To_Millisecond = 1000;
constexpr u64 Tick_To_Microsecond = 1000 * 1000;
2023-09-20 11:43:55 -07:00
2023-09-22 12:13:18 -07:00
global u64 Performance_Counter_Frequency;
2023-09-23 18:03:33 -07:00
// As of 2023 the highest refreshrate on the market is 500 hz. I'll just make this higher if something comes out beyond that...
constexpr u32 Monitor_Refresh_Max_Supported = 500;
// Anything at or below the high performance frame-time is too low latency to sleep against the window's scheduler.
constexpr f32 High_Perf_Frametime_MS = 1000.f / 240.f;
global u32 Monitor_Refresh_Hz = 60;
2023-09-28 10:44:43 -07:00
global u32 Engine_Refresh_Hz = Monitor_Refresh_Hz;
2023-09-23 18:03:33 -07:00
global f32 Engine_Frame_Target_MS = 1000.f / scast(f32, Engine_Refresh_Hz);
#pragma endregion Static Data
2023-09-26 22:16:41 -07:00
#pragma region Internal
2023-09-28 10:44:43 -07:00
internal
FILETIME file_get_last_write_time( char const* path )
{
2023-09-29 12:58:18 -07:00
WIN32_FILE_ATTRIBUTE_DATA engine_dll_file_attributes = {};
GetFileAttributesExA( path, GetFileExInfoStandard, & engine_dll_file_attributes );
return engine_dll_file_attributes.ftLastWriteTime;
#if 0
2023-09-28 10:44:43 -07:00
WIN32_FIND_DATAA dll_file_info = {};
HANDLE dll_file_handle = FindFirstFileA( path, & dll_file_info );
if ( dll_file_handle == INVALID_HANDLE_VALUE )
{
FindClose( dll_file_handle );
}
return dll_file_info.ftLastWriteTime;
2023-09-29 12:58:18 -07:00
#endif
2023-09-28 10:44:43 -07:00
}
2023-09-28 12:41:30 -07:00
struct AudioTimeMarker
2023-09-23 18:03:33 -07:00
{
DWORD output_play_cursor;
DWORD output_write_cursor;
DWORD output_location;
DWORD output_byte_count;
2023-09-24 19:37:05 -07:00
DWORD flip_play_curosr;
DWORD flip_write_cursor;
2023-09-24 19:37:05 -07:00
DWORD expected_flip_cursor;
2023-09-23 18:03:33 -07:00
};
2023-09-28 12:41:30 -07:00
#if Build_Debug
2023-09-23 18:03:33 -07:00
internal void
2023-09-24 19:37:05 -07:00
debug_draw_vertical( s32 x_pos, s32 top, s32 bottom, s32 color )
2023-09-23 18:03:33 -07:00
{
2023-09-24 19:37:05 -07:00
if ( top <= 0 )
{
top = 0;
}
2023-09-23 18:03:33 -07:00
if ( bottom > Surface_Back_Buffer.height )
2023-09-23 18:03:33 -07:00
{
bottom = Surface_Back_Buffer.height;
2023-09-24 19:37:05 -07:00
}
if ( x_pos >= 0 && x_pos < Surface_Back_Buffer.width )
2023-09-24 19:37:05 -07:00
{
u8*
pixel_byte = rcast(u8*, Surface_Back_Buffer.memory );
pixel_byte += x_pos * Surface_Back_Buffer.bytes_per_pixel;
pixel_byte += top * Surface_Back_Buffer.pitch;
2023-09-24 19:37:05 -07:00
for ( s32 y = top; y < bottom; ++ y )
{
s32* pixel = rcast(s32*, pixel_byte);
*pixel = color;
2023-09-23 18:03:33 -07:00
pixel_byte += Surface_Back_Buffer.pitch;
2023-09-24 19:37:05 -07:00
}
2023-09-23 18:03:33 -07:00
}
}
inline void
2023-09-24 19:37:05 -07:00
debug_draw_sound_buffer_marker( DirectSoundBuffer* sound_buffer, f32 ratio
2023-09-23 18:03:33 -07:00
, u32 pad_x, u32 pad_y
, u32 top, u32 bottom
, DWORD value, u32 color )
{
2023-09-24 19:37:05 -07:00
// assert( value < sound_buffer->SecondaryBufferSize );
u32 x = pad_x + scast(u32, ratio * scast(f32, value ));
debug_draw_vertical( x, top, bottom, color );
2023-09-23 18:03:33 -07:00
}
internal void
debug_sync_display( DirectSoundBuffer* sound_buffer
2023-09-28 12:41:30 -07:00
, u32 num_markers, AudioTimeMarker* markers
2023-09-24 19:37:05 -07:00
, u32 current_marker
2023-09-23 18:03:33 -07:00
, f32 ms_per_frame )
{
2023-09-24 19:37:05 -07:00
u32 pad_x = 32;
u32 pad_y = 16;
f32 buffers_ratio = scast(f32, Surface_Back_Buffer.width) / (scast(f32, sound_buffer->secondary_buffer_size) * 1);
2023-09-23 18:03:33 -07:00
2023-09-24 19:37:05 -07:00
u32 line_height = 64;
2023-09-23 18:03:33 -07:00
for ( u32 marker_index = 0; marker_index < num_markers; ++ marker_index )
{
2023-09-28 12:41:30 -07:00
AudioTimeMarker* marker = & markers[marker_index];
assert( marker->output_play_cursor < sound_buffer->secondary_buffer_size );
assert( marker->output_write_cursor < sound_buffer->secondary_buffer_size );
assert( marker->output_location < sound_buffer->secondary_buffer_size );
assert( marker->output_byte_count < sound_buffer->secondary_buffer_size );
assert( marker->flip_play_curosr < sound_buffer->secondary_buffer_size );
assert( marker->flip_write_cursor < sound_buffer->secondary_buffer_size );
2023-09-24 19:37:05 -07:00
DWORD play_color = 0x88888888;
DWORD write_color = 0x88800000;
DWORD expected_flip_color = 0xFFFFF000;
DWORD play_window_color = 0xFFFF00FF;
u32 top = pad_y;
u32 bottom = pad_y + line_height;
if ( marker_index == current_marker )
{
play_color = 0xFFFFFFFF;
write_color = 0xFFFF0000;
top += pad_y + line_height;
bottom += pad_y + line_height;
u32 row_2_top = top;
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->output_play_cursor, play_color );
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->output_write_cursor, write_color );
2023-09-24 19:37:05 -07:00
play_color = 0xFFFFFFFF;
write_color = 0xFFFF0000;
top += pad_y + line_height;
bottom += pad_y + line_height;
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->output_location, play_color );
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->output_location + marker->output_byte_count, write_color );
2023-09-24 19:37:05 -07:00
play_color = 0xFFFFFFFF;
write_color = 0xFFFF0000;
top += pad_y + line_height;
bottom += pad_y + line_height;
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, row_2_top, bottom, marker->expected_flip_cursor, expected_flip_color );
2023-09-24 19:37:05 -07:00
}
DWORD play_window = marker->flip_play_curosr + 480 * sound_buffer->bytes_per_sample;
2023-09-24 19:37:05 -07:00
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->flip_play_curosr, play_color );
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, play_window, play_window_color );
debug_draw_sound_buffer_marker( sound_buffer, buffers_ratio, pad_x, pad_y, top, bottom, marker->flip_write_cursor, write_color );
2023-09-23 18:03:33 -07:00
}
}
2023-09-20 11:43:55 -07:00
#endif
2023-09-28 10:44:43 -07:00
#pragma region Direct Sound
2023-09-10 07:40:22 -07:00
internal void
2023-09-23 18:03:33 -07:00
init_sound(HWND window_handle, DirectSoundBuffer* sound_buffer )
2023-09-10 07:40:22 -07:00
{
// Load library
HMODULE sound_library = LoadLibraryA( "dsound.dll" );
if ( ! ensure(sound_library, "Failed to load direct sound library" ) )
{
// TOOD : Diagnostic
return;
}
// Get direct sound object
2023-09-26 14:26:35 -07:00
direct_sound_create = get_procedure_from_library< DirectSoundCreateFn >( sound_library, "DirectSoundCreate" );
2023-09-10 11:14:47 -07:00
if ( ! ensure( direct_sound_create, "Failed to get direct_sound_create_procedure" ) )
2023-09-10 07:40:22 -07:00
{
// TOOD : Diagnostic
return;
}
LPDIRECTSOUND direct_sound;
if ( ! SUCCEEDED(direct_sound_create( 0, & direct_sound, 0 )) )
{
// TODO : Diagnostic
}
if ( ! SUCCEEDED( direct_sound->SetCooperativeLevel(window_handle, DSSCL_PRIORITY) ) )
{
// TODO : Diagnostic
}
WAVEFORMATEX
wave_format {};
wave_format.wFormatTag = WAVE_FORMAT_PCM; /* format type */
wave_format.nChannels = 2; /* number of channels (i.e. mono, stereo...) */
wave_format.nSamplesPerSec = scast(u32, sound_buffer->samples_per_second); /* sample rate */
2023-09-10 07:40:22 -07:00
wave_format.wBitsPerSample = 16; /* number of bits per sample of mono data */
wave_format.nBlockAlign = wave_format.nChannels * wave_format.wBitsPerSample / 8 ; /* block size of data */
wave_format.nAvgBytesPerSec = wave_format.nSamplesPerSec * wave_format.nBlockAlign; /* for buffer estimation */
wave_format.cbSize = 0; /* the count in bytes of the size of */
LPDIRECTSOUNDBUFFER primary_buffer;
{
DSBUFFERDESC
buffer_description { sizeof(buffer_description) };
buffer_description.dwFlags = DSBCAPS_PRIMARYBUFFER;
buffer_description.dwBufferBytes = 0;
if ( ! SUCCEEDED( direct_sound->CreateSoundBuffer( & buffer_description, & primary_buffer, 0 ) ))
{
// TODO : Diagnostic
}
if ( ! SUCCEEDED( primary_buffer->SetFormat( & wave_format ) ) )
{
// TODO : Diagnostic
}
}
2023-09-10 11:14:47 -07:00
DSBUFFERDESC
buffer_description { sizeof(buffer_description) };
2023-09-23 18:03:33 -07:00
buffer_description.dwFlags = DSBCAPS_GETCURRENTPOSITION2;
buffer_description.dwBufferBytes = sound_buffer->secondary_buffer_size;
2023-09-10 11:14:47 -07:00
buffer_description.lpwfxFormat = & wave_format;
2023-09-10 07:40:22 -07:00
if ( ! SUCCEEDED( direct_sound->CreateSoundBuffer( & buffer_description, & sound_buffer->secondary_buffer, 0 ) ))
2023-09-10 11:14:47 -07:00
{
// TODO : Diagnostic
}
if ( ! SUCCEEDED( sound_buffer->secondary_buffer->SetFormat( & wave_format ) ) )
2023-09-10 11:14:47 -07:00
{
// TODO : Diagnostic
2023-09-10 07:40:22 -07:00
}
}
2023-09-10 16:56:09 -07:00
internal void
2023-09-23 18:03:33 -07:00
ds_clear_sound_buffer( DirectSoundBuffer* sound_buffer )
2023-09-10 16:56:09 -07:00
{
2023-09-16 15:41:07 -07:00
LPVOID region_1;
DWORD region_1_size;
LPVOID region_2;
DWORD region_2_size;
HRESULT ds_lock_result = sound_buffer->secondary_buffer->Lock( 0, sound_buffer->secondary_buffer_size
2023-09-16 15:41:07 -07:00
, & region_1, & region_1_size
, & region_2, & region_2_size
, 0 );
if ( ! SUCCEEDED( ds_lock_result ) )
2023-09-10 16:56:09 -07:00
{
2023-09-16 15:41:07 -07:00
return;
}
2023-09-10 16:56:09 -07:00
2023-09-16 15:41:07 -07:00
u8* sample_out = rcast( u8*, region_1 );
for ( DWORD byte_index = 0; byte_index < region_1_size; ++ byte_index )
{
*sample_out = 0;
2023-09-10 16:56:09 -07:00
++ sample_out;
2023-09-16 15:41:07 -07:00
}
2023-09-10 16:56:09 -07:00
2023-09-16 15:41:07 -07:00
sample_out = rcast( u8*, region_2 );
for ( DWORD byte_index = 0; byte_index < region_2_size; ++ byte_index )
{
*sample_out = 0;
2023-09-10 16:56:09 -07:00
++ sample_out;
}
2023-09-16 15:41:07 -07:00
if ( ! SUCCEEDED( sound_buffer->secondary_buffer->Unlock( region_1, region_1_size, region_2, region_2_size ) ))
2023-09-16 15:41:07 -07:00
{
return;
}
2023-09-10 16:56:09 -07:00
}
internal void
2023-09-23 18:03:33 -07:00
ds_fill_sound_buffer( DirectSoundBuffer* sound_buffer, DWORD byte_to_lock, DWORD bytes_to_write )
2023-09-10 16:56:09 -07:00
{
LPVOID region_1;
DWORD region_1_size;
LPVOID region_2;
DWORD region_2_size;
HRESULT ds_lock_result = sound_buffer->secondary_buffer->Lock( byte_to_lock, bytes_to_write
2023-09-10 16:56:09 -07:00
, & region_1, & region_1_size
, & region_2, & region_2_size
, 0 );
if ( ! SUCCEEDED( ds_lock_result ) )
{
return;
}
// TODO : Assert that region sizes are valid
DWORD region_1_sample_count = region_1_size / sound_buffer->bytes_per_sample;
2023-09-16 15:41:07 -07:00
s16* sample_out = rcast( s16*, region_1 );
s16* sample_in = sound_buffer->samples;
2023-09-16 15:41:07 -07:00
for ( DWORD sample_index = 0; sample_index < region_1_sample_count; ++ sample_index )
{
*sample_out = *sample_in;
++ sample_out;
++ sample_in;
*sample_out = *sample_in;
++ sample_out;
++ sample_in;
++ sound_buffer->running_sample_index;
2023-09-16 15:41:07 -07:00
}
DWORD region_2_sample_count = region_2_size / sound_buffer->bytes_per_sample;
2023-09-16 15:41:07 -07:00
sample_out = rcast( s16*, region_2 );
for ( DWORD sample_index = 0; sample_index < region_2_sample_count; ++ sample_index )
{
*sample_out = *sample_in;
++ sample_out;
++ sample_in;
*sample_out = *sample_in;
++ sample_out;
++ sample_in;
++ sound_buffer->running_sample_index;
2023-09-16 15:41:07 -07:00
}
2023-09-10 16:56:09 -07:00
if ( ! SUCCEEDED( sound_buffer->secondary_buffer->Unlock( region_1, region_1_size, region_2, region_2_size ) ))
2023-09-10 16:56:09 -07:00
{
return;
}
}
2023-09-28 10:44:43 -07:00
#pragma endregion Direct Sound
#pragma region Input
// Max controllers for the platform layer and thus for all other layers is 4. (Sanity and xinput limit)
constexpr u32 Max_Controllers = 4;
using JSL_DeviceHandle = int;
using EngineXInputPadStates = engine::XInputPadState[ Max_Controllers ];
using EngineDSPadStates = engine::DualsensePadState[Max_Controllers];
internal void
input_process_digital_btn( engine::DigitalBtn* old_state, engine::DigitalBtn* new_state, u32 raw_btns, u32 btn_flag )
{
#define had_transition() ( old_state->ended_down != new_state->ended_down )
new_state->ended_down = (raw_btns & btn_flag) > 0;
2023-09-28 10:44:43 -07:00
if ( had_transition() )
new_state->half_transitions += 1;
2023-09-28 10:44:43 -07:00
else
new_state->half_transitions = 0;
2023-09-28 10:44:43 -07:00
#undef had_transition
}
internal f32
jsl_input_process_axis_value( f32 value, f32 deadzone_threshold )
{
f32 result = 0;
if ( value < -deadzone_threshold )
{
result = (value + deadzone_threshold ) / (1.0f - deadzone_threshold );
if (result < -1.0f)
result = -1.0f; // Clamp to ensure it doesn't go below -1
}
else if ( value > deadzone_threshold )
{
result = (value - deadzone_threshold ) / (1.0f - deadzone_threshold );
if (result > 1.0f)
result = 1.0f; // Clamp to ensure it doesn't exceed 1
}
return result;
}
internal f32
xinput_process_axis_value( s16 value, s16 deadzone_threshold )
{
f32 result = 0;
if ( value < -deadzone_threshold )
{
result = scast(f32, value + deadzone_threshold) / (32768.0f - scast(f32, deadzone_threshold));
}
else if ( value > deadzone_threshold )
{
result = scast(f32, value + deadzone_threshold) / (32767.0f - scast(f32, deadzone_threshold));
}
return result;
}
internal void
poll_input( HWND window_handle, engine::InputState* input, u32 jsl_num_devices, JSL_DeviceHandle* jsl_device_handles
2023-09-28 10:44:43 -07:00
, engine::KeyboardState* old_keyboard, engine::KeyboardState* new_keyboard
, engine::MousesState* old_mouse, engine::MousesState* new_mouse
, EngineXInputPadStates* old_xpads, EngineXInputPadStates* new_xpads
, EngineDSPadStates* old_ds_pads, EngineDSPadStates* new_ds_pads )
2023-09-28 10:44:43 -07:00
{
// Keyboard Polling
// Keyboards are unified for now.
{
constexpr u32 is_down = 0x80000000;
2023-09-30 12:40:27 -07:00
input_process_digital_btn( & old_keyboard->_1, & new_keyboard->_1, GetAsyncKeyState( '1' ), is_down );
input_process_digital_btn( & old_keyboard->_2, & new_keyboard->_2, GetAsyncKeyState( '2' ), is_down );
input_process_digital_btn( & old_keyboard->_3, & new_keyboard->_3, GetAsyncKeyState( '3' ), is_down );
input_process_digital_btn( & old_keyboard->_4, & new_keyboard->_4, GetAsyncKeyState( '4' ), is_down );
input_process_digital_btn( & old_keyboard->Q, & new_keyboard->Q, GetAsyncKeyState( 'Q' ), is_down );
input_process_digital_btn( & old_keyboard->E, & new_keyboard->E, GetAsyncKeyState( 'E' ), is_down );
input_process_digital_btn( & old_keyboard->W, & new_keyboard->W, GetAsyncKeyState( 'W' ), is_down );
input_process_digital_btn( & old_keyboard->A, & new_keyboard->A, GetAsyncKeyState( 'A' ), is_down );
input_process_digital_btn( & old_keyboard->S, & new_keyboard->S, GetAsyncKeyState( 'S' ), is_down );
input_process_digital_btn( & old_keyboard->D, & new_keyboard->D, GetAsyncKeyState( 'D' ), is_down );
input_process_digital_btn( & old_keyboard->K, & new_keyboard->K, GetAsyncKeyState( 'K' ), is_down );
input_process_digital_btn( & old_keyboard->L, & new_keyboard->L, GetAsyncKeyState( 'L' ), is_down );
input_process_digital_btn( & old_keyboard->escape, & new_keyboard->escape, GetAsyncKeyState( VK_ESCAPE ), is_down );
input_process_digital_btn( & old_keyboard->backspace, & new_keyboard->backspace, GetAsyncKeyState( VK_BACK ), is_down );
input_process_digital_btn( & old_keyboard->up, & new_keyboard->up, GetAsyncKeyState( VK_UP ), is_down );
input_process_digital_btn( & old_keyboard->down, & new_keyboard->down, GetAsyncKeyState( VK_DOWN ), is_down );
input_process_digital_btn( & old_keyboard->left, & new_keyboard->left, GetAsyncKeyState( VK_LEFT ), is_down );
input_process_digital_btn( & old_keyboard->right, & new_keyboard->right, GetAsyncKeyState( VK_RIGHT ), is_down );
input_process_digital_btn( & old_keyboard->space, & new_keyboard->space, GetAsyncKeyState( VK_SPACE ), is_down );
input_process_digital_btn( & old_keyboard->pause, & new_keyboard->pause, GetAsyncKeyState( VK_PAUSE ), is_down );
2023-09-30 12:40:27 -07:00
input_process_digital_btn( & old_keyboard->left_alt, & new_keyboard->left_alt, GetAsyncKeyState( VK_LMENU ), is_down );
input_process_digital_btn( & old_keyboard->right_alt, & new_keyboard->right_alt, GetAsyncKeyState( VK_RMENU ), is_down );
input_process_digital_btn( & old_keyboard->left_shift, & new_keyboard->left_shift, GetAsyncKeyState( VK_LSHIFT ), is_down );
2023-09-30 12:40:27 -07:00
input_process_digital_btn( & old_keyboard->right_shift, & new_keyboard->right_shift, GetAsyncKeyState( VK_RSHIFT ), is_down );
input->controllers[0].keyboard = new_keyboard;
}
// Mouse polling
{
// input->Controllers[0].Mouse = {};
2023-09-29 12:58:18 -07:00
2023-09-28 10:44:43 -07:00
constexpr u32 is_down = 0x80000000;
input_process_digital_btn( & old_mouse->left, & new_mouse->left, GetAsyncKeyState( VK_LBUTTON ), is_down );
input_process_digital_btn( & old_mouse->middle, & new_mouse->middle, GetAsyncKeyState( VK_MBUTTON ), is_down );
input_process_digital_btn( & old_mouse->right, & new_mouse->right, GetAsyncKeyState( VK_RBUTTON ), is_down );
POINT mouse_pos;
GetCursorPos( & mouse_pos );
ScreenToClient( window_handle, & mouse_pos );
new_mouse->vertical_wheel = {};
new_mouse->horizontal_wheel = {};
new_mouse->X.end = (f32)mouse_pos.x;
new_mouse->Y.end = (f32)mouse_pos.y;
input->controllers[0].mouse = new_mouse;
2023-09-28 10:44:43 -07:00
}
// XInput Polling
// TODO(Ed) : Should we poll this more frequently?
for ( DWORD controller_index = 0; controller_index < Max_Controllers; ++ controller_index )
{
XINPUT_STATE controller_state;
b32 xinput_detected = xinput_get_state( controller_index, & controller_state ) == XI_PluggedIn;
if ( xinput_detected )
{
XINPUT_GAMEPAD* xpad = & controller_state.Gamepad;
engine::XInputPadState* old_xpad = old_xpads[ controller_index ];
engine::XInputPadState* new_xpad = new_xpads[ controller_index ];
input_process_digital_btn( & old_xpad->dpad.up, & new_xpad->dpad.up, xpad->wButtons, XINPUT_GAMEPAD_DPAD_UP );
input_process_digital_btn( & old_xpad->dpad.down, & new_xpad->dpad.down, xpad->wButtons, XINPUT_GAMEPAD_DPAD_DOWN );
input_process_digital_btn( & old_xpad->dpad.left, & new_xpad->dpad.left, xpad->wButtons, XINPUT_GAMEPAD_DPAD_LEFT );
input_process_digital_btn( & old_xpad->dpad.right, & new_xpad->dpad.right, xpad->wButtons, XINPUT_GAMEPAD_DPAD_RIGHT );
2023-09-28 10:44:43 -07:00
input_process_digital_btn( & old_xpad->Y, & new_xpad->Y, xpad->wButtons, XINPUT_GAMEPAD_Y );
input_process_digital_btn( & old_xpad->A, & new_xpad->A, xpad->wButtons, XINPUT_GAMEPAD_A );
input_process_digital_btn( & old_xpad->B, & new_xpad->B, xpad->wButtons, XINPUT_GAMEPAD_B );
input_process_digital_btn( & old_xpad->X, & new_xpad->X, xpad->wButtons, XINPUT_GAMEPAD_X );
input_process_digital_btn( & old_xpad->back, & new_xpad->back, xpad->wButtons, XINPUT_GAMEPAD_BACK );
input_process_digital_btn( & old_xpad->start, & new_xpad->start, xpad->wButtons, XINPUT_GAMEPAD_START );
2023-09-28 10:44:43 -07:00
input_process_digital_btn( & old_xpad->left_shoulder, & new_xpad->left_shoulder, xpad->wButtons, XINPUT_GAMEPAD_LEFT_SHOULDER );
input_process_digital_btn( & old_xpad->right_shoulder, & new_xpad->right_shoulder, xpad->wButtons, XINPUT_GAMEPAD_RIGHT_SHOULDER );
2023-09-28 10:44:43 -07:00
new_xpad->stick.left.X.start = old_xpad->stick.left.X.end;
new_xpad->stick.left.Y.start = old_xpad->stick.left.Y.end;
2023-09-28 10:44:43 -07:00
f32 left_x = xinput_process_axis_value( xpad->sThumbLX, XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE );
f32 left_y = xinput_process_axis_value( xpad->sThumbLY, XINPUT_GAMEPAD_LEFT_THUMB_DEADZONE );
// TODO(Ed) : Min/Max macros!!!
new_xpad->stick.left.X.min = new_xpad->stick.left.X.max = new_xpad->stick.left.X.end = left_x;
new_xpad->stick.left.Y.min = new_xpad->stick.left.Y.max = new_xpad->stick.left.Y.end = left_y;
2023-09-28 10:44:43 -07:00
// TODO(Ed): Make this actually an average for later
new_xpad->stick.left.X.average = left_x;
new_xpad->stick.left.Y.average = left_y;
2023-09-28 10:44:43 -07:00
input->controllers[ controller_index ].xpad = new_xpad;
2023-09-28 10:44:43 -07:00
}
else
{
input->controllers[ controller_index ].xpad = nullptr;
2023-09-28 10:44:43 -07:00
}
}
// JSL Input Polling
for ( u32 jsl_device_index = 0; jsl_device_index < jsl_num_devices; ++ jsl_device_index )
{
if ( ! JslStillConnected( jsl_device_handles[ jsl_device_index ] ) )
{
OutputDebugStringA( "Error: JSLStillConnected returned false\n" );
continue;
}
// TODO : Won't support more than 4 for now... (or prob ever)
if ( jsl_device_index > 4 )
break;
JOY_SHOCK_STATE state = JslGetSimpleState( jsl_device_handles[ jsl_device_index ] );
// For now we're assuming anything that is detected via JSL is a dualsense pad.
// We'll eventually add support possibly for the nintendo pro controller.
engine::DualsensePadState* old_ds_pad = old_ds_pads[ jsl_device_index ];
engine::DualsensePadState* new_ds_pad = new_ds_pads[ jsl_device_index ];
input_process_digital_btn( & old_ds_pad->dpad.up, & new_ds_pad->dpad.up, state.buttons, JSMASK_UP );
input_process_digital_btn( & old_ds_pad->dpad.down, & new_ds_pad->dpad.down, state.buttons, JSMASK_DOWN );
input_process_digital_btn( & old_ds_pad->dpad.left, & new_ds_pad->dpad.left, state.buttons, JSMASK_LEFT );
input_process_digital_btn( & old_ds_pad->dpad.right, & new_ds_pad->dpad.right, state.buttons, JSMASK_RIGHT );
2023-09-28 10:44:43 -07:00
input_process_digital_btn( & old_ds_pad->triangle, & new_ds_pad->triangle, state.buttons, JSMASK_N );
input_process_digital_btn( & old_ds_pad->cross, & new_ds_pad->cross, state.buttons, JSMASK_S );
input_process_digital_btn( & old_ds_pad->square, & new_ds_pad->square, state.buttons, JSMASK_W );
input_process_digital_btn( & old_ds_pad->circle, & new_ds_pad->circle, state.buttons, JSMASK_E );
2023-09-28 10:44:43 -07:00
input_process_digital_btn( & old_ds_pad->share, & new_ds_pad->share, state.buttons, JSMASK_SHARE );
input_process_digital_btn( & old_ds_pad->options, & new_ds_pad->options, state.buttons, JSMASK_OPTIONS );
2023-09-28 10:44:43 -07:00
input_process_digital_btn( & old_ds_pad->L1, & new_ds_pad->L1, state.buttons, JSMASK_L );
input_process_digital_btn( & old_ds_pad->R1, & new_ds_pad->R1, state.buttons, JSMASK_R );
new_ds_pad->stick.left.X.start = old_ds_pad->stick.left.X.end;
new_ds_pad->stick.left.Y.start = old_ds_pad->stick.left.Y.end;
2023-09-28 10:44:43 -07:00
// Joyshock abstracts the sticks to a float value already for us of -1.f to 1.f.
// We'll assume a deadzone of 10% for now.
f32 left_x = jsl_input_process_axis_value( state.stickLX, 0.1f );
f32 left_y = jsl_input_process_axis_value( state.stickLY, 0.1f );
new_ds_pad->stick.left.X.min = new_ds_pad->stick.left.X.max = new_ds_pad->stick.left.X.end = left_x;
new_ds_pad->stick.left.Y.min = new_ds_pad->stick.left.Y.max = new_ds_pad->stick.left.Y.end = left_y;
2023-09-28 10:44:43 -07:00
// TODO(Ed): Make this actually an average for later
new_ds_pad->stick.left.X.average = left_x;
new_ds_pad->stick.left.Y.average = left_y;
2023-09-28 10:44:43 -07:00
input->controllers[ jsl_device_index ].ds_pad = new_ds_pad;
2023-09-28 10:44:43 -07:00
}
}
#pragma endregion Input
#pragma region Timing
inline f32
timing_get_ms_elapsed( u64 start, u64 end )
{
u64 delta = (end - start) * Tick_To_Millisecond;
f32 result = scast(f32, delta) / scast(f32, Performance_Counter_Frequency);
return result;
}
inline f32
timing_get_seconds_elapsed( u64 start, u64 end )
{
u64 delta = end - start;
f32 result = scast(f32, delta) / scast(f32, Performance_Counter_Frequency);
return result;
}
inline f32
timing_get_us_elapsed( u64 start, u64 end )
{
u64 delta = (end - start) * Tick_To_Microsecond;
f32 result = scast(f32, delta) / scast(f32, Performance_Counter_Frequency);
return result;
}
inline u64
timing_get_wall_clock()
{
u64 clock;
QueryPerformanceCounter( rcast( LARGE_INTEGER*, & clock) );
return clock;
}
#pragma endregion Timing
2023-09-10 16:56:09 -07:00
2023-09-10 07:40:22 -07:00
internal WinDimensions
get_window_dimensions( HWND window_handle )
2023-09-09 09:47:06 -07:00
{
RECT client_rect;
GetClientRect( window_handle, & client_rect );
WinDimensions result;
result.width = client_rect.right - client_rect.left;
result.height = client_rect.bottom - client_rect.top;
2023-09-09 09:47:06 -07:00
return result;
}
2023-09-28 10:44:43 -07:00
internal void
display_buffer_in_window( HDC device_context, u32 window_width, u32 window_height, OffscreenBuffer* buffer
, u32 x, u32 y
, u32 width, u32 height )
{
// TODO(Ed) : Aspect ratio correction
StretchDIBits( device_context
#if 0
, x, y, width, height
, x, y, width, height
#endif
, 0, 0, buffer->width, buffer->height
2023-09-29 12:58:18 -07:00
// , 0, 0, window_width, window_height
, 0, 0, buffer->width, buffer->height
, buffer->memory, & buffer->info
2023-09-28 10:44:43 -07:00
, DIB_ColorTable_RGB, RO_Source_To_Dest );
}
2023-09-08 21:01:53 -07:00
internal void
2023-09-09 09:47:06 -07:00
resize_dib_section( OffscreenBuffer* buffer, u32 width, u32 height )
2023-09-08 21:01:53 -07:00
{
// TODO(Ed) : Bulletproof memory handling here for the bitmap memory
if ( buffer->memory )
2023-09-08 21:01:53 -07:00
{
VirtualFree( buffer->memory, 0, MEM_RELEASE );
2023-09-08 21:01:53 -07:00
}
buffer->width = width;
buffer->height = height;
buffer->bytes_per_pixel = 4;
buffer->pitch = buffer->width * buffer->bytes_per_pixel;
// Negative means top-down in the context of the biHeight
# define Top_Down -
2023-09-09 09:47:06 -07:00
BITMAPINFOHEADER&
header = buffer->info.bmiHeader;
header.biSize = sizeof( buffer->info.bmiHeader );
header.biWidth = buffer->width;
header.biHeight = Top_Down buffer->height;
2023-09-09 09:47:06 -07:00
header.biPlanes = 1;
header.biBitCount = 32; // Need 24, but want 32 ( alignment )
header.biCompression = BI_RGB_Uncompressed;
// header.biSizeImage = 0;
// header.biXPelsPerMeter = 0;
// header.biYPelsPerMeter = 0;
// header.biClrUsed = 0;
// header.biClrImportant = 0;
# undef Top_Down
// We want to "touch" a pixel on every 4-byte boundary
u32 BitmapMemorySize = (buffer->width * buffer->height) * buffer->bytes_per_pixel;
buffer->memory = VirtualAlloc( NULL, BitmapMemorySize, MEM_Commit_Zeroed | MEM_Reserve, Page_Read_Write );
// TODO(Ed) : Clear to black
2023-09-08 21:01:53 -07:00
}
internal LRESULT CALLBACK
2023-09-23 18:03:33 -07:00
main_window_callback( HWND handle
, UINT system_messages
, WPARAM w_param
, LPARAM l_param )
2023-09-08 18:08:57 -07:00
{
2023-09-15 18:35:27 -07:00
LRESULT result = 0;
2023-09-08 18:08:57 -07:00
switch ( system_messages )
{
case WM_ACTIVATEAPP:
{
2023-09-28 10:44:43 -07:00
if ( scast( bool, w_param ) == true )
{
SetLayeredWindowAttributes( handle, RGB(0, 0, 0), 255, LWA_Alpha );
}
else
{
SetLayeredWindowAttributes( handle, RGB(0, 0, 0), 100, LWA_Alpha );
}
2023-09-08 18:08:57 -07:00
}
break;
case WM_CLOSE:
{
2023-09-08 21:01:53 -07:00
// TODO(Ed) : Handle with a message to the user
Running = false;
2023-09-08 18:08:57 -07:00
}
break;
case WM_DESTROY:
{
2023-09-08 21:01:53 -07:00
// TODO(Ed) : Handle with as an error and recreate the window
Running = false;
2023-09-08 18:08:57 -07:00
}
break;
case WM_PAINT:
{
PAINTSTRUCT info;
HDC device_context = BeginPaint( handle, & info );
u32 x = info.rcPaint.left;
u32 y = info.rcPaint.top;
u32 width = info.rcPaint.right - info.rcPaint.left;
u32 height = info.rcPaint.bottom - info.rcPaint.top;
2023-09-09 09:47:06 -07:00
WinDimensions dimensions = get_window_dimensions( handle );
2023-09-08 18:08:57 -07:00
display_buffer_in_window( device_context, dimensions.width, dimensions.height, &Surface_Back_Buffer
2023-09-08 18:08:57 -07:00
, x, y
2023-09-08 21:01:53 -07:00
, width, height );
2023-09-08 18:08:57 -07:00
EndPaint( handle, & info );
}
break;
case WM_MOUSEMOVE:
{
RECT rect;
POINT pt = { LOWORD(l_param), HIWORD(l_param) };
GetClientRect(handle, &rect);
if (PtInRect(&rect, pt))
{
// Hide the cursor when it's inside the window
while (ShowCursor(FALSE) >= 0);
}
else
{
// Show the cursor when it's outside the window
while (ShowCursor(TRUE) < 0);
}
}
break;
2023-09-08 18:08:57 -07:00
case WM_SIZE:
{
}
break;
default:
{
result = DefWindowProc( handle, system_messages, w_param, l_param );
}
}
return result;
}
2023-09-16 15:41:07 -07:00
2023-09-17 18:20:11 -07:00
internal void
process_pending_window_messages( engine::KeyboardState* keyboard, engine::MousesState* mouse )
2023-09-17 18:20:11 -07:00
{
2023-09-21 23:16:40 -07:00
MSG window_msg_info;
while ( PeekMessageA( & window_msg_info, 0, 0, 0, PM_Remove_Messages_From_Queue ) )
{
if ( window_msg_info.message == WM_QUIT )
{
OutputDebugStringA("WM_QUIT\n");
Running = false;
}
2023-09-17 18:20:11 -07:00
2023-09-21 23:16:40 -07:00
// Keyboard input handling
switch (window_msg_info.message)
2023-09-21 23:16:40 -07:00
{
// I rather do this with GetAsyncKeyState...
case WM_SYSKEYDOWN:
case WM_SYSKEYUP:
{
WPARAM vk_code = window_msg_info.wParam;
b32 is_down = scast(b32, (window_msg_info.lParam >> 31) == 0 );
b32 was_down = scast(b32, (window_msg_info.lParam >> 30) );
b32 alt_down = scast(b32, (window_msg_info.lParam & (1 << 29)) );
switch ( vk_code )
{
case VK_F4:
{
if ( alt_down )
Running = false;
}
break;
}
}
break;
case WM_MOUSEWHEEL:
{
// This captures the vertical scroll value
int verticalScroll = GET_WHEEL_DELTA_WPARAM(window_msg_info.wParam);
mouse->vertical_wheel.end += scast(f32, verticalScroll);
}
break;
case WM_MOUSEHWHEEL:
{
// This captures the horizontal scroll value
int horizontalScroll = GET_WHEEL_DELTA_WPARAM(window_msg_info.wParam);
mouse->horizontal_wheel.end += scast( f32, horizontalScroll);
}
break;
2023-09-21 23:16:40 -07:00
default:
TranslateMessage( & window_msg_info );
DispatchMessageW( & window_msg_info );
}
}
2023-09-20 21:26:23 -07:00
}
2023-09-28 10:44:43 -07:00
2023-09-26 22:16:41 -07:00
#pragma endregion Internal
2023-09-26 14:26:35 -07:00
#pragma region Platfom API
2023-09-26 22:16:41 -07:00
#if Build_Development
2023-09-28 12:41:30 -07:00
void debug_set_pause_rendering( b32 value )
{
Pause_Rendering = value;
}
#endif
2023-09-28 10:44:43 -07:00
b32 file_check_exists( Str path )
2023-09-26 22:16:41 -07:00
{
2023-09-28 10:44:43 -07:00
HANDLE file_handle = CreateFileA( path
, GENERIC_READ, FILE_SHARE_READ, 0
, OPEN_EXISTING, 0, 0
);
if ( file_handle != INVALID_HANDLE_VALUE )
{
CloseHandle( file_handle );
return true;
}
return false;
}
void file_close( File* file )
{
HANDLE handle = pcast(HANDLE, file->opaque_handle);
2023-09-28 10:44:43 -07:00
if ( handle == INVALID_HANDLE_VALUE )
return;
CloseHandle( handle );
if ( file->data )
2023-09-26 22:16:41 -07:00
{
2023-09-28 10:44:43 -07:00
// TODO(Ed): This should use our persistent memory block.
VirtualFree( file->data, 0, MEM_Release);
2023-09-26 22:16:41 -07:00
}
2023-09-28 10:44:43 -07:00
*file = {};
2023-09-26 22:16:41 -07:00
}
2023-09-26 14:26:35 -07:00
2023-09-28 10:44:43 -07:00
b32 file_delete( Str path )
2023-09-26 14:26:35 -07:00
{
2023-09-28 10:44:43 -07:00
return DeleteFileA( path );
}
2023-09-26 22:16:41 -07:00
2023-09-28 10:44:43 -07:00
b32 file_read_stream( File* file, u32 content_size, void* content_memory )
{
HANDLE file_handle;
if ( file->opaque_handle == nullptr )
2023-09-28 10:44:43 -07:00
{
file_handle = CreateFileA( file->path
2023-09-28 10:44:43 -07:00
, GENERIC_READ, FILE_SHARE_READ, 0
, OPEN_EXISTING, 0, 0
);
if ( file_handle == INVALID_HANDLE_VALUE )
{
// TODO : Logging
return {};
}
file->opaque_handle = file_handle;
2023-09-28 10:44:43 -07:00
}
else
{
file_handle = pcast(HANDLE, file->opaque_handle );
2023-09-28 10:44:43 -07:00
}
u32 bytes_read;
if ( ReadFile( file_handle, content_memory, content_size, rcast(LPDWORD, &bytes_read), 0 ) == false )
{
// TODO : Logging
return {};
}
if ( bytes_read != content_size )
{
// TODO : Logging
return {};
}
return bytes_read;
}
b32 file_read_content( File* file )
{
HANDLE file_handle = CreateFileA( file->path
2023-09-26 22:16:41 -07:00
, GENERIC_READ, FILE_SHARE_READ, 0
, OPEN_EXISTING, 0, 0
);
if ( file_handle == INVALID_HANDLE_VALUE )
2023-09-26 14:26:35 -07:00
{
2023-09-26 22:16:41 -07:00
// TODO(Ed) : Logging
2023-09-28 10:44:43 -07:00
return {};
2023-09-26 22:16:41 -07:00
}
2023-09-26 14:26:35 -07:00
2023-09-28 10:44:43 -07:00
u32 size;
GetFileSizeEx( file_handle, rcast(LARGE_INTEGER*, &size) );
if ( size == 0 )
2023-09-26 22:16:41 -07:00
{
// TODO(Ed) : Logging
2023-09-28 10:44:43 -07:00
CloseHandle( file_handle );
return {};
2023-09-26 22:16:41 -07:00
}
2023-09-28 10:44:43 -07:00
// TODO(Ed) : This should use our memory block.
file->data = rcast(HANDLE*, VirtualAlloc( 0, sizeof(HANDLE) + size, MEM_Commit_Zeroed | MEM_Reserve, Page_Read_Write ));
file->size = size;
file->opaque_handle = file_handle;
2023-09-26 14:26:35 -07:00
2023-09-26 22:16:41 -07:00
u32 bytes_read;
if ( ReadFile( file_handle, file->data, file->size, rcast(LPDWORD, &bytes_read), 0 ) == false )
2023-09-26 14:26:35 -07:00
{
2023-09-26 22:16:41 -07:00
// TODO(Ed) : Logging
2023-09-28 10:44:43 -07:00
CloseHandle( file_handle );
2023-09-26 22:16:41 -07:00
return {};
2023-09-26 14:26:35 -07:00
}
if ( bytes_read != file->size )
2023-09-26 14:26:35 -07:00
{
2023-09-26 22:16:41 -07:00
// TODO : Logging
2023-09-28 10:44:43 -07:00
CloseHandle( file_handle );
2023-09-26 22:16:41 -07:00
return {};
2023-09-26 14:26:35 -07:00
}
2023-09-28 10:44:43 -07:00
return bytes_read;
}
2023-09-26 22:16:41 -07:00
2023-09-28 10:44:43 -07:00
void file_rewind( File* file )
{
HANDLE file_handle = pcast(HANDLE, file->opaque_handle );
2023-09-28 10:44:43 -07:00
if ( file_handle == INVALID_HANDLE_VALUE )
return;
SetFilePointer(file_handle, 0, NULL, FILE_BEGIN);
2023-09-26 14:26:35 -07:00
}
2023-09-28 10:44:43 -07:00
u32 file_write_stream( File* file, u32 content_size, void* content_memory )
2023-09-26 22:16:41 -07:00
{
2023-09-28 10:44:43 -07:00
HANDLE file_handle;
if ( file->opaque_handle == nullptr )
2023-09-28 10:44:43 -07:00
{
file_handle = CreateFileA( file->path
2023-09-28 10:44:43 -07:00
,GENERIC_WRITE, 0, 0
, OPEN_ALWAYS, 0, 0
);
if ( file_handle == INVALID_HANDLE_VALUE )
{
// TODO(Ed) : Logging
return {};
}
file->opaque_handle = file_handle;
2023-09-28 10:44:43 -07:00
}
else
{
file_handle = pcast(HANDLE, file->opaque_handle );
2023-09-28 10:44:43 -07:00
}
DWORD bytes_written;
if ( WriteFile( file_handle, content_memory, content_size, & bytes_written, 0 ) == false )
{
// TODO : Logging
return false;
}
return bytes_written;
}
u32 file_write_content( File* file, u32 content_size, void* content_memory )
{
HANDLE file_handle = CreateFileA( file->path
2023-09-26 22:16:41 -07:00
, GENERIC_WRITE, 0, 0
, CREATE_ALWAYS, 0, 0
);
if ( file_handle == INVALID_HANDLE_VALUE )
{
// TODO : Logging
return false;
}
file->opaque_handle = file_handle;
2023-09-26 22:16:41 -07:00
DWORD bytes_written;
if ( WriteFile( file_handle, content_memory, content_size, & bytes_written, 0 ) == false )
{
// TODO : Logging
return false;
}
2023-09-28 10:44:43 -07:00
return bytes_written;
2023-09-26 22:16:41 -07:00
}
2023-09-28 10:44:43 -07:00
u32 get_monitor_refresh_rate()
{
return 0;
}
void set_monitor_refresh_rate( u32 refresh_rate )
{
}
u32 get_engine_refresh_rate()
{
return 0;
}
void set_engine_refresh_rate( u32 refresh_rate )
{
}
2023-09-26 22:16:41 -07:00
BinaryModule load_binary_module( char const* module_path )
{
HMODULE lib = LoadLibraryA( module_path );
return BinaryModule { scast(void*, lib) };
}
void unload_binary_module( BinaryModule* module )
{
FreeLibrary( scast(HMODULE, module->opaque_handle) );
2023-09-26 22:16:41 -07:00
*module = {};
}
void* get_binary_module_symbol( BinaryModule module, char const* symbol_name )
{
return rcast(void*, GetProcAddress( scast(HMODULE, module.opaque_handle), symbol_name ));
}
void memory_copy( void* dest, u64 src_size, void* src )
{
CopyMemory( dest, src, src_size );
2023-09-26 22:16:41 -07:00
}
#pragma endregion Platform API
2023-09-28 10:44:43 -07:00
#pragma region Engine Module API
2023-09-26 22:16:41 -07:00
constexpr const Str FName_Engine_DLL = str_ascii("handmade_engine.dll");
constexpr const Str FName_Engine_DLL_InUse = str_ascii("handmade_engine_in_use.dll");
constexpr const Str FName_Engine_PDB_Lock = str_ascii("handmade_engine.pdb.lock");
2023-09-26 22:16:41 -07:00
2023-09-28 10:44:43 -07:00
global HMODULE Lib_Handmade_Engine = nullptr;
global StrFixed< S16_MAX > Path_Engine_DLL;
global StrFixed< S16_MAX > Path_Engine_DLL_InUse;
2023-09-26 14:26:35 -07:00
2023-09-28 10:44:43 -07:00
internal
2023-09-26 14:26:35 -07:00
engine::ModuleAPI load_engine_module_api()
{
using ModuleAPI = engine::ModuleAPI;
2023-09-26 22:16:41 -07:00
CopyFileA( Path_Engine_DLL, Path_Engine_DLL_InUse, FALSE );
2023-09-26 14:26:35 -07:00
// Engine
2023-09-26 22:16:41 -07:00
Lib_Handmade_Engine = LoadLibraryA( Path_Engine_DLL_InUse );
2023-09-26 14:26:35 -07:00
if ( ! Lib_Handmade_Engine )
{
return {};
}
engine::ModuleAPI engine_api {};
engine_api.on_module_reload = get_procedure_from_library< engine::OnModuleRelaodFn > ( Lib_Handmade_Engine, engine::symbol_on_module_load );
engine_api.startup = get_procedure_from_library< engine::StartupFn > ( Lib_Handmade_Engine, engine::symbol_startup );
engine_api.shutdown = get_procedure_from_library< engine::ShutdownFn > ( Lib_Handmade_Engine, engine::symbol_shutdown );
engine_api.update_and_render = get_procedure_from_library< engine::UpdateAndRenderFn >( Lib_Handmade_Engine, engine::symbol_update_and_render );
engine_api.update_audio = get_procedure_from_library< engine::UpdateAudioFn > ( Lib_Handmade_Engine, engine::symbol_update_audio );
2023-09-26 14:26:35 -07:00
engine_api.IsValid =
engine_api.on_module_reload
&& engine_api.startup
&& engine_api.shutdown
&& engine_api.update_and_render
&& engine_api.update_audio;
if ( engine_api.IsValid )
{
OutputDebugStringA( "Loaded engine module API\n" );
}
return engine_api;
}
2023-09-28 10:44:43 -07:00
internal
2023-09-26 14:26:35 -07:00
void unload_engine_module_api( engine::ModuleAPI* engine_api )
{
if ( engine_api->IsValid )
{
FreeLibrary( Lib_Handmade_Engine );
*engine_api = {};
OutputDebugStringA( "Unloaded engine module API\n" );
}
}
2023-09-26 22:16:41 -07:00
#pragma endregion Engine Module API
2023-09-26 14:26:35 -07:00
2023-09-20 11:43:55 -07:00
NS_PLATFORM_END
2023-09-08 21:01:53 -07:00
int CALLBACK
2023-09-23 18:03:33 -07:00
WinMain( HINSTANCE instance, HINSTANCE prev_instance, LPSTR commandline, int show_command )
{
2023-09-08 21:01:53 -07:00
using namespace win32;
2023-09-20 11:43:55 -07:00
using namespace platform;
2023-09-28 10:44:43 -07:00
#pragma region Startup
2023-09-22 12:13:18 -07:00
// Timing
#if Build_Development
u64 launch_clock = timing_get_wall_clock();
u64 launch_cycle = __rdtsc();
#endif
// Sets the windows scheduler granulaity for this process to 1 ms
2023-09-23 18:03:33 -07:00
constexpr u32 desired_scheduler_ms = 1;
2023-09-22 12:13:18 -07:00
b32 sleep_is_granular = ( timeBeginPeriod( desired_scheduler_ms ) == TIMERR_NOERROR );
2023-09-23 18:03:33 -07:00
// If its a high-perofmrance frame-time (a refresh rate that produces a target frametime at or below 4.16~ ms, we cannot allow the scheduler to mess things up)
b32 sub_ms_granularity_required = scast(f32, Engine_Refresh_Hz) <= High_Perf_Frametime_MS;
2023-09-22 12:13:18 -07:00
QueryPerformanceFrequency( rcast(LARGE_INTEGER*, & Performance_Counter_Frequency) );
2023-09-30 12:40:27 -07:00
// Setup pathing
StrFixed< S16_MAX > path_pdb_lock {};
{
// TODO(Ed): This will not support long paths, NEEDS to be changed to support long paths.
char path_buffer[S16_MAX];
GetModuleFileNameA( 0, path_buffer, sizeof(path_buffer) );
if ( GetCurrentDirectoryA( S16_MAX, Path_Binaries ) == 0 )
{
fatal( "Failed to get the root directory!" );
}
Path_Binaries.len = str_length( Path_Binaries );
Path_Binaries[ Path_Binaries.len ] = '\\';
++ Path_Binaries.len;
if ( SetCurrentDirectoryA( ".." ) == 0 )
{
fatal( "Failed to set current directory to root!");
}
if ( GetCurrentDirectoryA( S16_MAX, Path_Root.ptr ) == 0 )
{
fatal( "Failed to get the root directory!" );
}
Path_Root.len = str_length(Path_Root.ptr);
Path_Root.ptr[ Path_Root.len ] = '\\';
++ Path_Root.len;
Path_Engine_DLL. concat( Path_Binaries, FName_Engine_DLL );
Path_Engine_DLL_InUse.concat( Path_Binaries, FName_Engine_DLL_InUse );
path_pdb_lock.concat( Path_Binaries, FName_Engine_PDB_Lock );
Path_Scratch.concat( Path_Root, str_ascii("scratch") );
Path_Scratch.ptr[ Path_Scratch.len ] = '\\';
++ Path_Scratch.len;
CreateDirectoryA( Path_Scratch, 0 );
}
2023-09-18 17:16:40 -07:00
// Memory
engine::Memory engine_memory {};
{
engine_memory.persistent_size = megabytes( 128 );
2023-09-29 12:58:18 -07:00
// engine_memory.FrameSize = megabytes( 64 );
engine_memory.transient_size = gigabytes( 2 );
u64 total_size = engine_memory.persistent_size
// + engine_memory.FrameSize
+ engine_memory.transient_size;
2023-09-18 17:16:40 -07:00
#if Build_Debug
2023-09-23 18:03:33 -07:00
void* base_address = rcast(void*, terabytes( 1 ));
2023-09-18 17:16:40 -07:00
#else
2023-09-23 18:03:33 -07:00
void* base_address = 0;
2023-09-18 17:16:40 -07:00
#endif
engine_memory.persistent = VirtualAlloc( base_address, total_size , MEM_Commit_Zeroed | MEM_Reserve, Page_Read_Write );
engine_memory.transient = rcast( u8*, engine_memory.persistent ) + engine_memory.persistent_size;
2023-09-18 17:16:40 -07:00
2023-09-30 12:40:27 -07:00
#if Build_Development
for (u32 slot = 0; slot < engine_memory.Num_Snapshot_Slots; ++slot)
{
engine::MemorySnapshot& snapshot = engine_memory.snapshots[ slot ];
2023-09-30 12:40:27 -07:00
snapshot.file_path.concat( Path_Scratch, str_ascii("snapshot_") );
wsprintfA( snapshot.file_path.ptr, "%s%d.hm_snapshot", snapshot.file_path.ptr, slot );
HANDLE snapshot_file = CreateFileA( snapshot.file_path
, GENERIC_READ | GENERIC_WRITE, 0, 0
, CREATE_ALWAYS, 0, 0 );
LARGE_INTEGER file_size {};
file_size.QuadPart = total_size;
HANDLE snapshot_mapping = CreateFileMappingA( snapshot_file, 0
, Page_Read_Write
, file_size.HighPart, file_size.LowPart
, 0 );
snapshot.memory = MapViewOfFile( snapshot_mapping, FILE_MAP_ALL_ACCESS, 0, 0, total_size );
snapshot.opaque_handle = snapshot_file;
snapshot.opaque_handle_2 = snapshot_mapping;
}
2023-09-30 12:40:27 -07:00
#endif
if ( engine_memory.persistent == nullptr
|| engine_memory.transient == nullptr )
2023-09-18 17:16:40 -07:00
{
// TODO : Diagnostic Logging
return -1;
}
}
2023-09-17 18:20:11 -07:00
WNDCLASSW window_class {};
HWND window_handle = nullptr;
{
2023-09-17 18:20:11 -07:00
window_class.style = CS_Horizontal_Redraw | CS_Vertical_Redraw;
window_class.lpfnWndProc = main_window_callback;
// window_class.cbClsExtra = ;
2023-09-17 18:20:11 -07:00
// window_class.cbWndExtra = ;
window_class.hInstance = instance;
// window_class.hIcon = ;
// window_class.hCursor = ;
// window_class.hbrBackground = ;
window_class.lpszMenuName = L"Handmade Hero!";
window_class.lpszClassName = L"HandmadeHeroWindowClass";
if ( ! RegisterClassW( & window_class ) )
2023-09-13 21:43:35 -07:00
{
2023-09-17 18:20:11 -07:00
// TODO : Diagnostic Logging
return 0;
2023-09-13 21:43:35 -07:00
}
2023-09-17 18:20:11 -07:00
window_handle = CreateWindowExW(
2023-09-30 17:26:00 -07:00
// WS_EX_LAYERED | WS_EX_TOPMOST,
WS_EX_LAYERED,
2023-09-17 18:20:11 -07:00
window_class.lpszClassName,
L"Handmade Hero",
WS_Overlapped_Window | WS_Initially_Visible,
CW_Use_Default, CW_Use_Default, // x, y
2023-09-30 17:26:00 -07:00
1920, 1080, // width, height
2023-09-17 18:20:11 -07:00
0, 0, // parent, menu
instance, 0 // instance, param
);
if ( ! window_handle )
2023-09-09 20:58:28 -07:00
{
2023-09-17 18:20:11 -07:00
// TODO : Diagnostic Logging
return 0;
2023-09-09 20:58:28 -07:00
}
// WinDimensions dimensions = get_window_dimensions( window_handle );
resize_dib_section( &Surface_Back_Buffer, 1920, 1080 );
// Setup monitor refresh and associated timers
HDC refresh_dc = GetDC( window_handle );
u32 monitor_refresh_hz = GetDeviceCaps( refresh_dc, VREFRESH );
if ( monitor_refresh_hz > 1 )
{
Monitor_Refresh_Hz = monitor_refresh_hz;
}
ReleaseDC( window_handle, refresh_dc );
Engine_Refresh_Hz = 60;
Engine_Frame_Target_MS = 1000.f / scast(f32, Engine_Refresh_Hz);
2023-09-09 20:58:28 -07:00
}
2023-09-08 18:08:57 -07:00
2023-09-28 10:44:43 -07:00
// Prepare platform API
ModuleAPI platform_api {};
{
2023-09-29 12:58:18 -07:00
platform_api.path_root = Path_Root;
platform_api.path_binaries = Path_Binaries;
platform_api.path_scratch = Path_Scratch;
2023-09-28 10:44:43 -07:00
#if Build_Development
platform_api.debug_set_pause_rendering = & debug_set_pause_rendering;
#endif
// Not implemented yet
platform_api.get_monitor_refresh_rate = nullptr;
platform_api.set_monitor_refresh_rate = nullptr;
platform_api.get_engine_frame_target = nullptr;
platform_api.set_engine_frame_target = nullptr;
platform_api.load_binary_module = & load_binary_module;
platform_api.unload_binary_module = & unload_binary_module;
platform_api.get_module_procedure = & get_binary_module_symbol;
platform_api.file_check_exists = & file_check_exists;
platform_api.file_close = & file_close;
platform_api.file_delete = & file_delete;
platform_api.file_read_content = & file_read_content;
platform_api.file_read_stream = & file_read_stream;
platform_api.file_rewind = & file_rewind;
platform_api.file_write_content = & file_write_content;
platform_api.file_write_stream = & file_write_stream;
platform_api.memory_copy = & memory_copy;
2023-09-26 22:16:41 -07:00
}
// Load engine module
FILETIME engine_api_load_time = file_get_last_write_time( Path_Engine_DLL );
engine::ModuleAPI engine_api = load_engine_module_api();
2023-09-28 12:41:30 -07:00
b32 sound_is_valid = false;
DWORD ds_cursor_byte_delta = 0;
f32 ds_latency_ms = 0;
2023-09-23 18:03:33 -07:00
DirectSoundBuffer ds_sound_buffer;
2023-09-28 12:41:30 -07:00
u32 audio_marker_index = 0;
AudioTimeMarker audio_time_markers[ Monitor_Refresh_Max_Supported ] {};
u32 audio_time_markers_size = Engine_Refresh_Hz / 2;
assert( audio_time_markers_size <= Monitor_Refresh_Max_Supported )
2023-09-10 11:14:47 -07:00
{
ds_sound_buffer.is_playing = 0;
ds_sound_buffer.samples_per_second = 48000;
ds_sound_buffer.bytes_per_sample = sizeof(s16) * 2;
2023-09-10 11:14:47 -07:00
ds_sound_buffer.secondary_buffer_size = ds_sound_buffer.samples_per_second * ds_sound_buffer.bytes_per_sample;
2023-09-23 18:03:33 -07:00
init_sound( window_handle, & ds_sound_buffer );
2023-09-10 11:14:47 -07:00
ds_sound_buffer.samples = rcast( s16*, VirtualAlloc( 0, 48000 * 2 * sizeof(s16)
2023-09-17 18:20:11 -07:00
, MEM_Commit_Zeroed | MEM_Reserve, Page_Read_Write ));
2023-09-10 11:14:47 -07:00
assert( ds_sound_buffer.samples );
ds_sound_buffer.running_sample_index = 0;
2023-09-17 18:20:11 -07:00
// ds_clear_sound_buffer( & sound_output );
ds_sound_buffer.secondary_buffer->Play( 0, 0, DSBPLAY_LOOPING );
2023-09-10 11:14:47 -07:00
ds_sound_buffer.bytes_per_second = ds_sound_buffer.samples_per_second * ds_sound_buffer.bytes_per_sample;
ds_sound_buffer.guard_sample_bytes = (ds_sound_buffer.bytes_per_second / Engine_Refresh_Hz) / 2;
2023-09-24 19:37:05 -07:00
// TODO(Ed): When switching to core audio at minimum, this will be 1 ms of lag and guard samples wont really be needed.
u32 min_guard_sample_bytes = 1540;
if ( ds_sound_buffer.guard_sample_bytes < min_guard_sample_bytes )
2023-09-24 19:37:05 -07:00
{
ds_sound_buffer.guard_sample_bytes = min_guard_sample_bytes;
2023-09-24 19:37:05 -07:00
}
2023-09-23 18:03:33 -07:00
}
2023-09-17 18:20:11 -07:00
engine::InputState input {};
2023-09-23 18:03:33 -07:00
// There can be 4 of any of each input API type : KB & Mouse, XInput, JSL.
#if 0
using EngineKeyboardStates = engine::KeyboardState[ Max_Controllers ];
EngineKeyboardStates keyboard_states[2] {};
EngineKeyboardStates* old_keyboards = & keyboard_states[0];
EngineKeyboardStates* new_keyboards = & keyboard_states[1];
#endif
2023-09-22 12:13:18 -07:00
engine::KeyboardState keyboard_states[2] {};
2023-09-21 23:16:40 -07:00
engine::KeyboardState* old_keyboard = & keyboard_states[0];
engine::KeyboardState* new_keyboard = & keyboard_states[1];
2023-09-20 21:26:23 -07:00
// Important: Assuming keyboard always connected for now, and assigning to first controller.
engine::MousesState mouse_states[2] {};
engine::MousesState* old_mouse = & mouse_states[0];
engine::MousesState* new_mouse = & mouse_states[1];
2023-09-22 12:13:18 -07:00
EngineXInputPadStates xpad_states[2] {};
2023-09-17 18:20:11 -07:00
EngineXInputPadStates* old_xpads = & xpad_states[0];
EngineXInputPadStates* new_xpads = & xpad_states[1];
2023-09-22 12:13:18 -07:00
EngineDSPadStates ds_pad_states[2] {};
2023-09-17 18:20:11 -07:00
EngineDSPadStates* old_ds_pads = & ds_pad_states[0];
EngineDSPadStates* new_ds_pads = & ds_pad_states[1];
2023-09-22 12:13:18 -07:00
u32 jsl_num_devices = JslConnectDevices();
2023-09-17 18:20:11 -07:00
JSL_DeviceHandle jsl_device_handles[4] {};
{
xinput_load_library_bindings();
u32 jsl_getconnected_found = JslGetConnectedDeviceHandles( jsl_device_handles, jsl_num_devices );
{
if ( jsl_getconnected_found != jsl_num_devices )
{
OutputDebugStringA( "Error: JSLGetConnectedDeviceHandles didn't find as many as were stated with JslConnectDevices\n");
}
if ( jsl_num_devices > 0 )
{
OutputDebugStringA( "JSL Connected Devices:\n" );
for ( u32 jsl_device_index = 0; jsl_device_index < jsl_num_devices; ++ jsl_device_index )
{
JslSetLightColour( jsl_device_handles[ jsl_device_index ], (255 << 8) );
}
}
}
if ( jsl_num_devices > 4 )
{
jsl_num_devices = 4;
2023-09-28 10:44:43 -07:00
MessageBoxA( window_handle, "More than 4 JSL devices found, this engine will only support the first four found."
, "Warning", MB_ICONEXCLAMATION );
2023-09-17 18:20:11 -07:00
}
}
2023-09-26 14:26:35 -07:00
engine_api.startup( & engine_memory, & platform_api );
2023-09-22 12:13:18 -07:00
u64 last_frame_clock = timing_get_wall_clock();
u64 last_frame_cycle = __rdtsc();
2023-09-24 19:37:05 -07:00
u64 flip_wall_clock = last_frame_clock;
2023-09-22 12:13:18 -07:00
#if Build_Development
u64 startup_cycles = last_frame_cycle - launch_cycle;
f32 startup_ms = timing_get_ms_elapsed( launch_clock, last_frame_clock );
2023-09-26 14:26:35 -07:00
char text_buffer[256];
sprintf_s( text_buffer, sizeof(text_buffer), "Startup MS: %f\n", startup_ms );
OutputDebugStringA( text_buffer );
2023-09-22 12:13:18 -07:00
#endif
2023-09-28 10:44:43 -07:00
#pragma endregion Startup
2023-09-22 12:13:18 -07:00
// Placeholder
engine::ThreadContext thread_context_placeholder {};
2023-09-17 18:20:11 -07:00
Running = true;
2023-09-23 18:03:33 -07:00
#if 0
// This tests the play & write cursor update frequency.
while ( Running )
{
DWORD play_cursor;
DWORD write_cursor;
ds_sound_buffer.SecondaryBuffer->GetCurrentPosition( & play_cursor, & write_cursor );
char text_buffer[256];
sprintf_s( text_buffer, sizeof(text_buffer), "PC:%u WC:%u\n", (u32)play_cursor, (u32)write_cursor );
OutputDebugStringA( text_buffer );
}
#endif
2023-09-10 11:14:47 -07:00
while( Running )
2023-09-08 18:08:57 -07:00
{
2023-09-28 10:44:43 -07:00
// Engine Module Hot-Reload
do {
FILETIME engine_api_current_time = file_get_last_write_time( Path_Engine_DLL );
if ( CompareFileTime( & engine_api_load_time, & engine_api_current_time ) == 0 )
break;
2023-09-28 10:44:43 -07:00
WIN32_FIND_DATAA lock_file_info = {};
for(;;)
2023-09-10 11:14:47 -07:00
{
2023-09-28 10:44:43 -07:00
HANDLE lock_file = FindFirstFileA( path_pdb_lock, & lock_file_info );
if ( lock_file != INVALID_HANDLE_VALUE )
{
2023-09-28 10:44:43 -07:00
FindClose( lock_file );
Sleep( 1 );
2023-09-10 11:14:47 -07:00
continue;
}
2023-09-28 10:44:43 -07:00
break;
}
2023-09-28 10:44:43 -07:00
engine_api_load_time = engine_api_current_time;
unload_engine_module_api( & engine_api );
engine_api = load_engine_module_api();
} while (0);
2023-09-22 12:13:18 -07:00
2023-09-17 18:20:11 -07:00
// Swapping at the beginning of the input frame instead of the end.
swap( old_keyboard, new_keyboard );
swap( old_mouse, new_mouse );
swap( old_xpads, new_xpads );
swap( old_ds_pads, new_ds_pads );
poll_input( window_handle, & input, jsl_num_devices, jsl_device_handles
2023-09-28 10:44:43 -07:00
, old_keyboard, new_keyboard
, old_mouse, new_mouse
2023-09-28 10:44:43 -07:00
, old_xpads, new_xpads
, old_ds_pads, new_ds_pads );
2023-09-10 11:14:47 -07:00
process_pending_window_messages( new_keyboard, new_mouse );
2023-09-30 17:26:00 -07:00
f32 delta_time = timing_get_seconds_elapsed( last_frame_clock, timing_get_wall_clock() );
2023-09-24 19:37:05 -07:00
// Engine's logical iteration and rendering process
2023-09-30 17:26:00 -07:00
engine_api.update_and_render( delta_time, & input, rcast(engine::OffscreenBuffer*, & Surface_Back_Buffer.memory )
, & engine_memory, & platform_api, & thread_context_placeholder );
2023-09-24 19:37:05 -07:00
u64 audio_frame_start = timing_get_wall_clock();
f32 flip_to_audio_ms = timing_get_ms_elapsed( flip_wall_clock, audio_frame_start );
DWORD ds_play_cursor;
DWORD ds_write_cursor;
do {
/*
2023-09-28 10:44:43 -07:00
Audio Processing:
2023-09-24 19:37:05 -07:00
There is a sync boundary value, that is the number of samples that the engine's frame-time may vary by
(ex: approx 2ms of variance between frame-times).
On wakeup : Check play cursor position and forcast ahead where the cursor will be for the next sync boundary.
Based on that, check the write cursor position, if its (at least) before the synch boundary, the target write position is
the frame boundary plus one frame. (Low latency)
If its after (sync boundary), we cannot sync audio.
Write a frame's worth of audio plus some number of "guard" samples. (High Latency)
*/
if ( ! SUCCEEDED( ds_sound_buffer.secondary_buffer->GetCurrentPosition( & ds_play_cursor, & ds_write_cursor ) ))
2023-09-24 19:37:05 -07:00
{
sound_is_valid = false;
break;
}
if ( ! sound_is_valid )
{
ds_sound_buffer.running_sample_index = ds_write_cursor / ds_sound_buffer.bytes_per_sample;
2023-09-24 19:37:05 -07:00
sound_is_valid = true;
}
DWORD byte_to_lock = 0;
DWORD target_cursor = 0;
DWORD bytes_to_write = 0;
byte_to_lock = (ds_sound_buffer.running_sample_index * ds_sound_buffer.bytes_per_sample) % ds_sound_buffer.secondary_buffer_size;
2023-09-24 19:37:05 -07:00
DWORD bytes_per_second = ds_sound_buffer.bytes_per_sample * ds_sound_buffer.samples_per_second;
2023-09-24 19:37:05 -07:00
DWORD expected_samplebytes_per_frame = bytes_per_second / Engine_Refresh_Hz;
f32 left_until_flip_ms = Engine_Frame_Target_MS - flip_to_audio_ms;
DWORD expected_bytes_until_flip = scast(DWORD, (left_until_flip_ms / Engine_Frame_Target_MS) * scast(f32, expected_samplebytes_per_frame));
DWORD expected_sync_boundary_byte = ds_play_cursor + expected_bytes_until_flip;
DWORD sync_write_cursor = ds_write_cursor;
if ( sync_write_cursor < ds_play_cursor )
{
// unwrap the cursor so its ahead of the play curosr linearly.
sync_write_cursor += ds_sound_buffer.secondary_buffer_size;
2023-09-24 19:37:05 -07:00
}
assert( sync_write_cursor >= ds_play_cursor );
sync_write_cursor += ds_sound_buffer.guard_sample_bytes;
2023-09-24 19:37:05 -07:00
b32 audio_interface_is_low_latency = sync_write_cursor < expected_sync_boundary_byte;
if ( audio_interface_is_low_latency )
{
target_cursor = ( expected_sync_boundary_byte + expected_samplebytes_per_frame );
}
else
{
target_cursor = (ds_write_cursor + expected_samplebytes_per_frame + ds_sound_buffer.guard_sample_bytes);
2023-09-24 19:37:05 -07:00
}
target_cursor %= ds_sound_buffer.secondary_buffer_size;
2023-09-16 15:41:07 -07:00
2023-09-17 18:20:11 -07:00
if ( byte_to_lock > target_cursor)
2023-09-16 15:41:07 -07:00
{
// Infront of play cursor |--play--byte_to_write-->--|
bytes_to_write = ds_sound_buffer.secondary_buffer_size - byte_to_lock;
2023-09-16 15:41:07 -07:00
bytes_to_write += target_cursor;
}
else
{
// Behind play cursor |--byte_to_write-->--play--|
bytes_to_write = target_cursor - byte_to_lock;
}
2023-09-24 19:37:05 -07:00
// Engine Sound
2023-09-30 17:26:00 -07:00
delta_time = timing_get_seconds_elapsed( last_frame_clock, timing_get_wall_clock() );
2023-09-24 19:37:05 -07:00
// s16 samples[ 48000 * 2 ];
engine::AudioBuffer sound_buffer {};
sound_buffer.num_samples = bytes_to_write / ds_sound_buffer.bytes_per_sample;
sound_buffer.running_sample_index = ds_sound_buffer.running_sample_index;
sound_buffer.samples_per_second = ds_sound_buffer.samples_per_second;
sound_buffer.samples = ds_sound_buffer.samples;
2023-09-30 17:26:00 -07:00
engine_api.update_audio( delta_time, & sound_buffer, & engine_memory, & platform_api, & thread_context_placeholder );
2023-09-24 19:37:05 -07:00
2023-09-28 12:41:30 -07:00
AudioTimeMarker* marker = & audio_time_markers[ audio_marker_index ];
marker->output_play_cursor = ds_play_cursor;
marker->output_write_cursor = ds_write_cursor;
marker->output_location = byte_to_lock;
marker->output_byte_count = bytes_to_write;
marker->expected_flip_cursor = expected_sync_boundary_byte;
2023-09-23 18:03:33 -07:00
// Update audio buffer
if ( ! sound_is_valid )
break;
#if Build_Development && 0
2023-09-24 19:37:05 -07:00
#if 0
2023-09-23 18:03:33 -07:00
DWORD play_cursor;
DWORD write_cursor;
ds_sound_buffer.SecondaryBuffer->GetCurrentPosition( & play_cursor, & write_cursor );
2023-09-24 19:37:05 -07:00
#endif
DWORD unwrapped_write_cursor = ds_write_cursor;
if ( unwrapped_write_cursor < ds_play_cursor )
{
unwrapped_write_cursor += ds_sound_buffer.SecondaryBufferSize;
}
ds_cursor_byte_delta = unwrapped_write_cursor - ds_play_cursor;
constexpr f32 to_milliseconds = 1000.f;
f32 sample_delta = scast(f32, ds_cursor_byte_delta) / scast(f32, ds_sound_buffer.BytesPerSample);
f32 ds_latency_s = sample_delta / scast(f32, ds_sound_buffer.SamplesPerSecond);
ds_latency_ms = ds_latency_s * to_milliseconds;
2023-09-23 18:03:33 -07:00
char text_buffer[256];
2023-09-24 19:37:05 -07:00
sprintf_s( text_buffer, sizeof(text_buffer), "BTL:%u TC:%u BTW:%u - PC:%u WC:%u DELTA:%u bytes %f ms\n"
, (u32)byte_to_lock, (u32)target_cursor, (u32)bytes_to_write
, (u32)play_cursor, (u32)write_cursor, (u32)ds_cursor_byte_delta, ds_latency_ms );
2023-09-23 18:03:33 -07:00
OutputDebugStringA( text_buffer );
#endif
2023-09-24 19:37:05 -07:00
ds_fill_sound_buffer( & ds_sound_buffer, byte_to_lock, bytes_to_write );
2023-09-23 18:03:33 -07:00
2023-09-24 19:37:05 -07:00
DWORD ds_status = 0;
if ( SUCCEEDED( ds_sound_buffer.secondary_buffer->GetStatus( & ds_status ) ) )
2023-09-24 19:37:05 -07:00
{
ds_sound_buffer.is_playing = ds_status & DSBSTATUS_PLAYING;
2023-09-24 19:37:05 -07:00
}
if ( ds_sound_buffer.is_playing )
2023-09-23 18:03:33 -07:00
break;
ds_sound_buffer.secondary_buffer->Play( 0, 0, DSBPLAY_LOOPING );
2023-09-23 18:03:33 -07:00
} while(0);
2023-09-15 18:35:27 -07:00
2023-09-24 19:37:05 -07:00
// Timing Update
2023-09-08 18:08:57 -07:00
{
2023-09-24 19:37:05 -07:00
u64 work_frame_end_cycle = __rdtsc();
u64 work_frame_end_clock = timing_get_wall_clock();
2023-09-22 12:13:18 -07:00
2023-09-24 19:37:05 -07:00
f32 work_frame_ms = timing_get_ms_elapsed( last_frame_clock, work_frame_end_clock ); // WorkSecondsElapsed
f32 work_cycles = timing_get_ms_elapsed( last_frame_cycle, work_frame_end_cycle );
f32 frame_elapsed_ms = work_frame_ms; // SecondsElapsedForFrame
2023-09-23 18:03:33 -07:00
if ( frame_elapsed_ms < Engine_Frame_Target_MS )
{
2023-09-24 19:37:05 -07:00
s32 sleep_ms = scast(DWORD, (Engine_Frame_Target_MS - frame_elapsed_ms)) - 1;
if ( sleep_ms > 0 && ! sub_ms_granularity_required && sleep_is_granular )
{
Sleep( scast(DWORD, sleep_ms) );
}
2023-09-23 18:03:33 -07:00
2023-09-24 19:37:05 -07:00
u64 frame_clock = timing_get_wall_clock();
2023-09-22 12:13:18 -07:00
frame_elapsed_ms = timing_get_ms_elapsed( last_frame_clock, frame_clock );
2023-09-24 19:37:05 -07:00
if ( frame_elapsed_ms < Engine_Frame_Target_MS )
{
// TODO(Ed) : Log missed sleep here.
}
while ( frame_elapsed_ms < Engine_Frame_Target_MS )
{
frame_clock = timing_get_wall_clock();
frame_elapsed_ms = timing_get_ms_elapsed( last_frame_clock, frame_clock );
}
}
else
{
// TODO(Ed) : Missed the display sync window!
2023-09-22 12:13:18 -07:00
}
2023-09-10 11:14:47 -07:00
2023-09-24 19:37:05 -07:00
last_frame_clock = timing_get_wall_clock(); // LastCouner
last_frame_cycle = __rdtsc();
}
2023-09-16 15:41:07 -07:00
2023-09-23 18:03:33 -07:00
// Update surface back buffer
2023-09-24 19:37:05 -07:00
if ( ! Pause_Rendering )
2023-09-22 12:13:18 -07:00
{
WinDimensions dimensions = get_window_dimensions( window_handle );
HDC device_context = GetDC( window_handle );
2023-09-23 18:03:33 -07:00
2023-09-30 12:40:27 -07:00
#if Build_Development && 0
2023-09-24 19:37:05 -07:00
// Note: debug_marker_index is wrong for the 0th index
debug_sync_display( & ds_sound_buffer
2023-09-28 12:41:30 -07:00
, audio_time_markers_size, audio_time_markers, audio_marker_index - 1
2023-09-24 19:37:05 -07:00
, Engine_Frame_Target_MS );
2023-09-23 18:03:33 -07:00
#endif
display_buffer_in_window( device_context, dimensions.width, dimensions.height, &Surface_Back_Buffer
2023-09-22 12:13:18 -07:00
, 0, 0
, dimensions.width, dimensions.height );
2023-09-28 10:44:43 -07:00
ReleaseDC( window_handle, device_context );
2023-09-22 12:13:18 -07:00
}
2023-09-13 21:43:35 -07:00
2023-09-24 19:37:05 -07:00
flip_wall_clock = timing_get_wall_clock();
#if Build_Development
2023-09-23 18:03:33 -07:00
{
2023-09-24 19:37:05 -07:00
// Audio Debug
2023-09-23 18:03:33 -07:00
DWORD play_cursor = 0;
DWORD write_cursor = 0;
if ( SUCCEEDED( ds_sound_buffer.secondary_buffer->GetCurrentPosition( & play_cursor, & write_cursor ) ) )
2023-09-23 18:03:33 -07:00
{
if ( ! sound_is_valid )
{
ds_sound_buffer.running_sample_index = write_cursor / ds_sound_buffer.bytes_per_sample;
2023-09-23 18:03:33 -07:00
sound_is_valid = true;
}
2023-09-24 19:37:05 -07:00
2023-09-28 12:41:30 -07:00
assert( audio_marker_index < audio_time_markers_size )
AudioTimeMarker* marker = & audio_time_markers[ audio_marker_index ];
2023-09-24 19:37:05 -07:00
marker->flip_play_curosr = play_cursor;
marker->flip_write_cursor = write_cursor;
2023-09-23 18:03:33 -07:00
}
2023-09-24 19:37:05 -07:00
}
#endif
2023-09-23 18:03:33 -07:00
#if Build_Development
2023-09-28 12:41:30 -07:00
audio_marker_index++;
if ( audio_marker_index >= audio_time_markers_size )
audio_marker_index = 0;
2023-09-23 18:03:33 -07:00
#endif
2023-09-08 18:08:57 -07:00
}
2023-09-08 15:54:16 -07:00
2023-09-26 14:26:35 -07:00
engine_api.shutdown( & engine_memory, & platform_api );
2023-09-23 18:03:33 -07:00
2023-09-26 22:16:41 -07:00
unload_engine_module_api( & engine_api );
DeleteFileA( Path_Engine_DLL_InUse );
2023-09-09 20:58:28 -07:00
if ( jsl_num_devices > 0 )
{
for ( u32 jsl_device_index = 0; jsl_device_index < jsl_num_devices; ++ jsl_device_index )
{
2023-09-17 18:20:11 -07:00
JslSetLightColour( jsl_device_handles[ jsl_device_index ], 0 );
2023-09-09 20:58:28 -07:00
}
}
return 0;
}