2023-09-15 18:35:27 -07:00
|
|
|
#include "engine.h"
|
2023-09-17 18:20:11 -07:00
|
|
|
#include "win32.h"
|
2023-09-15 18:35:27 -07:00
|
|
|
|
|
|
|
NS_ENGINE_BEGIN
|
|
|
|
|
2023-09-18 17:16:40 -07:00
|
|
|
struct EngineState
|
|
|
|
{
|
|
|
|
s32 WavePeriod;
|
|
|
|
s32 WaveToneHz;
|
|
|
|
s32 ToneVolume;
|
|
|
|
s32 XOffset;
|
|
|
|
s32 YOffset;
|
|
|
|
};
|
2023-09-16 15:41:07 -07:00
|
|
|
|
2023-09-18 17:16:40 -07:00
|
|
|
using GetSoundSampleValueFn = s16( EngineState* state, SoundBuffer* sound_buffer );
|
2023-09-16 15:41:07 -07:00
|
|
|
|
|
|
|
internal s16
|
2023-09-18 17:16:40 -07:00
|
|
|
square_wave_sample_value( EngineState* state, SoundBuffer* sound_buffer )
|
2023-09-16 15:41:07 -07:00
|
|
|
{
|
2023-09-20 21:26:23 -07:00
|
|
|
s32 sample_value = (sound_buffer->RunningSampleIndex / (state->WavePeriod / 2) ) % 2 ?
|
2023-09-18 17:16:40 -07:00
|
|
|
state->ToneVolume : - state->ToneVolume;
|
2023-09-16 15:41:07 -07:00
|
|
|
|
2023-09-20 21:26:23 -07:00
|
|
|
return scast(s16, sample_value);
|
2023-09-16 15:41:07 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
internal s16
|
2023-09-18 17:16:40 -07:00
|
|
|
sine_wave_sample_value( EngineState* state, SoundBuffer* sound_buffer )
|
2023-09-16 15:41:07 -07:00
|
|
|
{
|
|
|
|
local_persist f32 time = 0.f;
|
|
|
|
|
2023-09-17 18:20:11 -07:00
|
|
|
// time = TAU * (f32)sound_buffer->RunningSampleIndex / (f32)SoundTest_WavePeriod;
|
2023-09-16 15:41:07 -07:00
|
|
|
f32 sine_value = sinf( time );
|
2023-09-20 21:26:23 -07:00
|
|
|
s16 sample_value = scast(s16, sine_value * scast(f32, state->ToneVolume));
|
2023-09-16 15:41:07 -07:00
|
|
|
|
2023-09-18 17:16:40 -07:00
|
|
|
time += TAU * 1.0f / scast(f32, state->WavePeriod );
|
2023-09-16 15:41:07 -07:00
|
|
|
return sample_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
2023-09-18 17:16:40 -07:00
|
|
|
output_sound( EngineState* state, SoundBuffer* sound_buffer, GetSoundSampleValueFn* get_sample_value )
|
2023-09-16 15:41:07 -07:00
|
|
|
{
|
|
|
|
s16* sample_out = sound_buffer->Samples;
|
2023-09-20 21:26:23 -07:00
|
|
|
for ( s32 sample_index = 0; sample_index < sound_buffer->NumSamples; ++ sample_index )
|
2023-09-16 15:41:07 -07:00
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
s16 sample_value = get_sample_value( state, sound_buffer );
|
2023-09-17 18:20:11 -07:00
|
|
|
sound_buffer->RunningSampleIndex++;
|
|
|
|
|
|
|
|
// char ms_timing_debug[256] {};
|
|
|
|
// wsprintfA( ms_timing_debug, "sample_value: %d\n", sample_value );
|
|
|
|
// OutputDebugStringA( ms_timing_debug );
|
2023-09-16 15:41:07 -07:00
|
|
|
|
|
|
|
*sample_out = sample_value;
|
|
|
|
++ sample_out;
|
|
|
|
|
|
|
|
*sample_out = sample_value;
|
|
|
|
++ sample_out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-15 18:35:27 -07:00
|
|
|
internal void
|
|
|
|
render_weird_graident(OffscreenBuffer* buffer, u32 x_offset, u32 y_offset )
|
|
|
|
{
|
|
|
|
// TODO(Ed): See if with optimizer if buffer should be passed by value.
|
|
|
|
|
|
|
|
struct Pixel {
|
|
|
|
u8 Blue;
|
|
|
|
u8 Green;
|
|
|
|
u8 Red;
|
|
|
|
u8 Alpha;
|
|
|
|
};
|
|
|
|
|
|
|
|
u8* row = rcast( u8*, buffer->Memory);
|
|
|
|
local_persist float wildcard = 0;
|
|
|
|
for ( u32 y = 0; y < buffer->Height; ++ y )
|
|
|
|
{
|
|
|
|
// u8* pixel = rcast(u8*, row);
|
|
|
|
// Pixel* pixel = rcast( Pixel*, row );
|
|
|
|
u32* pixel = rcast(u32*, row);
|
|
|
|
for ( u32 x = 0; x < buffer->Width; ++ x )
|
|
|
|
{
|
|
|
|
/* Pixel in memory:
|
|
|
|
-----------------------------------------------
|
|
|
|
Pixel + 0 Pixel + 1 Pixel + 2 Pixel + 3
|
|
|
|
RR GG GG XX
|
|
|
|
-----------------------------------------------
|
|
|
|
x86-64 : Little Endian Arch
|
|
|
|
0x XX BB GG RR
|
|
|
|
*/
|
|
|
|
#if 0
|
|
|
|
u8 blue = scast(u8, x + x_offset * u8(wildcard) % 256);
|
|
|
|
u8 green = scast(u8, y + y_offset - u8(wildcard) % 128);
|
|
|
|
u8 red = scast(u8, wildcard) % 256 - x * 0.4f;
|
|
|
|
#else
|
|
|
|
u8 blue = scast(u8, x + x_offset);
|
|
|
|
u8 green = scast(u8, y + y_offset);
|
|
|
|
u8 red = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2023-09-20 21:26:23 -07:00
|
|
|
*pixel++ = u32(red << 16) | u32(green << 8) | blue;
|
2023-09-15 18:35:27 -07:00
|
|
|
}
|
|
|
|
wildcard += 0.5375f;
|
|
|
|
row += buffer->Pitch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-17 18:20:11 -07:00
|
|
|
b32 input_using_analog()
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-18 17:16:40 -07:00
|
|
|
internal void
|
|
|
|
startup()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
|
|
|
shutdown()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-09-17 18:20:11 -07:00
|
|
|
// TODO : I rather expose the back_buffer and sound_buffer using getters for access in any function.
|
2023-09-20 21:26:23 -07:00
|
|
|
void update_and_render( InputState* input, OffscreenBuffer* back_buffer, SoundBuffer* sound_buffer, Memory* memory )
|
2023-09-15 18:35:27 -07:00
|
|
|
{
|
2023-09-17 18:20:11 -07:00
|
|
|
// Graphics & Input Test
|
|
|
|
local_persist u32 x_offset = 0;
|
|
|
|
local_persist u32 y_offset = 0;
|
|
|
|
|
|
|
|
// Wave Sound Test
|
|
|
|
local_persist bool wave_switch = false;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
if ( input_using_analog() )
|
|
|
|
{
|
|
|
|
// TODO(Ed) : Use analog movement tuning
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// TODO(Ed) : Use digital movement tuning
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-09-18 17:16:40 -07:00
|
|
|
EngineState* state = rcast( EngineState*, memory->Persistent );
|
2023-09-17 18:20:11 -07:00
|
|
|
do_once_start
|
2023-09-18 17:16:40 -07:00
|
|
|
assert( sizeof(EngineState) <= memory->PersistentSize );
|
|
|
|
|
|
|
|
state->ToneVolume = 3000;
|
|
|
|
state->WaveToneHz = 262;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
|
|
|
|
|
|
|
state->XOffset = 0;
|
|
|
|
state->YOffset = 0;
|
|
|
|
|
2023-09-20 11:43:55 -07:00
|
|
|
#if Build_Debug
|
|
|
|
{
|
|
|
|
using namespace platform;
|
|
|
|
|
|
|
|
char* file_path = __FILE__;
|
|
|
|
Debug_FileContent file_content = debug_file_read_content( file_path );
|
|
|
|
if ( file_content.Size )
|
|
|
|
{
|
|
|
|
debug_file_write_content( "test.out", file_content.Size, file_content.Data );
|
|
|
|
debug_file_free_content( & file_content );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2023-09-17 18:20:11 -07:00
|
|
|
do_once_end
|
|
|
|
|
|
|
|
ControllerState* controller = & input->Controllers[0];
|
|
|
|
|
|
|
|
if ( controller->DSPad )
|
|
|
|
{
|
|
|
|
DualsensePadState* pad = controller->DSPad;
|
|
|
|
|
|
|
|
x_offset += pad->DPad.Right.State;
|
|
|
|
x_offset -= pad->DPad.Left.State;
|
|
|
|
y_offset += pad->DPad.Down.State;
|
|
|
|
y_offset -= pad->DPad.Up.State;
|
|
|
|
|
2023-09-20 21:26:23 -07:00
|
|
|
x_offset += scast(u32, pad->Stick.Left.X.End);
|
|
|
|
y_offset += scast(u32, pad->Stick.Left.Y.End);
|
2023-09-17 18:20:11 -07:00
|
|
|
|
|
|
|
if ( pad->Triangle.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->ToneVolume += 10;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
if ( pad->Circle.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->ToneVolume -= 10;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->Square.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->WaveToneHz += 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
if ( pad->X.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->WaveToneHz -= 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->Options.State )
|
|
|
|
{
|
|
|
|
wave_switch ^= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->Share.State )
|
|
|
|
{
|
|
|
|
// TODO(Ed) : Add rumble test
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 21:26:23 -07:00
|
|
|
if ( controller->XPad )
|
2023-09-17 18:20:11 -07:00
|
|
|
{
|
|
|
|
XInputPadState* pad = controller->XPad;
|
|
|
|
|
|
|
|
x_offset += pad->DPad.Right.State;
|
|
|
|
x_offset -= pad->DPad.Left.State;
|
2023-09-20 21:26:23 -07:00
|
|
|
y_offset -= pad->DPad.Down.State;
|
|
|
|
y_offset += pad->DPad.Up.State;
|
2023-09-17 18:20:11 -07:00
|
|
|
|
2023-09-20 21:26:23 -07:00
|
|
|
x_offset += scast(u32, pad->Stick.Left.X.End);
|
|
|
|
y_offset += scast(u32, pad->Stick.Left.Y.End);
|
2023-09-17 18:20:11 -07:00
|
|
|
|
|
|
|
if ( pad->Y.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->ToneVolume += 10;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
if ( pad->B.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->ToneVolume -= 10;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->X.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->WaveToneHz += 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
if ( pad->A.State )
|
|
|
|
{
|
2023-09-18 17:16:40 -07:00
|
|
|
state->WaveToneHz -= 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
2023-09-17 18:20:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->Start.State )
|
|
|
|
{
|
|
|
|
wave_switch ^= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( pad->Back.State )
|
|
|
|
{
|
|
|
|
// TODO(Ed) : Add rumble test
|
|
|
|
}
|
|
|
|
}
|
2023-09-20 21:26:23 -07:00
|
|
|
if ( controller->Keyboard )
|
|
|
|
{
|
|
|
|
KeyboardState* keyboard = controller->Keyboard;
|
|
|
|
|
|
|
|
x_offset += keyboard->D.State;
|
|
|
|
x_offset -= keyboard->A.State;
|
|
|
|
y_offset += keyboard->W.State;
|
|
|
|
y_offset -= keyboard->S.State;
|
|
|
|
|
|
|
|
if ( keyboard->Esc.State )
|
|
|
|
{
|
|
|
|
// TODO : Add exit game
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( keyboard->Space.State )
|
|
|
|
{
|
|
|
|
wave_switch ^= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( keyboard->Up.State )
|
|
|
|
{
|
|
|
|
state->ToneVolume += 10;
|
|
|
|
}
|
|
|
|
if ( keyboard->Down.State )
|
|
|
|
{
|
|
|
|
state->ToneVolume -= 10;
|
|
|
|
}
|
|
|
|
if ( keyboard->Left.State )
|
|
|
|
{
|
|
|
|
state->WaveToneHz -= 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
|
|
|
}
|
|
|
|
if ( keyboard->Right.State )
|
|
|
|
{
|
|
|
|
state->WaveToneHz += 1;
|
|
|
|
state->WavePeriod = sound_buffer->SamplesPerSecond / state->WaveToneHz;
|
|
|
|
}
|
|
|
|
}
|
2023-09-17 18:20:11 -07:00
|
|
|
|
2023-09-16 15:41:07 -07:00
|
|
|
// TODO(Ed) : Allow sample offsets here for more robust platform options
|
2023-09-17 18:20:11 -07:00
|
|
|
if ( ! wave_switch )
|
2023-09-18 17:16:40 -07:00
|
|
|
output_sound( state, sound_buffer, sine_wave_sample_value );
|
2023-09-17 18:20:11 -07:00
|
|
|
else
|
2023-09-18 17:16:40 -07:00
|
|
|
output_sound( state, sound_buffer, square_wave_sample_value );
|
2023-09-16 15:41:07 -07:00
|
|
|
|
2023-09-15 18:35:27 -07:00
|
|
|
render_weird_graident( back_buffer, x_offset, y_offset );
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ENGINE_END
|