[host] windows: re-implement open log safely

Instead of doing ShellExecute from the service, we instead get the token
of the currently logged in user, and do CreateProcessAsUserA to run
notepad with that token. This should be safe.
This commit is contained in:
Quantum 2021-03-18 00:12:40 -04:00 committed by Geoffrey McRae
parent a089c4ea32
commit 5bfb33c739
3 changed files with 149 additions and 56 deletions

View file

@ -27,6 +27,8 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include <fcntl.h> #include <fcntl.h>
#include <powrprof.h> #include <powrprof.h>
#include <ntstatus.h> #include <ntstatus.h>
#include <wtsapi32.h>
#include <userenv.h>
#include "interface/platform.h" #include "interface/platform.h"
#include "common/debug.h" #include "common/debug.h"
@ -71,6 +73,34 @@ static ZwSetTimerResolution_t ZwSetTimerResolution = NULL;
typedef WINBOOL WINAPI (*PChangeWindowMessageFilterEx)(HWND hwnd, UINT message, DWORD action, void * pChangeFilterStruct); typedef WINBOOL WINAPI (*PChangeWindowMessageFilterEx)(HWND hwnd, UINT message, DWORD action, void * pChangeFilterStruct);
PChangeWindowMessageFilterEx _ChangeWindowMessageFilterEx = NULL; PChangeWindowMessageFilterEx _ChangeWindowMessageFilterEx = NULL;
CreateProcessAsUserA_t f_CreateProcessAsUserA = NULL;
bool windowsSetupAPI(void)
{
/* first look in kernel32.dll */
HMODULE mod;
mod = GetModuleHandleA("kernel32.dll");
if (mod)
{
f_CreateProcessAsUserA = (CreateProcessAsUserA_t)
GetProcAddress(mod, "CreateProcessAsUserA");
if (f_CreateProcessAsUserA)
return true;
}
mod = GetModuleHandleA("advapi32.dll");
if (mod)
{
f_CreateProcessAsUserA = (CreateProcessAsUserA_t)
GetProcAddress(mod, "CreateProcessAsUserA");
if (f_CreateProcessAsUserA)
return true;
}
return false;
}
static void RegisterTrayIcon(void) static void RegisterTrayIcon(void)
{ {
// register our TrayIcon // register our TrayIcon
@ -86,6 +116,99 @@ static void RegisterTrayIcon(void)
Shell_NotifyIcon(NIM_ADD, &app.iconData); Shell_NotifyIcon(NIM_ADD, &app.iconData);
} }
// This function executes notepad as the logged in user, and therefore is secure to use.
static bool OpenLogFile(const char * logFile)
{
bool result = false;
DWORD console = WTSGetActiveConsoleSessionId();
if (console == 0xFFFFFFFF)
{
DEBUG_WINERROR("Failed to get active console session ID", GetLastError());
return false;
}
WTS_CONNECTSTATE_CLASS * state;
DWORD size;
if (!WTSQuerySessionInformationA(WTS_CURRENT_SERVER_HANDLE, console, WTSConnectState,
(LPSTR *) &state, &size))
{
DEBUG_WINERROR("Failed to get session information", GetLastError());
return false;
}
if (*state != WTSActive)
{
DEBUG_WINERROR("Will not open log file because user is not logged in", GetLastError());
WTSFreeMemory(state);
return false;
}
WTSFreeMemory(state);
char system32[MAX_PATH];
if (!GetSystemDirectoryA(system32, MAX_PATH))
{
DEBUG_WINERROR("Failed to get system directory", GetLastError());
return false;
}
if (!f_CreateProcessAsUserA && !windowsSetupAPI())
{
DEBUG_WINERROR("Failed to get CreateProcessAsUserA", GetLastError());
return false;
}
HANDLE hToken;
if (!WTSQueryUserToken(console, &hToken))
{
DEBUG_WINERROR("Failed to get active console session user token", GetLastError());
return false;
}
LPVOID env;
if (!CreateEnvironmentBlock(&env, hToken, FALSE))
{
DEBUG_WINERROR("Failed to create environment", GetLastError());
goto fail_token;
}
char notepad[MAX_PATH];
PathCombineA(notepad, system32, "notepad.exe");
char cmdline[MAX_PATH + 10];
snprintf(cmdline, sizeof(cmdline), "notepad \"%s\"", logFile);
STARTUPINFO si = { .cb = sizeof(STARTUPINFO) };
PROCESS_INFORMATION pi = {0};
if (!f_CreateProcessAsUserA(
hToken,
notepad,
cmdline,
NULL,
NULL,
FALSE,
CREATE_UNICODE_ENVIRONMENT,
env,
os_getDataPath(),
&si,
&pi
))
{
DEBUG_WINERROR("Failed to open log file", GetLastError());
goto fail_env;
}
result = true;
CloseHandle(pi.hThread);
CloseHandle(pi.hProcess);
fail_env:
DestroyEnvironmentBlock(env);
fail_token:
CloseHandle(hToken);
return result;
}
LRESULT CALLBACK DummyWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) LRESULT CALLBACK DummyWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{ {
switch(msg) switch(msg)
@ -123,16 +246,10 @@ LRESULT CALLBACK DummyWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
const char * logFile = option_get_string("os", "logFile"); const char * logFile = option_get_string("os", "logFile");
if (strcmp(logFile, "stderr") == 0) if (strcmp(logFile, "stderr") == 0)
DEBUG_INFO("Ignoring request to open the logFile, logging to stderr"); DEBUG_INFO("Ignoring request to open the logFile, logging to stderr");
else else if (!OpenLogFile(logFile))
{
/* If LG is running as SYSTEM, ShellExecute would launch a process
* as the SYSTEM user also, for security we will just show the file
* location instead */
//ShellExecute(NULL, NULL, logFile, NULL, NULL, SW_SHOWNORMAL);
MessageBoxA(hwnd, logFile, "Log File Location", MB_OK | MB_ICONINFORMATION); MessageBoxA(hwnd, logFile, "Log File Location", MB_OK | MB_ICONINFORMATION);
} }
} }
}
break; break;
} }
@ -295,7 +412,7 @@ int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine
MessageHWND = app.messageWnd; MessageHWND = app.messageWnd;
app.trayMenu = CreatePopupMenu(); app.trayMenu = CreatePopupMenu();
AppendMenu(app.trayMenu, MF_STRING , ID_MENU_SHOW_LOG, "Log File Location"); AppendMenu(app.trayMenu, MF_STRING , ID_MENU_SHOW_LOG, "Open Log File");
AppendMenu(app.trayMenu, MF_SEPARATOR, 0 , NULL ); AppendMenu(app.trayMenu, MF_SEPARATOR, 0 , NULL );
AppendMenu(app.trayMenu, MF_STRING , ID_MENU_EXIT , "Exit" ); AppendMenu(app.trayMenu, MF_STRING , ID_MENU_EXIT , "Exit" );

