// // File: win32_foldhaus.cpp // Author: Peter Slattery // Creation Date: 2020-01-01 // #ifndef WIN32_FOLDHAUS_CPP #include #include #include #include #include #include "../../meta/gs_meta_include.h" #include "../foldhaus_platform.h" #include "../../gs_libs/gs_win32.cpp" #include "win32_foldhaus_utils.h" #include "win32_foldhaus_memory.h" #include "win32_foldhaus_fileio.h" #include "win32_foldhaus_dll.h" #include "win32_foldhaus_timing.h" #include "win32_foldhaus_work_queue.h" #include "win32_foldhaus_serial.h" #include "win32_foldhaus_socket.h" #include "../foldhaus_renderer.cpp" global b32 Running = false; global b32 WindowIsActive = false; char DLLName[] = "foldhaus.dll"; char WorkingDLLName[] = "foldhaus_temp.dll"; char DLLLockFileName[] = "lock.tmp"; window MainWindow; PLATFORM_GET_GPU_TEXTURE_HANDLE(Win32GetGPUTextureHandle) { s32 Handle = SubmitTexture(Memory, Width, Height); return Handle; } HDC FontDrawingDC; HBITMAP FontBitmap; HFONT CurrentFont; GET_FONT_INFO(Win32GetFontInfo) { platform_font_info Result = {}; FontDrawingDC = CreateCompatibleDC(NULL); SetBkColor(FontDrawingDC, RGB(0, 0, 0)); SetTextColor(FontDrawingDC, RGB(255, 255, 255)); FontBitmap = CreateCompatibleBitmap(FontDrawingDC, PixelHeight * 2, PixelHeight * 2); HGDIOBJ SelectObjectResult = SelectObject(FontDrawingDC, FontBitmap); CurrentFont = CreateFont(PixelHeight, 0, 0, 0, FontWeight, Italic, Underline, Strikeout, ANSI_CHARSET, OUT_OUTLINE_PRECIS, CLIP_DEFAULT_PRECIS, PROOF_QUALITY, FIXED_PITCH, FontName); SelectFont(FontDrawingDC, CurrentFont); TEXTMETRIC WindowsFontMetrics = {}; if (GetTextMetrics(FontDrawingDC, &WindowsFontMetrics)) { Result.PixelHeight = WindowsFontMetrics.tmHeight; Result.Ascent = WindowsFontMetrics.tmAscent; Result.Descent = WindowsFontMetrics.tmDescent; Result.Leading = WindowsFontMetrics.tmExternalLeading; Result.MaxCharWidth = WindowsFontMetrics.tmMaxCharWidth; Result.CodepointStart = WindowsFontMetrics.tmFirstChar; Result.CodepointOnePastLast = WindowsFontMetrics.tmLastChar + 1; } return Result; } DRAW_FONT_CODEPOINT(Win32DrawFontCodepoint) { SIZE CodepointSize = {}; if (GetTextExtentPoint32(FontDrawingDC, &Codepoint, 1, &CodepointSize)) { *OutWidth = CodepointSize.cx; *OutHeight = CodepointSize.cy; RECT TextRect = {}; TextRect.left = 0; TextRect.right = *OutWidth; TextRect.top = 0; TextRect.bottom = *OutHeight; int Error = DrawText(FontDrawingDC, &Codepoint, 1, &TextRect, DT_LEFT | DT_NOCLIP | DT_TOP); u8* Row = DestBuffer + (YOffset * (DestBufferWidth * 4)); COLORREF PixelColor; for (u32 Y = 0; Y < *OutHeight; Y++) { // NOTE(Peter): XOffset * 4 b/c its 4 bytes per pixel. u8* Channel = (u8*)Row + (XOffset * 4); for (u32 X = 0; X < *OutWidth; X++) { PixelColor = GetPixel(FontDrawingDC, X + TextRect.left, TextRect.bottom - Y); Assert(PixelColor != CLR_INVALID); u8 RValue = GetRValue(PixelColor); *Channel++ = RValue; *Channel++ = RValue; *Channel++ = RValue; *Channel++ = RValue; } Row += DestBufferWidth * 4; } } } LRESULT CALLBACK HandleWindowEvents (HWND WindowHandle, UINT Msg, WPARAM WParam, LPARAM LParam) { LRESULT Result = 0; switch (Msg) { case WM_SIZE: { Win32UpdateWindowDimension(&MainWindow); //Win32ResizeDIBSection(&GlobalBackbuffer, MainWindow.Info.Width, MainWindow.Info.Height); }break; case WM_CLOSE: { Result = DefWindowProc(WindowHandle, Msg, WParam, LParam); Running = false; }break; case WM_DESTROY: { }break; case WM_PAINT: { PAINTSTRUCT PaintStruct; HDC DeviceContext; b32 PaintResult; DeviceContext = BeginPaint(WindowHandle, &PaintStruct); PaintResult = EndPaint(WindowHandle, &PaintStruct); }break; case WM_ACTIVATE: { WindowIsActive = (LOWORD(WParam) == WA_ACTIVE || LOWORD(WParam) == WA_CLICKACTIVE); }break; default: { Result = DefWindowProc(WindowHandle, Msg, WParam, LParam); } } return Result; } internal void HandleWindowMessage (MSG Message, window* Window, input_queue* InputQueue, mouse_state* Mouse) { switch (Message.message) { case WM_MOUSEWHEEL: { Mouse->Scroll = GET_WHEEL_DELTA_WPARAM(Message.wParam); }break; case WM_LBUTTONDOWN: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseLeftButton, false, true, ShiftDown, AltDown, CtrlDown, false); Mouse->LeftButtonState = KeyState_IsDown & ~KeyState_WasDown; Mouse->DownPos = Mouse->Pos; // :Win32MouseEventCapture // NOTE(Peter): We capture events when the mouse goes down so that // if the user drags outside the window, we still get the mouse up // event and can process it. Otherwise, we can get into cases where // an event was started, didn't end, but the user can click again and // try to start the event again. // We relase event capture on mouse up. SetCapture(Window->Handle); }break; case WM_MBUTTONDOWN: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseMiddleButton, false, true, ShiftDown, AltDown, CtrlDown, false); Mouse->MiddleButtonState = KeyState_IsDown & ~KeyState_WasDown; // :Win32MouseEventCapture SetCapture(Window->Handle); }break; case WM_RBUTTONDOWN: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseRightButton, false, true, ShiftDown, AltDown, CtrlDown, false); Mouse->RightButtonState = KeyState_IsDown & ~KeyState_WasDown; Mouse->DownPos = Mouse->Pos; // :Win32MouseEventCapture SetCapture(Window->Handle); }break; case WM_LBUTTONUP: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseLeftButton, true, false, ShiftDown, AltDown, CtrlDown, false); Mouse->LeftButtonState = ~KeyState_IsDown & KeyState_WasDown; // :Win32MouseEventCapture ReleaseCapture(); }break; case WM_MBUTTONUP: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseMiddleButton, true, false, ShiftDown, AltDown, CtrlDown, false); Mouse->MiddleButtonState = ~KeyState_IsDown & KeyState_WasDown; // :Win32MouseEventCapture ReleaseCapture(); }break; case WM_RBUTTONUP: { b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; AddInputEventEntry(InputQueue, KeyCode_MouseRightButton, true, false, ShiftDown, AltDown, CtrlDown, false); Mouse->RightButtonState = ~KeyState_IsDown & KeyState_WasDown; // :Win32MouseEventCapture ReleaseCapture(); }break; case WM_SYSKEYDOWN: case WM_SYSKEYUP: case WM_KEYDOWN: case WM_KEYUP: { int VirtualKey = (int)Message.wParam; key_code Key = Win32GetKeyCode(VirtualKey, true, false); s32 KeyIndex = (int)Key; b32 KeyWasDown = (Message.lParam & (1 << 30)) != 0; b32 KeyIsDown = (Message.lParam & (1 << 31)) == 0; b32 ShiftDown = GetKeyState(VK_SHIFT) & 0x8000; b32 AltDown = GetKeyState(VK_MENU) & 0x8000; b32 CtrlDown = GetKeyState(VK_CONTROL) & 0x8000; // New Input Queue AddInputEventEntry(InputQueue, Key, KeyWasDown, KeyIsDown, ShiftDown, AltDown, CtrlDown, false); }break; default: { TranslateMessage(&Message); DispatchMessage(&Message); }break; } } internal void DebugPrint (char* Format, ...) { char Buffer[256]; gs_string StringBuffer = MakeString(Buffer, 256); va_list Args; va_start(Args, Format); PrintF(&StringBuffer, Format, Args); OutputDebugStringA(Buffer); va_end(Args); } internal void SetApplicationLinks (context* Context, win32_dll_refresh DLL, gs_work_queue* WorkQueue) { if (DLL.IsValid) { Context->InitializeApplication = (initialize_application*)GetProcAddress(DLL.DLL, "InitializeApplication"); Context->ReloadStaticData = (reload_static_data*)GetProcAddress(DLL.DLL, "ReloadStaticData"); Context->UpdateAndRender = (update_and_render*)GetProcAddress(DLL.DLL, "UpdateAndRender"); Context->CleanupApplication = (cleanup_application*)GetProcAddress(DLL.DLL, "CleanupApplication"); } else { Context->InitializeApplication = 0; Context->ReloadStaticData = 0; Context->UpdateAndRender = 0; Context->CleanupApplication = 0; } } // TODO(Peter): :Redundant remove internal u8* DEBUGAlloc(s32 ElementSize, s32 ElementCount) { return (u8*)Win32Alloc(ElementSize * ElementCount, 0); } // TODO(Peter): :Redundant remove internal u8* Win32Realloc(u8* Buf, s32 OldSize, s32 NewSize) { u8* NewMemory = (u8*)Win32Alloc(NewSize, 0); CopyMemoryTo(Buf, NewMemory, OldSize); return NewMemory; } // NOTE(Peter): Only meant to take one of the values specified below: // IDC_APPSTARTING, IDC_ARROW, IDC_CROSS, IDC_HAND, IDC_HELP, IDC_IBEAM, // IDC_ICON, IDC_NO, IDC_SIZE, IDC_SIZEALL, IDC_SIZENESW, IDC_SIZENS, IDC_SIZENWSE, // IDC_SIZEWE, IDC_UPARROW, IDC_WAIT internal HCURSOR Win32LoadSystemCursor(char* CursorIdentifier) { u32 Error = 0; HCURSOR Result = LoadCursorA(NULL, CursorIdentifier); if (Result == NULL) { Error = GetLastError(); InvalidCodePath; } return Result; } internal void Win32_SendAddressedDataBuffers(gs_thread_context Context, addressed_data_buffer_list OutputData) { DEBUG_TRACK_FUNCTION; u32 BuffersSent = 0; for (addressed_data_buffer* BufferAt = OutputData.Root; BufferAt != 0; BufferAt = BufferAt->Next) { switch(BufferAt->AddressType) { case AddressType_NetworkIP: { Win32Socket_SendTo(BufferAt->SendSocket, BufferAt->V4SendAddress, BufferAt->SendPort, (const char*)BufferAt->Memory, BufferAt->MemorySize, 0); }break; case AddressType_ComPort: { if (BufferAt->ComPort.Length > 0) { HANDLE SerialPort = Win32SerialArray_GetOrOpen(BufferAt->ComPort, 2000000, 8, NOPARITY, 1); if (SerialPort != INVALID_HANDLE_VALUE) { if (Win32SerialPort_Write(SerialPort, BufferAt->Data)) { BuffersSent += 1; } } } else { OutputDebugStringA("Skipping data buffer because its COM Port isn't set"); } }break; InvalidDefaultCase; } } gs_string OutputStr = AllocatorAllocString(Context.Allocator, 256); PrintF(&OutputStr, "Buffers Sent: %d\n", BuffersSent); NullTerminate(&OutputStr); //OutputDebugStringA(OutputStr.Str); } internal void Win32_SendAddressedDataBuffers_Job(gs_thread_context Context, gs_data Arg) { addressed_data_buffer_list* OutputData = (addressed_data_buffer_list*)Arg.Memory; Win32_SendAddressedDataBuffers(Context, *OutputData); } int WINAPI WinMain ( HINSTANCE HInstance, HINSTANCE HPrevInstance, PSTR CmdLineArgs, INT NCmdShow ) { gs_thread_context ThreadContext = Win32CreateThreadContext(); #if 0 u32 LedCount = 48; u32 MessageBaseSize = sizeof(uart_header) + sizeof(uart_channel) + sizeof(uart_footer); MessageBaseSize += sizeof(u8) * 3 * LedCount; gs_data MessageBuffer = PushSizeToData(ThreadContext.Transient); gs_memory_cursor WriteCursor = CreateMemoryCursor(MessageBuffer); uart_header* Header = PushStructOnCursor(WriteCursor, uart_header); UART_FillHeader(Header, Strip.UARTAddr.Channel, UART_SET_CHANNEL_WS2812); uart_channel* Channel = PushStructOnCursor(WriteCursor, uart_channel); *Channel = ChannelSettings; for (u32 i = 0; i < LedCount; i++) { u8* OutputPixel = PushArrayOnCursor(WriteCursor, u8, 3); OutputPixel[Channel->RedIndex] = (u8)(i); OutputPixel[Channel->GreenIndex] = 0; OutputPixel[Channel->BlueIndex] = 0; } uart_footer* Footer = PushStructOnCursor(WriteCursor, uart_footer); UART_FillFooter(Footer, (u8*)Header); #endif MainWindow = Win32CreateWindow (HInstance, "Foldhaus", 1440, 768, HandleWindowEvents); Win32UpdateWindowDimension(&MainWindow); win32_opengl_window_info OpenGLWindowInfo = {}; OpenGLWindowInfo.ColorBits = 32; OpenGLWindowInfo.AlphaBits = 8; OpenGLWindowInfo.DepthBits = 0; CreateOpenGLWindowContext(OpenGLWindowInfo, &MainWindow); s32 InitialMemorySize = MB(64); u8* InitialMemory = (u8*)Win32Alloc(InitialMemorySize, 0); context Context = {}; Context.ThreadContext = ThreadContext; Context.MemorySize = InitialMemorySize; Context.MemoryBase = InitialMemory; Context.WindowBounds = rect2{v2{0, 0}, v2{(r32)MainWindow.Width, (r32)MainWindow.Height}}; Context.Mouse = {0}; gs_memory_arena PlatformPermanent = CreateMemoryArena(Context.ThreadContext.Allocator); s64 PerformanceCountFrequency = GetPerformanceFrequency(); s64 LastFrameEnd = GetWallClock(); r32 TargetSecondsPerFrame = 1 / 60.0f; r32 LastFrameSecondsElapsed = 0.0f; GlobalDebugServices = PushStruct(&PlatformPermanent, debug_services); s32 DebugThreadCount = PLATFORM_THREAD_COUNT + 1; InitDebugServices(GlobalDebugServices, PerformanceCountFrequency, DEBUGAlloc, Win32Realloc, GetWallClock, Win32GetThreadId, DebugThreadCount); input_queue InputQueue; { s32 InputQueueMemorySize = sizeof(input_entry) * 32; u8* InputQueueMemory = (u8*)Win32Alloc(InputQueueMemorySize, 0); InputQueue = InitializeInputQueue(InputQueueMemory, InputQueueMemorySize); } // // Set up worker threads const s32 WorkerThreadCount = PLATFORM_THREAD_COUNT; worker_thread_info* WorkerThreads = 0; if (PLATFORM_THREAD_COUNT > 0) { WorkerThreads = PushArray(&PlatformPermanent, worker_thread_info, PLATFORM_THREAD_COUNT); } HANDLE WorkQueueSemaphoreHandle = CreateSemaphoreEx(0, 0, PLATFORM_THREAD_COUNT, 0, 0, SEMAPHORE_ALL_ACCESS); gs_work_queue WorkQueue = {}; WorkQueue.SemaphoreHandle = &WorkQueueSemaphoreHandle; WorkQueue.JobsMax = 512; WorkQueue.Jobs = PushArray(&PlatformPermanent, gs_threaded_job, WorkQueue.JobsMax); WorkQueue.NextJobIndex = 0; WorkQueue.PushWorkOnQueue = Win32PushWorkOnQueue; WorkQueue.CompleteQueueWork = Win32DoQueueWorkUntilDone; WorkQueue.ResetWorkQueue = ResetWorkQueue; for (s32 i = 0; i < PLATFORM_THREAD_COUNT; i++) { // ID = 0 is reserved for this thread WorkerThreads[i].Queue = &WorkQueue; WorkerThreads[i].Handle = CreateThread(0, 0, &WorkerThreadProc, (void*)&WorkerThreads[i], 0, 0); } // Cursors HCURSOR CursorArrow = Win32LoadSystemCursor(IDC_ARROW); HCURSOR CursorPointer = Win32LoadSystemCursor(IDC_HAND); HCURSOR CursorLoading = Win32LoadSystemCursor(IDC_WAIT); HCURSOR CursorHorizontalArrows = Win32LoadSystemCursor(IDC_SIZEWE); HCURSOR CursorVerticalArrows = Win32LoadSystemCursor(IDC_SIZENS); HCURSOR CursorDiagonalTopLeftArrows = Win32LoadSystemCursor(IDC_SIZENWSE); HCURSOR CursorDiagonalTopRightArrows = Win32LoadSystemCursor(IDC_SIZENESW); // Platform functions Context.GeneralWorkQueue = &WorkQueue; Context.PlatformGetGPUTextureHandle = Win32GetGPUTextureHandle; Context.PlatformGetSocketHandle = Win32GetSocketHandle; Context.PlatformGetFontInfo = Win32GetFontInfo; Context.PlatformDrawFontCodepoint = Win32DrawFontCodepoint; win32_dll_refresh DLLRefresh = InitializeDLLHotReloading(DLLName, WorkingDLLName, DLLLockFileName); if (HotLoadDLL(&DLLRefresh)) { SetApplicationLinks(&Context, DLLRefresh, &WorkQueue); Context.ReloadStaticData(Context, GlobalDebugServices); } else { MessageBox(MainWindow.Handle, "Unable to load application DLL at startup.\nAborting\n", "Set Up Error", MB_OK); return -1; } WSADATA WSAData; WSAStartup(MAKEWORD(2, 2), &WSAData); Win32Sockets = Win32SocketArray_Create(16, &PlatformPermanent); Win32SerialArray_Create(ThreadContext); s32 RenderMemorySize = MB(12); u8* RenderMemory = PushSize(&PlatformPermanent, RenderMemorySize); render_command_buffer RenderBuffer = AllocateRenderCommandBuffer(RenderMemory, RenderMemorySize, Win32Realloc); addressed_data_buffer_list OutputData = {}; OutputData.Arena = AllocatorAllocStruct(Context.ThreadContext.Allocator, gs_memory_arena); *OutputData.Arena = CreateMemoryArena(Context.ThreadContext.Allocator); Context.InitializeApplication(Context); Running = true; Context.WindowIsVisible = true; while (Running) { if (GlobalDebugServices->RecordFrames) { EndDebugFrame(GlobalDebugServices); } DEBUG_TRACK_SCOPE(MainLoop); ResetInputQueue(&InputQueue); if (HotLoadDLL(&DLLRefresh)) { SetApplicationLinks(&Context, DLLRefresh, &WorkQueue); Context.ReloadStaticData(Context, GlobalDebugServices); } AddressedDataBufferList_Clear(&OutputData); { // Mouse Position POINT MousePos; GetCursorPos (&MousePos); ScreenToClient(MainWindow.Handle, &MousePos); Context.Mouse.Scroll = 0; Context.Mouse.OldPos = Context.Mouse.Pos; Context.Mouse.Pos = v2{(r32)MousePos.x, (r32)MainWindow.Height - MousePos.y}; Context.Mouse.DeltaPos = Context.Mouse.Pos - Context.Mouse.OldPos; } MSG Message; while (PeekMessageA(&Message, MainWindow.Handle, 0, 0, PM_REMOVE)) { DEBUG_TRACK_SCOPE(PeekWindowsMessages); HandleWindowMessage(Message, &MainWindow, &InputQueue, &Context.Mouse); } Context.WindowBounds = rect2{v2{0, 0}, v2{(r32)MainWindow.Width, (r32)MainWindow.Height}}; RenderBuffer.ViewWidth = MainWindow.Width; RenderBuffer.ViewHeight = MainWindow.Height; Context.DeltaTime = LastFrameSecondsElapsed; Context.UpdateAndRender(&Context, InputQueue, &RenderBuffer, &OutputData); RenderCommandBuffer(RenderBuffer); ClearRenderBuffer(&RenderBuffer); if (true) { gs_data ProcArg = {}; ProcArg.Memory = (u8*)&OutputData; ProcArg.Size = sizeof(OutputData); Win32PushWorkOnQueue(&WorkQueue, Win32_SendAddressedDataBuffers_Job, ProcArg, ConstString("Send UART Data")); } Context.Mouse.LeftButtonState = GetMouseButtonStateAdvanced(Context.Mouse.LeftButtonState); Context.Mouse.MiddleButtonState = GetMouseButtonStateAdvanced(Context.Mouse.MiddleButtonState); Context.Mouse.RightButtonState = GetMouseButtonStateAdvanced(Context.Mouse.RightButtonState); switch (Context.Mouse.CursorType) { case CursorType_Arrow: { SetCursor(CursorArrow); }break; case CursorType_Pointer: { SetCursor(CursorPointer); }break; case CursorType_Loading: { SetCursor(CursorLoading); }break; case CursorType_HorizontalArrows: { SetCursor(CursorHorizontalArrows); }break; case CursorType_VerticalArrows: { SetCursor(CursorVerticalArrows); }break; case CursorType_DiagonalTopLeftArrows: { SetCursor(CursorDiagonalTopLeftArrows); }break; case CursorType_DiagonalTopRightArrows: { SetCursor(CursorDiagonalTopRightArrows); }break; InvalidDefaultCase; } HDC DeviceContext = GetDC(MainWindow.Handle); SwapBuffers(DeviceContext); ReleaseDC(MainWindow.Handle, DeviceContext); //Win32DoQueueWorkUntilDone(&WorkQueue, Context.ThreadContext); s64 FinishedWorkTime = GetWallClock(); r32 SecondsElapsed = GetSecondsElapsed(LastFrameEnd, FinishedWorkTime, PerformanceCountFrequency); while (SecondsElapsed < TargetSecondsPerFrame) { DEBUG_TRACK_SCOPE(UnusedTime); u32 SleepTime = 1000.0f * (TargetSecondsPerFrame - SecondsElapsed); Sleep(SleepTime); SecondsElapsed = GetSecondsElapsed(LastFrameEnd, GetWallClock(), PerformanceCountFrequency); } LastFrameSecondsElapsed = SecondsElapsed; LastFrameEnd = GetWallClock(); OutputDebugStringA("-- Frame END -- \n"); } Context.CleanupApplication(Context); for (s32 SocketIdx = 0; SocketIdx < Win32Sockets.Count; SocketIdx++) { Win32Socket_Close(Win32Sockets.Values + SocketIdx); } s32 CleanupResult = 0; do { CleanupResult = WSACleanup(); }while(CleanupResult == SOCKET_ERROR); for (s32 Thread = 0; Thread < PLATFORM_THREAD_COUNT; Thread++) { TerminateThread(WorkerThreads[Thread].Handle, 0); } return 0; } #define WIN32_FOLDHAUS_CPP #endif // WIN32_FOLDHAUS_CPP