332 lines
13 KiB
C++
332 lines
13 KiB
C++
|
//
|
||
|
// File: blumen_lumen.cpp
|
||
|
// Author: Peter Slattery
|
||
|
// Creation Date: 2021-01-23
|
||
|
//
|
||
|
#ifndef BLUMEN_LUMEN_CPP
|
||
|
|
||
|
|
||
|
internal void
|
||
|
BlumenLumen_MicListenJob(gs_thread_context* Ctx, u8* UserData)
|
||
|
{
|
||
|
mic_listen_job_data* Data = (mic_listen_job_data*)UserData;
|
||
|
|
||
|
gs_data Msg = {};
|
||
|
|
||
|
u8 WeathermanIPAddr[4] = {};
|
||
|
WeathermanIPAddr[0] = 127;
|
||
|
WeathermanIPAddr[1] = 0;
|
||
|
WeathermanIPAddr[2] = 0;
|
||
|
WeathermanIPAddr[3] = 1;
|
||
|
|
||
|
u32 WeathermanIPV4 = (u32)UpackB4(WeathermanIPAddr);
|
||
|
u32 WeathermanPort = 20185;
|
||
|
|
||
|
while (*Data->Running)
|
||
|
{
|
||
|
if (SocketQueryStatus(Data->SocketManager, Data->ListenSocket))
|
||
|
{
|
||
|
if (SocketPeek(Data->SocketManager, Data->ListenSocket))
|
||
|
{
|
||
|
// TODO(pjs): Make this a peek operation
|
||
|
Msg = SocketRecieve(Data->SocketManager, Data->ListenSocket, Ctx->Transient);
|
||
|
if (Msg.Size > 0)
|
||
|
{
|
||
|
Data->MicPacketBuffer->Values[Data->MicPacketBuffer->WriteHead++] = Msg;
|
||
|
if (Data->MicPacketBuffer->WriteHead >= PACKETS_MAX)
|
||
|
{
|
||
|
Data->MicPacketBuffer->WriteHead = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
while (Data->OutgoingMsgQueue->ReadHead != Data->OutgoingMsgQueue->WriteHead)
|
||
|
{
|
||
|
u32 ReadIndex = Data->OutgoingMsgQueue->ReadHead++;
|
||
|
if (Data->OutgoingMsgQueue->ReadHead >= BLUMEN_MESSAGE_QUEUE_COUNT)
|
||
|
{
|
||
|
Data->OutgoingMsgQueue->ReadHead = 0;
|
||
|
}
|
||
|
|
||
|
Msg = Data->OutgoingMsgQueue->Buffers[ReadIndex];
|
||
|
u32 Address = WeathermanIPV4;
|
||
|
u32 Port = WeathermanPort;
|
||
|
s32 Flags = 0;
|
||
|
SocketSend(Data->SocketManager, Data->ListenSocket, Address, Port, Msg, Flags);
|
||
|
|
||
|
OutputDebugString("Sending Motor Packet\n");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
CloseSocket(Data->SocketManager, Data->ListenSocket);
|
||
|
}
|
||
|
|
||
|
internal void
|
||
|
BlumenLumen_LoadPatterns(app_state* State)
|
||
|
{
|
||
|
animation_pattern_array* Patterns = &State->Patterns;
|
||
|
if (Patterns->CountMax == 0)
|
||
|
{
|
||
|
*Patterns = Patterns_Create(&State->Permanent, 32);
|
||
|
}
|
||
|
|
||
|
Patterns->Count = 0;
|
||
|
Patterns_PushPattern(Patterns, TestPatternOne);
|
||
|
Patterns_PushPattern(Patterns, TestPatternTwo);
|
||
|
Patterns_PushPattern(Patterns, TestPatternThree);
|
||
|
Patterns_PushPattern(Patterns, Pattern_AllGreen);
|
||
|
Patterns_PushPattern(Patterns, Pattern_HueShift);
|
||
|
Patterns_PushPattern(Patterns, Pattern_HueFade);
|
||
|
Patterns_PushPattern(Patterns, Pattern_Spots);
|
||
|
Patterns_PushPattern(Patterns, Pattern_LighthouseRainbow);
|
||
|
Patterns_PushPattern(Patterns, Pattern_SmoothGrowRainbow);
|
||
|
Patterns_PushPattern(Patterns, Pattern_GrowAndFade);
|
||
|
Patterns_PushPattern(Patterns, Pattern_ColorToWhite);
|
||
|
Patterns_PushPattern(Patterns, Pattern_Blue);
|
||
|
Patterns_PushPattern(Patterns, Pattern_Green);
|
||
|
Patterns_PushPattern(Patterns, Pattern_FlowerColors);
|
||
|
Patterns_PushPattern(Patterns, Pattern_FlowerColorToWhite);
|
||
|
Patterns_PushPattern(Patterns, Pattern_BasicFlowers);
|
||
|
}
|
||
|
|
||
|
internal pixel
|
||
|
TEMP_Saturate(pixel P)
|
||
|
{
|
||
|
v4 CRGB = v4{ (r32)P.R / 255.f, (r32)P.G / 255.f, (r32)P.B / 255.f, 1.f };
|
||
|
v4 CHSV = RGBToHSV(CRGB);
|
||
|
if (CHSV.g > .3f)
|
||
|
{
|
||
|
CHSV.g = 1;
|
||
|
CRGB = HSVToRGB(CHSV);
|
||
|
}
|
||
|
return V4ToRGBPixel(CRGB);
|
||
|
}
|
||
|
|
||
|
internal gs_data
|
||
|
BlumenLumen_CustomInit(app_state* State, context Context)
|
||
|
{
|
||
|
// This is memory for any custom data that we want to use
|
||
|
// as a part of a particular sculpture.
|
||
|
// By returning it from here, it will be sent as an argument to
|
||
|
// the sculpture's CustomUpdate function;
|
||
|
gs_data Result = {};
|
||
|
|
||
|
Result = PushSizeToData(&State->Permanent, sizeof(blumen_lumen_state));
|
||
|
|
||
|
blumen_lumen_state* BLState = (blumen_lumen_state*)Result.Memory;
|
||
|
BLState->Running = true;
|
||
|
|
||
|
BLState->MicListenJobData.Running = &BLState->Running;
|
||
|
BLState->MicListenJobData.SocketManager = Context.SocketManager;
|
||
|
BLState->MicListenJobData.MicPacketBuffer = &BLState->MicPacketBuffer;
|
||
|
BLState->MicListenJobData.OutgoingMsgQueue = &BLState->OutgoingMsgQueue;
|
||
|
BLState->MicListenJobData.ListenSocket = CreateSocket(Context.SocketManager, "127.0.0.1", "20185");
|
||
|
|
||
|
#if 0
|
||
|
BLState->MicListenThread = CreateThread(Context.ThreadManager, BlumenLumen_MicListenJob, (u8*)&BLState->MicListenJobData);
|
||
|
#endif
|
||
|
|
||
|
gs_const_string SculpturePath = ConstString("data/test_blumen.fold");
|
||
|
LoadAssembly(&State->Assemblies, &State->LedSystem, State->Transient, Context, SculpturePath, State->GlobalLog);
|
||
|
|
||
|
{ // Animation PLAYGROUND
|
||
|
animation Anim0 = {0};
|
||
|
Anim0.Name = PushStringF(&State->Permanent, 256, "test_anim_zero");
|
||
|
Anim0.Layers = AnimLayerArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim0.Blocks_ = AnimBlockArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim0.PlayableRange.Min = 0;
|
||
|
Anim0.PlayableRange.Max = SecondsToFrames(15, State->AnimationSystem);
|
||
|
Animation_AddLayer(&Anim0, MakeString("Base Layer"), BlendMode_Overwrite, &State->AnimationSystem);
|
||
|
|
||
|
Animation_AddBlock(&Anim0, 0, Anim0.PlayableRange.Max, Patterns_IndexToHandle(15), 0);
|
||
|
|
||
|
BLState->AnimHandles[0] = AnimationArray_Push(&State->AnimationSystem.Animations, Anim0);
|
||
|
|
||
|
animation Anim1 = {0};
|
||
|
Anim1.Name = PushStringF(&State->Permanent, 256, "test_anim_one");
|
||
|
Anim1.Layers = AnimLayerArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim1.Blocks_ = AnimBlockArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim1.PlayableRange.Min = 0;
|
||
|
Anim1.PlayableRange.Max = SecondsToFrames(15, State->AnimationSystem);
|
||
|
Animation_AddLayer(&Anim1, MakeString("Base Layer"), BlendMode_Overwrite, &State->AnimationSystem);
|
||
|
|
||
|
Animation_AddBlock(&Anim1, 0, Anim0.PlayableRange.Max, Patterns_IndexToHandle(12), 0);
|
||
|
|
||
|
BLState->AnimHandles[1] = AnimationArray_Push(&State->AnimationSystem.Animations, Anim1);
|
||
|
|
||
|
animation Anim2 = {0};
|
||
|
Anim2.Name = PushStringF(&State->Permanent, 256, "i_love_you");
|
||
|
Anim2.Layers = AnimLayerArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim2.Blocks_ = AnimBlockArray_Create(State->AnimationSystem.Storage, 8);
|
||
|
Anim2.PlayableRange.Min = 0;
|
||
|
Anim2.PlayableRange.Max = SecondsToFrames(15, State->AnimationSystem);
|
||
|
Animation_AddLayer(&Anim2, MakeString("Base Layer"), BlendMode_Overwrite, &State->AnimationSystem);
|
||
|
|
||
|
Animation_AddBlock(&Anim2, 0, 100, Patterns_IndexToHandle(5), 0);
|
||
|
Animation_AddBlock(&Anim2, 50, Anim0.PlayableRange.Max, Patterns_IndexToHandle(10), 0);
|
||
|
|
||
|
BLState->AnimHandles[2] = AnimationArray_Push(&State->AnimationSystem.Animations, Anim2);
|
||
|
|
||
|
State->AnimationSystem.ActiveFadeGroup.From = BLState->AnimHandles[2];
|
||
|
State->AnimationSystem.TimelineShouldAdvance = true;
|
||
|
} // End Animation Playground
|
||
|
|
||
|
for (u32 i = 0; i < FLOWER_COLORS_COUNT; i++)
|
||
|
{
|
||
|
FlowerAColors[i] = TEMP_Saturate(FlowerAColors[i]);
|
||
|
FlowerBColors[i] = TEMP_Saturate(FlowerBColors[i]);
|
||
|
FlowerCColors[i] = TEMP_Saturate(FlowerCColors[i]);
|
||
|
}
|
||
|
|
||
|
return Result;
|
||
|
}
|
||
|
|
||
|
internal void
|
||
|
BlumenLumen_CustomUpdate(gs_data UserData, app_state* State, context* Context)
|
||
|
{
|
||
|
blumen_lumen_state* BLState = (blumen_lumen_state*)UserData.Memory;
|
||
|
|
||
|
MotorTimeElapsed += Context->DeltaTime;
|
||
|
#if 0
|
||
|
BLState->TimeElapsed += Context->DeltaTime;
|
||
|
|
||
|
if (BLState->TimeElapsed > 5)
|
||
|
{
|
||
|
u32 NextIndex = ++BLState->CurrAnim % 3;
|
||
|
animation_handle Next = BLState->AnimHandles[NextIndex];
|
||
|
AnimationFadeGroup_FadeTo(&State->AnimationSystem.ActiveFadeGroup, Next, 5);
|
||
|
BLState->TimeElapsed = 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
gs_string BlueString = MakeString("blue");
|
||
|
gs_string GreenString = MakeString("green");
|
||
|
gs_string ILoveYouString = MakeString("i_love_you");
|
||
|
|
||
|
while (BLState->MicPacketBuffer.ReadHead != BLState->MicPacketBuffer.WriteHead)
|
||
|
{
|
||
|
gs_data PacketData = BLState->MicPacketBuffer.Values[BLState->MicPacketBuffer.ReadHead++];
|
||
|
|
||
|
u8 PacketType = PacketData.Memory[0];
|
||
|
switch (PacketType) {
|
||
|
case PacketType_PatternCommand:
|
||
|
{
|
||
|
microphone_packet Packet = *(microphone_packet*)(PacketData.Memory + 1);
|
||
|
|
||
|
u32 NameLen = CStringLength(Packet.AnimationFileName);
|
||
|
if (StringEqualsCharArray(BlueString.ConstString, Packet.AnimationFileName, NameLen))
|
||
|
{
|
||
|
State->AnimationSystem.ActiveFadeGroup.From.Index = 0;
|
||
|
}
|
||
|
else if (StringEqualsCharArray(GreenString.ConstString, Packet.AnimationFileName, NameLen))
|
||
|
{
|
||
|
State->AnimationSystem.ActiveFadeGroup.From.Index = 1;
|
||
|
}
|
||
|
else if (StringEqualsCharArray(ILoveYouString.ConstString, Packet.AnimationFileName, NameLen))
|
||
|
{
|
||
|
State->AnimationSystem.ActiveFadeGroup.From.Index = 2;
|
||
|
}
|
||
|
|
||
|
OutputDebugStringA("Received Pattern Packet\n");
|
||
|
}break;
|
||
|
|
||
|
case PacketType_MotorState:
|
||
|
{
|
||
|
motor_packet Packet = *(motor_packet*)(PacketData.Memory + 1);
|
||
|
BLState->LastKnownMotorState = Packet;
|
||
|
|
||
|
gs_string Temp = PushStringF(State->Transient, 256, "Received Motor States: %d %d %d\n",
|
||
|
Packet.FlowerPositions[0],
|
||
|
Packet.FlowerPositions[1],
|
||
|
Packet.FlowerPositions[2]);
|
||
|
NullTerminate(&Temp);
|
||
|
|
||
|
OutputDebugStringA(Temp.Str);
|
||
|
}break;
|
||
|
|
||
|
case PacketType_Temperature:
|
||
|
{
|
||
|
temp_packet Packet = *(temp_packet*)(PacketData.Memory + 1);
|
||
|
|
||
|
gs_string Temp = PushStringF(State->Transient, 256, "Temperature: %d\n",
|
||
|
Packet.Temperature);
|
||
|
NullTerminate(&Temp);
|
||
|
|
||
|
OutputDebugStringA(Temp.Str);
|
||
|
}break;
|
||
|
|
||
|
InvalidDefaultCase;
|
||
|
}
|
||
|
|
||
|
|
||
|
if (BLState->MicPacketBuffer.ReadHead >= PACKETS_MAX)
|
||
|
{
|
||
|
BLState->MicPacketBuffer.ReadHead = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (false && MotorTimeElapsed > 0)
|
||
|
{
|
||
|
// NOTE(pjs):
|
||
|
MotorTimeElapsed = 0;
|
||
|
u8 Position = LastPosition;
|
||
|
if (LastPosition == 2)
|
||
|
{
|
||
|
LastPosition = 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
LastPosition = 2;
|
||
|
}
|
||
|
|
||
|
if ((BLState->OutgoingMsgQueue.WriteHead >= BLState->OutgoingMsgQueue.ReadHead) ||
|
||
|
(BLState->OutgoingMsgQueue.WriteHead < BLState->OutgoingMsgQueue.ReadHead))
|
||
|
{
|
||
|
u32 WriteIndex = BLState->OutgoingMsgQueue.WriteHead;
|
||
|
|
||
|
gs_data* Msg = BLState->OutgoingMsgQueue.Buffers + WriteIndex;
|
||
|
if (Msg->Size == 0)
|
||
|
{
|
||
|
*Msg = PushSizeToData(&State->Permanent, sizeof(motor_packet));
|
||
|
}
|
||
|
motor_packet* Packet = (motor_packet*)Msg->Memory;
|
||
|
Packet->FlowerPositions[0] = Position;
|
||
|
Packet->FlowerPositions[1] = Position;
|
||
|
Packet->FlowerPositions[2] = Position;
|
||
|
|
||
|
// NOTE(pjs): We increment the write head AFTER we've written so that
|
||
|
// the network thread doesn't think the buffer is ready to send before
|
||
|
// the data is set. We want to avoid the case of:
|
||
|
// 1. Main Thread increments write head to 1
|
||
|
// 2. Network Thread thinks theres a new message to send at 0
|
||
|
// 3. Network Thread sends the message at 0
|
||
|
// 4. Main Thread sets the message at 0
|
||
|
BLState->OutgoingMsgQueue.WriteHead += 1;
|
||
|
if (BLState->OutgoingMsgQueue.WriteHead >= BLUMEN_MESSAGE_QUEUE_COUNT)
|
||
|
{
|
||
|
BLState->OutgoingMsgQueue.WriteHead = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
US_CUSTOM_CLEANUP(BlumenLumen_CustomCleanup)
|
||
|
{
|
||
|
blumen_lumen_state* BLState = (blumen_lumen_state*)UserData.Memory;
|
||
|
BLState->Running = false;
|
||
|
}
|
||
|
|
||
|
internal user_space_desc
|
||
|
BlumenLumen_UserSpaceCreate()
|
||
|
{
|
||
|
user_space_desc Result = {};
|
||
|
Result.LoadPatterns = BlumenLumen_LoadPatterns;
|
||
|
Result.CustomInit = BlumenLumen_CustomInit;
|
||
|
Result.CustomUpdate = BlumenLumen_CustomUpdate;
|
||
|
Result.CustomCleanup = BlumenLumen_CustomCleanup;
|
||
|
return Result;
|
||
|
}
|
||
|
|
||
|
#define BLUMEN_LUMEN_CPP
|
||
|
#endif // BLUMEN_LUMEN_CPP
|