View file

@ -17,8 +17,27 @@ this program; if not, write to the Free Software Foundation, Inc., 59 Temple
Place, Suite 330, Boston, MA 02111-1307 USA Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
#include <stdbool.h>
#include <windows.h> #include <windows.h>
/*
* Windows 10 provides this API via kernel32.dll as well as advapi32.dll and
* mingw opts for linking against the kernel32.dll version which is fine
* provided you don't intend to run this on earlier versions of windows. As such
* we need to lookup this method at runtime. */
typedef WINBOOL WINAPI (*CreateProcessAsUserA_t)(HANDLE hToken,
LPCSTR lpApplicationName,
LPSTR lpCommandLine,
LPSECURITY_ATTRIBUTES lpProcessAttributes,
LPSECURITY_ATTRIBUTES lpThreadAttributes,
WINBOOL bInheritHandles,
DWORD dwCreationFlags,
LPVOID lpEnvironment,
LPCSTR lpCurrentDirectory,
LPSTARTUPINFOA lpStartupInfo,
LPPROCESS_INFORMATION lpProcessInformation);
extern CreateProcessAsUserA_t f_CreateProcessAsUserA;
#define WM_CALL_FUNCTION (WM_USER+1) #define WM_CALL_FUNCTION (WM_USER+1)
#define WM_TRAYICON (WM_USER+2) #define WM_TRAYICON (WM_USER+2)
@ -30,5 +49,6 @@ struct MSG_CALL_FUNCTION
LPARAM lParam; LPARAM lParam;
}; };
bool windowsSetupAPI(void);
const char *getSystemLogDirectory(void); const char *getSystemLogDirectory(void);
LRESULT sendAppMessage(UINT Msg, WPARAM wParam, LPARAM lParam); LRESULT sendAppMessage(UINT Msg, WPARAM wParam, LPARAM lParam);

View file

@ -44,24 +44,6 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#define FAIL_MAX_RETRIES 5 #define FAIL_MAX_RETRIES 5
#define FAIL_RETRY_INIT_INTERVAL 1000 #define FAIL_RETRY_INIT_INTERVAL 1000
/*
* Windows 10 provides this API via kernel32.dll as well as advapi32.dll and
* mingw opts for linking against the kernel32.dll version which is fine
* provided you don't intend to run this on earlier versions of windows. As such
* we need to lookup this method at runtime. */
typedef WINBOOL WINAPI (*CreateProcessAsUserA_t)(HANDLE hToken,
LPCSTR lpApplicationName,
LPSTR lpCommandLine,
LPSECURITY_ATTRIBUTES lpProcessAttributes,
LPSECURITY_ATTRIBUTES lpThreadAttributes,
WINBOOL bInheritHandles,
DWORD dwCreationFlags,
LPVOID lpEnvironment,
LPCSTR lpCurrentDirectory,
LPSTARTUPINFOA lpStartupInfo,
LPPROCESS_INFORMATION lpProcessInformation);
static CreateProcessAsUserA_t f_CreateProcessAsUserA = NULL;
struct Service struct Service
{ {
FILE * logFile; FILE * logFile;
@ -90,32 +72,6 @@ void doLogReal(const char * fmt, ...)
#define doLog(fmt, ...) doLogReal("[%s] " fmt, currentTime(), ##__VA_ARGS__) #define doLog(fmt, ...) doLogReal("[%s] " fmt, currentTime(), ##__VA_ARGS__)
static bool setupAPI(void)
{
/* first look in kernel32.dll */
HMODULE mod;
mod = GetModuleHandleA("kernel32.dll");
if (mod)
{
f_CreateProcessAsUserA = (CreateProcessAsUserA_t)
GetProcAddress(mod, "CreateProcessAsUserA");
if (f_CreateProcessAsUserA)
return true;
}
mod = GetModuleHandleA("advapi32.dll");
if (mod)
{
f_CreateProcessAsUserA = (CreateProcessAsUserA_t)
GetProcAddress(mod, "CreateProcessAsUserA");
if (f_CreateProcessAsUserA)
return true;
}
return false;
}
static void setupLogging(void) static void setupLogging(void)
{ {
char logFilePath[MAX_PATH]; char logFilePath[MAX_PATH];
@ -260,9 +216,9 @@ void Launch(void)
service.process = NULL; service.process = NULL;
} }
if (!setupAPI()) if (!windowsSetupAPI())
{ {
doLog("setupAPI failed\n"); doLog("windowsSetupAPI failed\n");
return; return;
} }