// SemaphoreDemo.cpp : Defines the entry point for the application.
//

#include "stdafx.h"
#include "resource.h"

#define MAX_LOADSTRING 100

// Global Variables:
HINSTANCE hInst;								// current instance
TCHAR szTitle[MAX_LOADSTRING];								// The title bar text
TCHAR szWindowClass[MAX_LOADSTRING];								// The title bar text

//Eliminate globals by passing a struct which is static to the windows proc
typedef struct s_global
	{
	int	Counter;
	HWND GlobalhWnd;
	HANDLE		hSyncDAQ;	//synchronization object for thread
	}GLOBAL;

// Foward declarations of functions included in this code module:
ATOM				MyRegisterClass(HINSTANCE hInstance);
BOOL				InitInstance(HINSTANCE, int);
LRESULT CALLBACK	WndProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK	About(HWND, UINT, WPARAM, LPARAM);
DWORD WINAPI		CountThread(GLOBAL *);


int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
 	// TODO: Place code here.
	MSG msg;
	HACCEL hAccelTable;

	// Initialize global strings
	LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
	LoadString(hInstance, IDC_SEMAPHOREDEMO, szWindowClass, MAX_LOADSTRING);
	MyRegisterClass(hInstance);

	// Perform application initialization:
	if (!InitInstance (hInstance, nCmdShow)) 
	{
		return FALSE;
	}

	hAccelTable = LoadAccelerators(hInstance, (LPCTSTR)IDC_SEMAPHOREDEMO);

	// Main message loop:
	while (GetMessage(&msg, NULL, 0, 0)) 
	{
		if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) 
		{
			TranslateMessage(&msg);
			DispatchMessage(&msg);
		}
	}

	return msg.wParam;
}



//
//  FUNCTION: MyRegisterClass()
//
//  PURPOSE: Registers the window class.
//
//  COMMENTS:
//
//    This function and its usage is only necessary if you want this code
//    to be compatible with Win32 systems prior to the 'RegisterClassEx'
//    function that was added to Windows 95. It is important to call this function
//    so that the application will get 'well formed' small icons associated
//    with it.
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
	WNDCLASSEX wcex;

	wcex.cbSize = sizeof(WNDCLASSEX); 

	wcex.style			= CS_HREDRAW | CS_VREDRAW;
	wcex.lpfnWndProc	= (WNDPROC)WndProc;
	wcex.cbClsExtra		= 0;
	wcex.cbWndExtra		= 0;
	wcex.hInstance		= hInstance;
	wcex.hIcon			= LoadIcon(hInstance, (LPCTSTR)IDI_SEMAPHOREDEMO);
	wcex.hCursor		= LoadCursor(NULL, IDC_ARROW);
	wcex.hbrBackground	= (HBRUSH)(COLOR_WINDOW+1);
	wcex.lpszMenuName	= (LPCSTR)IDC_SEMAPHOREDEMO;
	wcex.lpszClassName	= szWindowClass;
	wcex.hIconSm		= LoadIcon(wcex.hInstance, (LPCTSTR)IDI_SMALL);

	return RegisterClassEx(&wcex);
}

//
//   FUNCTION: InitInstance(HANDLE, int)
//
//   PURPOSE: Saves instance handle and creates main window
//
//   COMMENTS:
//
//        In this function, we save the instance handle in a global variable and
//        create and display the main program window.
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
   HWND hWnd;

   hInst = hInstance; // Store instance handle in our global variable

   hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
      CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);

   if (!hWnd)
   {
      return FALSE;
   }

   ShowWindow(hWnd, nCmdShow);
   UpdateWindow(hWnd);

   return TRUE;
}

