2019-02-28 08:20:35 +00:00
|
|
|
/*
|
|
|
|
Looking Glass - KVM FrameRelay (KVMFR) Client
|
2020-01-02 11:21:42 +00:00
|
|
|
Copyright (C) 2017-2020 Geoffrey McRae <geoff@hostfission.com>
|
2019-02-28 08:20:35 +00:00
|
|
|
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/mousehook.h"
|
2019-05-09 09:30:09 +00:00
|
|
|
|
2019-02-28 08:20:35 +00:00
|
|
|
#include <windows.h>
|
2019-05-09 09:30:09 +00:00
|
|
|
#include <shellapi.h>
|
2019-12-13 12:22:11 +00:00
|
|
|
#include <fcntl.h>
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-04-09 06:28:11 +00:00
|
|
|
#include "interface/platform.h"
|
2019-04-11 01:03:30 +00:00
|
|
|
#include "common/debug.h"
|
2020-01-02 11:21:42 +00:00
|
|
|
#include "common/windebug.h"
|
2019-05-10 09:48:38 +00:00
|
|
|
#include "common/option.h"
|
2019-12-17 05:36:43 +00:00
|
|
|
#include "common/locking.h"
|
2020-01-02 11:21:42 +00:00
|
|
|
#include "common/thread.h"
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-05-21 07:52:58 +00:00
|
|
|
#define ID_MENU_OPEN_LOG 3000
|
|
|
|
#define ID_MENU_EXIT 3001
|
|
|
|
|
2019-05-09 09:30:09 +00:00
|
|
|
struct AppState
|
|
|
|
{
|
2019-12-17 03:59:58 +00:00
|
|
|
LARGE_INTEGER perfFreq;
|
2019-05-21 07:52:58 +00:00
|
|
|
HINSTANCE hInst;
|
|
|
|
|
2019-05-09 09:32:19 +00:00
|
|
|
int argc;
|
|
|
|
char ** argv;
|
|
|
|
|
|
|
|
char executable[MAX_PATH + 1];
|
|
|
|
HWND messageWnd;
|
2019-05-21 07:52:58 +00:00
|
|
|
HMENU trayMenu;
|
2019-05-09 09:30:09 +00:00
|
|
|
};
|
|
|
|
|
2020-01-03 03:53:56 +00:00
|
|
|
static struct AppState app = {0};
|
2019-02-28 08:20:35 +00:00
|
|
|
|
2019-12-15 05:20:07 +00:00
|
|
|
// undocumented API to adjust the system timer resolution (yes, its a nasty hack)
|
|
|
|
typedef NTSTATUS (__stdcall *ZwSetTimerResolution_t)(ULONG RequestedResolution, BOOLEAN Set, PULONG ActualResolution);
|
|
|
|
static ZwSetTimerResolution_t ZwSetTimerResolution = NULL;
|
|
|
|
|
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_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-05-21 07:52:58 +00:00
|
|
|
case WM_TRAYICON:
|
|
|
|
{
|
|
|
|
if (lParam == WM_RBUTTONDOWN)
|
|
|
|
{
|
|
|
|
POINT curPoint;
|
|
|
|
GetCursorPos(&curPoint);
|
|
|
|
SetForegroundWindow(hwnd);
|
|
|
|
UINT clicked = TrackPopupMenu(
|
|
|
|
app.trayMenu,
|
|
|
|
TPM_RETURNCMD | TPM_NONOTIFY,
|
|
|
|
curPoint.x,
|
|
|
|
curPoint.y,
|
|
|
|
0,
|
|
|
|
hwnd,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
|
|
|
|
if (clicked == ID_MENU_EXIT ) app_quit();
|
|
|
|
else if (clicked == ID_MENU_OPEN_LOG)
|
|
|
|
{
|
|
|
|
const char * logFile = option_get_string("os", "logFile");
|
|
|
|
if (strcmp(logFile, "stderr") == 0)
|
|
|
|
DEBUG_INFO("Ignoring request to open the logFile, logging to stderr");
|
|
|
|
else
|
|
|
|
ShellExecute(NULL, NULL, logFile, NULL, NULL, SW_SHOWNORMAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
default:
|
|
|
|
return DefWindowProc(hwnd, msg, wParam, lParam);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int appThread(void * opaque)
|
|
|
|
{
|
2019-05-21 07:52:58 +00:00
|
|
|
// register our TrayIcon
|
|
|
|
NOTIFYICONDATA iconData =
|
|
|
|
{
|
|
|
|
.cbSize = sizeof(NOTIFYICONDATA),
|
|
|
|
.hWnd = app.messageWnd,
|
|
|
|
.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP,
|
|
|
|
.uCallbackMessage = WM_TRAYICON,
|
|
|
|
.szTip = "Looking Glass (host)"
|
|
|
|
};
|
|
|
|
iconData.hIcon = LoadIcon(app.hInst, IDI_APPLICATION);
|
|
|
|
Shell_NotifyIcon(NIM_ADD, &iconData);
|
|
|
|
|
2019-05-09 09:30:09 +00:00
|
|
|
int result = app_main(app.argc, app.argv);
|
2019-05-21 07:52:58 +00:00
|
|
|
|
|
|
|
Shell_NotifyIcon(NIM_DELETE, &iconData);
|
|
|
|
mouseHook_remove();
|
|
|
|
SendMessage(app.messageWnd, WM_DESTROY, 0, 0);
|
2019-03-01 04:45:46 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-04-10 11:07:56 +00:00
|
|
|
LRESULT sendAppMessage(UINT Msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
2019-05-09 09:30:09 +00:00
|
|
|
return SendMessage(app.messageWnd, Msg, wParam, lParam);
|
2019-04-10 11:07:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL WINAPI CtrlHandler(DWORD dwCtrlType)
|
|
|
|
{
|
|
|
|
if (dwCtrlType == CTRL_C_EVENT)
|
|
|
|
{
|
2019-05-09 09:30:09 +00:00
|
|
|
SendMessage(app.messageWnd, WM_CLOSE, 0, 0);
|
2019-04-10 11:07:56 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
|
|
|
|
{
|
2019-12-13 12:22:11 +00:00
|
|
|
/* this is a bit of a hack but without this --help will produce no output in a windows command prompt */
|
2019-12-13 12:33:11 +00:00
|
|
|
if (!IsDebuggerPresent() && AttachConsole(ATTACH_PARENT_PROCESS))
|
2019-12-13 12:22:11 +00:00
|
|
|
{
|
|
|
|
HANDLE std_err = GetStdHandle(STD_ERROR_HANDLE);
|
|
|
|
HANDLE std_out = GetStdHandle(STD_OUTPUT_HANDLE);
|
|
|
|
int std_err_fd = _open_osfhandle((intptr_t)std_err, _O_TEXT);
|
|
|
|
int std_out_fd = _open_osfhandle((intptr_t)std_out, _O_TEXT);
|
|
|
|
|
|
|
|
if (std_err_fd > 0)
|
|
|
|
*stderr = *_fdopen(std_err_fd, "w");
|
|
|
|
|
|
|
|
if (std_out_fd > 0)
|
|
|
|
*stdout = *_fdopen(std_out_fd, "w");
|
|
|
|
}
|
|
|
|
|
2019-05-09 12:13:31 +00:00
|
|
|
int result = 0;
|
2019-05-21 07:52:58 +00:00
|
|
|
app.hInst = hInstance;
|
2019-05-09 12:13:31 +00:00
|
|
|
|
2019-05-16 23:26:42 +00:00
|
|
|
char tempPath[MAX_PATH+1];
|
|
|
|
GetTempPathA(sizeof(tempPath), tempPath);
|
|
|
|
int len = snprintf(NULL, 0, "%slooking-glass-host.txt", tempPath);
|
|
|
|
char * logFilePath = malloc(len + 1);
|
|
|
|
sprintf(logFilePath, "%slooking-glass-host.txt", tempPath);
|
|
|
|
|
2019-05-10 09:48:38 +00:00
|
|
|
struct Option options[] =
|
|
|
|
{
|
2019-05-16 23:26:42 +00:00
|
|
|
{
|
|
|
|
.module = "os",
|
|
|
|
.name = "logFile",
|
|
|
|
.description = "The log file to write to",
|
|
|
|
.type = OPTION_TYPE_STRING,
|
|
|
|
.value.x_string = logFilePath
|
2019-05-10 09:48:38 +00:00
|
|
|
},
|
|
|
|
{0}
|
|
|
|
};
|
|
|
|
|
|
|
|
option_register(options);
|
2019-05-16 23:26:42 +00:00
|
|
|
free(logFilePath);
|
2019-05-10 09:48:38 +00:00
|
|
|
|
2019-05-09 09:30:09 +00:00
|
|
|
// convert the command line to the standard argc and argv
|
2019-05-09 09:32:19 +00:00
|
|
|
LPWSTR * wargv = CommandLineToArgvW(GetCommandLineW(), &app.argc);
|
2019-05-09 09:30:09 +00:00
|
|
|
app.argv = malloc(sizeof(char *) * app.argc);
|
|
|
|
for(int i = 0; i < app.argc; ++i)
|
|
|
|
{
|
|
|
|
const size_t s = (wcslen(wargv[i])+1) * 2;
|
|
|
|
app.argv[i] = malloc(s);
|
2019-06-12 05:31:18 +00:00
|
|
|
wcstombs(app.argv[i], wargv[i], s);
|
2019-05-09 09:30:09 +00:00
|
|
|
}
|
|
|
|
LocalFree(wargv);
|
|
|
|
|
|
|
|
GetModuleFileName(NULL, app.executable, sizeof(app.executable));
|
2019-04-11 07:15:17 +00:00
|
|
|
|
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";
|
2019-05-21 07:52:58 +00:00
|
|
|
wx.hIcon = LoadIcon(NULL, IDI_APPLICATION);
|
|
|
|
wx.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
|
|
|
|
wx.hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
|
|
wx.hbrBackground = (HBRUSH)COLOR_APPWORKSPACE;
|
2019-03-01 04:45:46 +00:00
|
|
|
if (!RegisterClassEx(&wx))
|
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to register message window class");
|
|
|
|
result = -1;
|
2019-05-09 12:13:31 +00:00
|
|
|
goto finish;
|
2019-03-01 04:45:46 +00:00
|
|
|
}
|
2019-05-09 09:30:09 +00:00
|
|
|
app.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-05-21 07:52:58 +00:00
|
|
|
app.trayMenu = CreatePopupMenu();
|
|
|
|
AppendMenu(app.trayMenu, MF_STRING , ID_MENU_OPEN_LOG, "Open Log File");
|
|
|
|
AppendMenu(app.trayMenu, MF_SEPARATOR, 0 , NULL );
|
|
|
|
AppendMenu(app.trayMenu, MF_STRING , ID_MENU_EXIT , "Exit" );
|
|
|
|
|
2019-05-09 09:30:09 +00:00
|
|
|
// create the application thread
|
2020-01-02 11:21:42 +00:00
|
|
|
LGThread * thread;
|
|
|
|
if (!lgCreateThread("appThread", appThread, NULL, &thread))
|
2019-03-01 04:45:46 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to create the main application thread");
|
|
|
|
result = -1;
|
2019-05-09 12:13:31 +00:00
|
|
|
goto finish;
|
2019-03-01 04:45:46 +00:00
|
|
|
}
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2019-02-28 09:50:22 +00:00
|
|
|
}
|
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
shutdown:
|
2019-05-21 07:52:58 +00:00
|
|
|
DestroyMenu(app.trayMenu);
|
2019-03-02 09:31:33 +00:00
|
|
|
app_quit();
|
2020-01-02 11:21:42 +00:00
|
|
|
if (!lgJoinThread(thread, &result))
|
2019-03-01 04:45:46 +00:00
|
|
|
{
|
|
|
|
DEBUG_ERROR("Failed to join the main application thread");
|
|
|
|
result = -1;
|
|
|
|
}
|
2019-05-09 12:13:31 +00:00
|
|
|
|
2019-03-01 04:45:46 +00:00
|
|
|
finish:
|
2019-05-09 09:30:09 +00:00
|
|
|
|
|
|
|
for(int i = 0; i < app.argc; ++i)
|
|
|
|
free(app.argv[i]);
|
|
|
|
free(app.argv);
|
|
|
|
|
2019-02-28 08:20:35 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-05-09 12:06:58 +00:00
|
|
|
bool app_init()
|
|
|
|
{
|
2019-05-16 23:26:42 +00:00
|
|
|
const char * logFile = option_get_string("os", "logFile" );
|
|
|
|
|
|
|
|
// redirect stderr to a file
|
|
|
|
if (logFile && strcmp(logFile, "stderr") != 0)
|
|
|
|
freopen(logFile, "a", stderr);
|
|
|
|
|
|
|
|
// always flush stderr
|
|
|
|
setbuf(stderr, NULL);
|
2019-05-10 09:48:38 +00:00
|
|
|
|
2019-12-15 05:20:07 +00:00
|
|
|
// Increase the timer resolution
|
|
|
|
ZwSetTimerResolution = (ZwSetTimerResolution_t)GetProcAddress(GetModuleHandle("ntdll.dll"), "ZwSetTimerResolution");
|
|
|
|
if (ZwSetTimerResolution)
|
|
|
|
{
|
|
|
|
ULONG actualResolution;
|
|
|
|
ZwSetTimerResolution(1, true, &actualResolution);
|
|
|
|
DEBUG_INFO("System timer resolution: %.2f ns", (float)actualResolution / 100.0f);
|
|
|
|
}
|
|
|
|
|
2019-12-17 03:59:58 +00:00
|
|
|
// get the performance frequency for spinlocks
|
|
|
|
QueryPerformanceFrequency(&app.perfFreq);
|
|
|
|
|
2019-05-09 12:06:58 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-04-11 07:15:17 +00:00
|
|
|
const char * os_getExecutable()
|
|
|
|
{
|
2019-05-09 09:30:09 +00:00
|
|
|
return app.executable;
|
2019-02-28 08:20:35 +00:00
|
|
|
}
|