2019-02-28 08:20:35 +00:00
|
|
|
/*
|
|
|
|
Looking Glass - KVM FrameRelay (KVMFR) Client
|
|
|
|
Copyright (C) 2017-2019 Geoffrey McRae <geoff@hostfission.com>
|
|
|
|
https://looking-glass.hostfission.com
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify it under
|
|
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation; either version 2 of the License, or (at your option) any later
|
|
|
|
version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
|
|
|
PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
|
|
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
|
|
|
|
Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
#include "platform.h"
|
|
|
|
#include "windows/platform.h"
|
|
|
|
#include "windows/mousehook.h"
|
2019-02-28 08:20:35 +00:00
|
|
|
#include <windows.h>
|
|
|
|
#include <setupapi.h>
|
|
|
|
|
2019-04-09 06:28:11 +00:00
|
|
|
#include "interface/platform.h"
|
2019-02-28 08:20:35 +00:00
|
|
|
#include "debug.h"
|
2019-04-09 06:28:11 +00:00
|
|
|
#include "windows/windebug.h"
|
2019-02-28 08:20:35 +00:00
|
|
|
#include "ivshmem/Public.h"
|
|
|
|
|
2019-02-28 08:27:17 +00:00
|
|
|
static HANDLE shmemHandle = INVALID_HANDLE_VALUE;
|
|
|
|
static bool shmemOwned = false;
|
|
|
|
static IVSHMEM_MMAP shmemMap = {0};
|
2019-03-01 04:45:46 +00:00
|
|
|
static HWND messageWnd;
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-03-01 01:24:23 +00:00
|
|
|
struct osThreadHandle
|
|
|
|
{
|
|
|
|
const char * name;
|
|
|
|
osThreadFunction function;
|
|
|
|
void * opaque;
|
|
|
|
HANDLE handle;
|
|
|
|
DWORD threadID;
|
2019-04-10 11:07:56 +00:00
|
|
|
|
2019-03-01 01:24:23 +00:00
|
|
|
int resultCode;
|
|
|
|
};
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
LRESULT CALLBACK DummyWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
2019-02-28 08:20:35 +00:00
|
|
|
{
|
2019-03-01 04:45:46 +00:00
|
|
|
switch(msg)
|
|
|
|
{
|
|
|
|
case WM_CLOSE:
|
2019-04-10 11:07:56 +00:00
|
|
|
mouseHook_remove();
|
2019-03-01 04:45:46 +00:00
|
|
|
DestroyWindow(hwnd);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_DESTROY:
|
|
|
|
PostQuitMessage(0);
|
|
|
|
break;
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
case WM_CALL_FUNCTION:
|
|
|
|
{
|
|
|
|
struct MSG_CALL_FUNCTION * cf = (struct MSG_CALL_FUNCTION *)lParam;
|
|
|
|
return cf->fn(cf->wParam, cf->lParam);
|
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
default:
|
|
|
|
return DefWindowProc(hwnd, msg, wParam, lParam);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int appThread(void * opaque)
|
|
|
|
{
|
2019-03-02 09:31:33 +00:00
|
|
|
int result = app_main();
|
2019-03-01 04:45:46 +00:00
|
|
|
SendMessage(messageWnd, WM_CLOSE, 0, 0);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
LRESULT sendAppMessage(UINT Msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
return SendMessage(messageWnd, Msg, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL WINAPI CtrlHandler(DWORD dwCtrlType)
|
|
|
|
{
|
|
|
|
if (dwCtrlType == CTRL_C_EVENT)
|
|
|
|
{
|
|
|
|
SendMessage(messageWnd, WM_CLOSE, 0, 0);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
|
|
|
|
{
|
|
|
|
int result = 0;
|
2019-02-28 08:20:35 +00:00
|
|
|
HDEVINFO deviceInfoSet;
|
|
|
|
PSP_DEVICE_INTERFACE_DETAIL_DATA infData = NULL;
|
|
|
|
SP_DEVICE_INTERFACE_DATA deviceInterfaceData;
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
#if 0
|
|
|
|
// redirect stderr to a file
|
|
|
|
{
|
|
|
|
char tempPath[MAX_PATH+1];
|
|
|
|
GetTempPathA(sizeof(tempPath), tempPath);
|
|
|
|
int len = snprintf(NULL, 0, "%slooking-glass-host.txt", tempPath);
|
|
|
|
char * path = malloc(len + 1);
|
|
|
|
sprintf(path, "%slooking-glass-host.txt", tempPath);
|
|
|
|
freopen(path, "a", stderr);
|
|
|
|
free(path);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// always flush stderr
|
|
|
|
setbuf(stderr, NULL);
|
|
|
|
|
2019-02-28 08:20:35 +00:00
|
|
|
deviceInfoSet = SetupDiGetClassDevs(NULL, NULL, NULL, DIGCF_PRESENT | DIGCF_ALLCLASSES | DIGCF_DEVICEINTERFACE);
|
|
|
|
memset(&deviceInterfaceData, 0, sizeof(SP_DEVICE_INTERFACE_DATA));
|
|
|
|
deviceInterfaceData.cbSize = sizeof(SP_DEVICE_INTERFACE_DATA);
|
|
|
|
|
|
|
|
if (SetupDiEnumDeviceInterfaces(deviceInfoSet, NULL, &GUID_DEVINTERFACE_IVSHMEM, 0, &deviceInterfaceData) == FALSE)
|
|
|
|
{
|
|
|
|
DWORD error = GetLastError();
|
|
|
|
if (error == ERROR_NO_MORE_ITEMS)
|
|
|
|
{
|
|
|
|
DEBUG_WINERROR("Unable to enumerate the device, is it attached?", error);
|
2019-03-01 04:45:46 +00:00
|
|
|
result = -1;
|
|
|
|
goto finish;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_WINERROR("SetupDiEnumDeviceInterfaces failed", error);
|
2019-03-01 04:45:46 +00:00
|
|
|
result = -1;
|
|
|
|
goto finish;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DWORD reqSize = 0;
|
|
|
|
SetupDiGetDeviceInterfaceDetail(deviceInfoSet, &deviceInterfaceData, NULL, 0, &reqSize, NULL);
|
|
|
|
if (!reqSize)
|
|
|
|
{
|
|
|
|
DEBUG_WINERROR("SetupDiGetDeviceInterfaceDetail", GetLastError());
|
2019-03-01 04:45:46 +00:00
|
|
|
result = -1;
|
|
|
|
goto finish;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
infData = (PSP_DEVICE_INTERFACE_DETAIL_DATA)calloc(reqSize, 1);
|
|
|
|
infData->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA);
|
|
|
|
if (!SetupDiGetDeviceInterfaceDetail(deviceInfoSet, &deviceInterfaceData, infData, reqSize, NULL, NULL))
|
|
|
|
{
|
|
|
|
free(infData);
|
|
|
|
DEBUG_WINERROR("SetupDiGetDeviceInterfaceDetail", GetLastError());
|
2019-03-01 04:45:46 +00:00
|
|
|
result = -1;
|
|
|
|
goto finish;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
shmemHandle = CreateFile(infData->DevicePath, 0, 0, NULL, OPEN_EXISTING, 0, 0);
|
|
|
|
if (shmemHandle == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
SetupDiDestroyDeviceInfoList(deviceInfoSet);
|
|
|
|
free(infData);
|
|
|
|
DEBUG_WINERROR("CreateFile returned INVALID_HANDLE_VALUE", GetLastError());
|
2019-03-01 04:45:46 +00:00
|
|
|
result = -1;
|
|
|
|
goto finish;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
free(infData);
|
|
|
|
SetupDiDestroyDeviceInfoList(deviceInfoSet);
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
// setup a handler for ctrl+c
|
|
|
|
SetConsoleCtrlHandler(CtrlHandler, TRUE);
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
// create a message window so that our message pump works
|
|
|
|
WNDCLASSEX wx = {};
|
|
|
|
wx.cbSize = sizeof(WNDCLASSEX);
|
|
|
|
wx.lpfnWndProc = DummyWndProc;
|
|
|
|
wx.hInstance = hInstance;
|
|
|
|
wx.lpszClassName = "DUMMY_CLASS";
|
|
|
|
if (!RegisterClassEx(&wx))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to register message window class");
|
|
|
|
result = -1;
|
|
|
|
goto finish_shmem;
|
|
|
|
}
|
|
|
|
messageWnd = CreateWindowEx(0, "DUMMY_CLASS", "DUMMY_NAME", 0, 0, 0, 0, 0, HWND_MESSAGE, NULL, NULL, NULL);
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
osThreadHandle * thread;
|
|
|
|
if (!os_createThread("appThread", appThread, NULL, &thread))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to create the main application thread");
|
|
|
|
result = -1;
|
|
|
|
goto finish_shmem;
|
|
|
|
}
|
2019-02-28 09:50:22 +00:00
|
|
|
|
2019-03-02 09:31:33 +00:00
|
|
|
while(true)
|
2019-02-28 09:50:22 +00:00
|
|
|
{
|
2019-03-01 04:45:46 +00:00
|
|
|
MSG msg;
|
|
|
|
BOOL bRet = GetMessage(&msg, NULL, 0, 0);
|
|
|
|
if (bRet > 0)
|
|
|
|
{
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessage(&msg);
|
2019-04-10 11:07:56 +00:00
|
|
|
continue;
|
2019-03-01 04:45:46 +00:00
|
|
|
}
|
|
|
|
else if (bRet < 0)
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Unknown error from GetMessage");
|
|
|
|
result = -1;
|
|
|
|
goto shutdown;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_INFO("Platform shutdown");
|
|
|
|
break;
|
2019-02-28 09:50:22 +00:00
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
shutdown:
|
2019-03-02 09:31:33 +00:00
|
|
|
app_quit();
|
2019-03-01 04:45:46 +00:00
|
|
|
if (!os_joinThread(thread, &result))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to join the main application thread");
|
|
|
|
result = -1;
|
|
|
|
}
|
|
|
|
finish_shmem:
|
|
|
|
os_shmemUnmap();
|
|
|
|
CloseHandle(shmemHandle);
|
|
|
|
finish:
|
2019-02-28 08:20:35 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int os_shmemSize()
|
|
|
|
{
|
|
|
|
IVSHMEM_SIZE size;
|
|
|
|
if (!DeviceIoControl(shmemHandle, IOCTL_IVSHMEM_REQUEST_SIZE, NULL, 0, &size, sizeof(IVSHMEM_SIZE), NULL, NULL))
|
|
|
|
{
|
|
|
|
DEBUG_WINERROR("DeviceIoControl Failed", GetLastError());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (unsigned int)size;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool os_shmemMmap(void **ptr)
|
|
|
|
{
|
2019-02-28 08:27:17 +00:00
|
|
|
if (shmemOwned)
|
|
|
|
{
|
|
|
|
*ptr = shmemMap.ptr;
|
|
|
|
return true;
|
|
|
|
}
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-02-28 08:27:17 +00:00
|
|
|
memset(&shmemMap, 0, sizeof(IVSHMEM_MMAP));
|
2019-02-28 08:20:35 +00:00
|
|
|
if (!DeviceIoControl(
|
|
|
|
shmemHandle,
|
|
|
|
IOCTL_IVSHMEM_REQUEST_MMAP,
|
|
|
|
NULL, 0,
|
2019-02-28 08:27:17 +00:00
|
|
|
&shmemMap, sizeof(IVSHMEM_MMAP),
|
2019-02-28 08:20:35 +00:00
|
|
|
NULL, NULL))
|
|
|
|
{
|
|
|
|
DEBUG_WINERROR("DeviceIoControl Failed", GetLastError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-28 08:27:17 +00:00
|
|
|
*ptr = shmemMap.ptr;
|
|
|
|
shmemOwned = true;
|
2019-02-28 08:20:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void os_shmemUnmap()
|
|
|
|
{
|
2019-02-28 08:27:17 +00:00
|
|
|
if (!shmemOwned)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!DeviceIoControl(shmemHandle, IOCTL_IVSHMEM_RELEASE_MMAP, NULL, 0, NULL, 0, NULL, NULL))
|
|
|
|
DEBUG_WINERROR("DeviceIoControl failed", GetLastError());
|
|
|
|
else
|
|
|
|
shmemOwned = false;
|
2019-03-01 01:24:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD WINAPI threadWrapper(LPVOID lpParameter)
|
|
|
|
{
|
|
|
|
osThreadHandle * handle = (osThreadHandle *)lpParameter;
|
|
|
|
handle->resultCode = handle->function(handle->opaque);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool os_createThread(const char * name, osThreadFunction function, void * opaque, osThreadHandle ** handle)
|
|
|
|
{
|
2019-03-01 01:41:37 +00:00
|
|
|
*handle = (osThreadHandle *)malloc(sizeof(osThreadHandle));
|
|
|
|
(*handle)->name = name;
|
|
|
|
(*handle)->function = function;
|
|
|
|
(*handle)->opaque = opaque;
|
|
|
|
(*handle)->handle = CreateThread(NULL, 0, threadWrapper, *handle, 0, &(*handle)->threadID);
|
2019-03-01 01:24:23 +00:00
|
|
|
|
|
|
|
if (!(*handle)->handle)
|
|
|
|
{
|
|
|
|
free(*handle);
|
|
|
|
*handle = NULL;
|
|
|
|
DEBUG_WINERROR("CreateThread failed", GetLastError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool os_joinThread(osThreadHandle * handle, int * resultCode)
|
|
|
|
{
|
|
|
|
while(true)
|
|
|
|
{
|
|
|
|
switch(WaitForSingleObject(handle->handle, INFINITE))
|
|
|
|
{
|
|
|
|
case WAIT_OBJECT_0:
|
|
|
|
if (resultCode)
|
|
|
|
*resultCode = handle->resultCode;
|
|
|
|
CloseHandle(handle->handle);
|
|
|
|
free(handle);
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case WAIT_ABANDONED:
|
|
|
|
case WAIT_TIMEOUT:
|
|
|
|
continue;
|
|
|
|
|
|
|
|
case WAIT_FAILED:
|
|
|
|
DEBUG_WINERROR("Wait for thread failed", GetLastError());
|
|
|
|
CloseHandle(handle->handle);
|
|
|
|
free(handle);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_WINERROR("Unknown failure waiting for thread", GetLastError());
|
|
|
|
return false;
|
2019-03-03 12:30:02 +00:00
|
|
|
}
|
|
|
|
|
2019-03-04 02:38:17 +00:00
|
|
|
osEventHandle * os_createEvent(bool autoReset)
|
2019-03-03 12:30:02 +00:00
|
|
|
{
|
2019-03-04 02:38:17 +00:00
|
|
|
HANDLE event = CreateEvent(NULL, autoReset ? FALSE : TRUE, FALSE, NULL);
|
|
|
|
if (!event)
|
|
|
|
{
|
|
|
|
DEBUG_WINERROR("Failed to create the event", GetLastError());
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-03-03 12:30:02 +00:00
|
|
|
return (osEventHandle*)event;
|
|
|
|
}
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
osEventHandle * os_wrapEvent(HANDLE event)
|
|
|
|
{
|
|
|
|
return (osEventHandle*)event;
|
|
|
|
}
|
|
|
|
|
2019-03-03 12:30:02 +00:00
|
|
|
void os_freeEvent(osEventHandle * handle)
|
|
|
|
{
|
|
|
|
CloseHandle((HANDLE)handle);
|
|
|
|
}
|
|
|
|
|
2019-03-04 05:56:45 +00:00
|
|
|
bool os_waitEvent(osEventHandle * handle, unsigned int timeout)
|
2019-03-03 12:30:02 +00:00
|
|
|
{
|
2019-03-04 05:56:45 +00:00
|
|
|
const DWORD to = (timeout == TIMEOUT_INFINITE) ? INFINITE : (DWORD)timeout;
|
2019-03-03 12:30:02 +00:00
|
|
|
while(true)
|
|
|
|
{
|
2019-03-04 05:56:45 +00:00
|
|
|
switch(WaitForSingleObject((HANDLE)handle, to))
|
2019-03-03 12:30:02 +00:00
|
|
|
{
|
|
|
|
case WAIT_OBJECT_0:
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case WAIT_ABANDONED:
|
|
|
|
continue;
|
|
|
|
|
2019-03-04 05:56:45 +00:00
|
|
|
case WAIT_TIMEOUT:
|
|
|
|
if (timeout == TIMEOUT_INFINITE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
2019-03-03 12:30:02 +00:00
|
|
|
case WAIT_FAILED:
|
|
|
|
DEBUG_WINERROR("Wait for event failed", GetLastError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_ERROR("Unknown wait event return code");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
bool os_waitEvents(osEventHandle * handles[], int count, bool waitAll, unsigned int timeout)
|
|
|
|
{
|
|
|
|
const DWORD to = (timeout == TIMEOUT_INFINITE) ? INFINITE : (DWORD)timeout;
|
|
|
|
while(true)
|
|
|
|
{
|
|
|
|
DWORD result = WaitForMultipleObjects(count, (HANDLE*)handles, waitAll, to);
|
|
|
|
if (result >= WAIT_OBJECT_0 && result < count)
|
|
|
|
{
|
|
|
|
// null non signalled events from the handle list
|
|
|
|
for(int i = 0; i < count; ++i)
|
|
|
|
if (i != result && !os_waitEvent(handles[i], 0))
|
|
|
|
handles[i] = NULL;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result >= WAIT_ABANDONED_0 && result - WAIT_ABANDONED_0 < count)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
switch(result)
|
|
|
|
{
|
|
|
|
case WAIT_TIMEOUT:
|
|
|
|
if (timeout == TIMEOUT_INFINITE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case WAIT_FAILED:
|
|
|
|
DEBUG_WINERROR("Wait for event failed", GetLastError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG_ERROR("Unknown wait event return code");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-03 12:30:02 +00:00
|
|
|
bool os_signalEvent(osEventHandle * handle)
|
|
|
|
{
|
|
|
|
return SetEvent((HANDLE)handle);
|
2019-03-04 02:38:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool os_resetEvent(osEventHandle * handle)
|
|
|
|
{
|
|
|
|
return ResetEvent((HANDLE)handle);
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|