//
//  FUNCTION: WndProc(HWND, unsigned, WORD, LONG)
//
//  PURPOSE:  Processes messages for the main window.
//
//  WM_COMMAND	- process the application menu
//  WM_PAINT	- Paint the main window
//  WM_DESTROY	- post a quit message and return
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
	int wmId, wmEvent;
	PAINTSTRUCT ps;
	HDC hdc;
	TCHAR szHello[MAX_LOADSTRING];
	LoadString(hInst, IDS_HELLO, szHello, MAX_LOADSTRING);

	//Data is now local to the Windows proc and shared with the thread - not global
	static GLOBAL SharedData;

	static HANDLE	hThreadDAQ;		//sampling thread handle
	static DWORD	dwDAQThreadId;	

	switch (message) 
	{
		case WM_CREATE:
			SharedData.Counter=0;
			//Grab the handle to share with the thread
			SharedData.GlobalhWnd=hWnd;
			//Create semaphore to control sampling thread 
			//Start thread at 10, max count is 11 (Normally, start 0, max 1)
			SharedData.hSyncDAQ = CreateSemaphore( NULL, 0, 2, NULL );
			if ( SharedData.hSyncDAQ == NULL )
				{
				char	szBuf[80];
				wsprintf( szBuf, "Semaphore Creation FAILED" );
				MessageBox( hWnd, szBuf, "SEMAPHORE Failed", MB_OK );
				}
			
			//Start thread.  Pass a single parameter - pointer to shared data
			hThreadDAQ = CreateThread( NULL,
			      0,
			      (LPTHREAD_START_ROUTINE) CountThread, 
			      &SharedData,
			      0,
			      &dwDAQThreadId );
			//Ensure it actually starts
			if ( hThreadDAQ == NULL )
				{
				char	szBuf[80];
				wsprintf( szBuf, "THREAD CREATE failed");
				MessageBox( hWnd, szBuf, "THREAD Failed", MB_OK );
				}
			break;
		case WM_COMMAND:
			wmId    = LOWORD(wParam); 
			wmEvent = HIWORD(wParam); 
			// Parse the menu selections:
			switch (wmId)
			{
				case IDM_ABOUT:
				   DialogBox(hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd, (DLGPROC)About);
				   break;
				case IDM_EXIT:
				   DestroyWindow(hWnd);
				   break;
				case ID_THREADCONTROL_START:
					ReleaseSemaphore( SharedData.hSyncDAQ, 1, NULL );
					break;				
				default:
				   return DefWindowProc(hWnd, message, wParam, lParam);
			}
			break;
		case THREAD_DONE:
			//Force a re-draw of the screen
			InvalidateRect(SharedData.GlobalhWnd,NULL,NULL);
			//Start Thread Again by releasing semaphore
			ReleaseSemaphore( SharedData.hSyncDAQ, 1, NULL );
			break;
		case WM_PAINT:
			hdc = BeginPaint(hWnd, &ps);
			// TODO: Add any drawing code here...
			RECT rt;
			GetClientRect(hWnd, &rt);
			wsprintf(szHello,"Counter = %d",SharedData.Counter);
			DrawText(hdc, szHello, strlen(szHello), &rt, DT_CENTER);
			EndPaint(hWnd, &ps);
			break;
		case WM_DESTROY:
			PostQuitMessage(0);
			break;
		default:
			return DefWindowProc(hWnd, message, wParam, lParam);
   }
   return 0;
}

// Mesage handler for about box.
LRESULT CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
	switch (message)
	{
		case WM_INITDIALOG:
				return TRUE;

		case WM_COMMAND:
			if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) 
			{
				EndDialog(hDlg, LOWORD(wParam));
				return TRUE;
			}
			break;
	}
    return FALSE;
}

//Actual thread code - contains a Sleep and sends a message to paint the window
DWORD WINAPI CountThread(GLOBAL *SharedData)
{
	while(1)
	{

		//Wait for the semaphore to start counter
		WaitForSingleObject( SharedData->hSyncDAQ, INFINITE );

		SharedData->Counter++;
		//Sleep(1000);
		InvalidateRect(SharedData->GlobalhWnd,NULL,NULL);
	    //SendMessage( SharedData->GlobalhWnd, THREAD_DONE, NULL, NULL);   

	}
}
