#include #include #include #include #include #include #include #include #include // NOLINT(build/include_order) #include #include void flog(char const *fmt, ...) { FILE *h = fopen("C:\\Windows\\temp\\test_rustdesk.log", "at"); if (!h) return; va_list arg; va_start(arg, fmt); vfprintf(h, fmt, arg); va_end(arg); fclose(h); } // ultravnc has rdp support // https://github.com/veyon/ultravnc/blob/master/winvnc/winvnc/service.cpp // https://github.com/TigerVNC/tigervnc/blob/master/win/winvnc/VNCServerService.cxx // https://blog.csdn.net/MA540213/article/details/84638264 DWORD GetLogonPid(DWORD dwSessionId, BOOL as_user) { DWORD dwLogonPid = 0; HANDLE hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0); if (hSnap != INVALID_HANDLE_VALUE) { PROCESSENTRY32W procEntry; procEntry.dwSize = sizeof procEntry; if (Process32FirstW(hSnap, &procEntry)) do { DWORD dwLogonSessionId = 0; if (_wcsicmp(procEntry.szExeFile, as_user ? L"explorer.exe" : L"winlogon.exe") == 0 && ProcessIdToSessionId(procEntry.th32ProcessID, &dwLogonSessionId) && dwLogonSessionId == dwSessionId) { dwLogonPid = procEntry.th32ProcessID; break; } } while (Process32NextW(hSnap, &procEntry)); CloseHandle(hSnap); } return dwLogonPid; } // START the app as system extern "C" { // if should try WTSQueryUserToken? // https://stackoverflow.com/questions/7285666/example-code-a-service-calls-createprocessasuser-i-want-the-process-to-run-in BOOL GetSessionUserTokenWin(OUT LPHANDLE lphUserToken, DWORD dwSessionId, BOOL as_user) { BOOL bResult = FALSE; DWORD Id = GetLogonPid(dwSessionId, as_user); if (HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, Id)) { bResult = OpenProcessToken(hProcess, TOKEN_ALL_ACCESS, lphUserToken); CloseHandle(hProcess); } return bResult; } bool is_windows_server() { return IsWindowsServer(); } HANDLE LaunchProcessWin(LPCWSTR cmd, DWORD dwSessionId, BOOL as_user) { HANDLE hProcess = NULL; HANDLE hToken = NULL; if (GetSessionUserTokenWin(&hToken, dwSessionId, as_user)) { STARTUPINFOW si; ZeroMemory(&si, sizeof si); si.cb = sizeof si; si.dwFlags = STARTF_USESHOWWINDOW; wchar_t buf[MAX_PATH]; wcscpy_s(buf, sizeof(buf), cmd); PROCESS_INFORMATION pi; LPVOID lpEnvironment = NULL; DWORD dwCreationFlags = DETACHED_PROCESS; if (as_user) { CreateEnvironmentBlock(&lpEnvironment, // Environment block hToken, // New token TRUE); // Inheritance } if (lpEnvironment) { dwCreationFlags |= CREATE_UNICODE_ENVIRONMENT; } if (CreateProcessAsUserW(hToken, NULL, buf, NULL, NULL, FALSE, dwCreationFlags, lpEnvironment, NULL, &si, &pi)) { CloseHandle(pi.hThread); hProcess = pi.hProcess; } CloseHandle(hToken); if (lpEnvironment) DestroyEnvironmentBlock(lpEnvironment); } return hProcess; } // Switch the current thread to the specified desktop static bool switchToDesktop(HDESK desktop) { HDESK old_desktop = GetThreadDesktop(GetCurrentThreadId()); if (!SetThreadDesktop(desktop)) { return false; } if (!CloseDesktop(old_desktop)) { // } return true; } // https://github.com/TigerVNC/tigervnc/blob/8c6c584377feba0e3b99eecb3ef33b28cee318cb/win/rfb_win32/Service.cxx // Determine whether the thread's current desktop is the input one BOOL inputDesktopSelected() { HDESK current = GetThreadDesktop(GetCurrentThreadId()); HDESK input = OpenInputDesktop(0, FALSE, DESKTOP_CREATEMENU | DESKTOP_CREATEWINDOW | DESKTOP_ENUMERATE | DESKTOP_HOOKCONTROL | DESKTOP_WRITEOBJECTS | DESKTOP_READOBJECTS | DESKTOP_SWITCHDESKTOP | GENERIC_WRITE); if (!input) { return FALSE; } DWORD size; char currentname[256]; char inputname[256]; if (!GetUserObjectInformation(current, UOI_NAME, currentname, sizeof(currentname), &size)) { CloseDesktop(input); return FALSE; } if (!GetUserObjectInformation(input, UOI_NAME, inputname, sizeof(inputname), &size)) { CloseDesktop(input); return FALSE; } CloseDesktop(input); // flog("%s %s\n", currentname, inputname); return strcmp(currentname, inputname) == 0 ? TRUE : FALSE; } // Switch the current thread into the input desktop bool selectInputDesktop() { // - Open the input desktop HDESK desktop = OpenInputDesktop(0, FALSE, DESKTOP_CREATEMENU | DESKTOP_CREATEWINDOW | DESKTOP_ENUMERATE | DESKTOP_HOOKCONTROL | DESKTOP_WRITEOBJECTS | DESKTOP_READOBJECTS | DESKTOP_SWITCHDESKTOP | GENERIC_WRITE); if (!desktop) { return false; } // - Switch into it if (!switchToDesktop(desktop)) { CloseDesktop(desktop); return false; } // *** DWORD size = 256; char currentname[256]; if (GetUserObjectInformation(desktop, UOI_NAME, currentname, 256, &size)) { // } return true; } int handleMask(uint8_t *rwbuffer, const uint8_t *mask, int width, int height, int bmWidthBytes, int bmHeight) { auto andMask = mask; auto andMaskSize = bmWidthBytes * bmHeight; auto offset = height * bmWidthBytes; auto xorMask = mask + offset; auto xorMaskSize = andMaskSize - offset; int doOutline = 0; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { int byte = y * bmWidthBytes + x / 8; int bit = 7 - x % 8; if (byte < andMaskSize && !(andMask[byte] & (1 << bit))) { // Valid pixel, so make it opaque rwbuffer[3] = 0xff; // Black or white? if (xorMask[byte] & (1 << bit)) rwbuffer[0] = rwbuffer[1] = rwbuffer[2] = 0xff; else rwbuffer[0] = rwbuffer[1] = rwbuffer[2] = 0; } else if (byte < xorMaskSize && xorMask[byte] & (1 << bit)) { // Replace any XORed pixels with black, because RFB doesn't support // XORing of cursors. XORing is used for the I-beam cursor, which is most // often used over a white background, but also sometimes over a black // background. We set the XOR'd pixels to black, then draw a white outline // around the whole cursor. rwbuffer[0] = rwbuffer[1] = rwbuffer[2] = 0; rwbuffer[3] = 0xff; doOutline = 1; } else { // Transparent pixel rwbuffer[0] = rwbuffer[1] = rwbuffer[2] = rwbuffer[3] = 0; } rwbuffer += 4; } } return doOutline; } void drawOutline(uint8_t *out0, const uint8_t *in0, int width, int height, int out0_size) { auto in = in0; auto out0_end = out0 + out0_size; auto offset = width * 4 + 4; auto out = out0 + offset; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { // Visible pixel? if (in[3] > 0) { auto n = 4 * 3; auto p = out - (width + 2) * 4 - 4; // Outline above... if (p >= out0 && p + n <= out0_end) memset(p, 0xff, n); // ...besides... p = out - 4; if (p + n <= out0_end) memset(p, 0xff, n); // ...and above p = out + (width + 2) * 4 - 4; if (p + n <= out0_end) memset(p, 0xff, n); } in += 4; out += 4; } // outline is slightly larger out += 2 * 4; } // Pass 2, overwrite with actual cursor in = in0; out = out0 + offset; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { if (in[3] > 0 && out + 4 <= out0_end) memcpy(out, in, 4); in += 4; out += 4; } out += 2 * 4; } } int ffi(unsigned v) { static const int MultiplyDeBruijnBitPosition[32] = { 0, 1, 28, 2, 29, 14, 24, 3, 30, 22, 20, 15, 25, 17, 4, 8, 31, 27, 13, 23, 21, 19, 16, 7, 26, 12, 18, 6, 11, 5, 10, 9}; return MultiplyDeBruijnBitPosition[((uint32_t)((v & -v) * 0x077CB531U)) >> 27]; } int get_di_bits(uint8_t *out, HDC dc, HBITMAP hbmColor, int width, int height) { BITMAPV5HEADER bi; memset(&bi, 0, sizeof(BITMAPV5HEADER)); bi.bV5Size = sizeof(BITMAPV5HEADER); bi.bV5Width = width; bi.bV5Height = -height; // Negative for top-down bi.bV5Planes = 1; bi.bV5BitCount = 32; bi.bV5Compression = BI_BITFIELDS; bi.bV5RedMask = 0x000000FF; bi.bV5GreenMask = 0x0000FF00; bi.bV5BlueMask = 0x00FF0000; bi.bV5AlphaMask = 0xFF000000; if (!GetDIBits(dc, hbmColor, 0, height, out, (LPBITMAPINFO)&bi, DIB_RGB_COLORS)) return 1; // We may not get the RGBA order we want, so shuffle things around int ridx, gidx, bidx, aidx; ridx = ffi(bi.bV5RedMask) / 8; gidx = ffi(bi.bV5GreenMask) / 8; bidx = ffi(bi.bV5BlueMask) / 8; // Usually not set properly aidx = 6 - ridx - gidx - bidx; if ((bi.bV5RedMask != ((unsigned)0xff << ridx * 8)) || (bi.bV5GreenMask != ((unsigned)0xff << gidx * 8)) || (bi.bV5BlueMask != ((unsigned)0xff << bidx * 8))) return 1; auto rwbuffer = out; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { uint8_t r, g, b, a; r = rwbuffer[ridx]; g = rwbuffer[gidx]; b = rwbuffer[bidx]; a = rwbuffer[aidx]; rwbuffer[0] = r; rwbuffer[1] = g; rwbuffer[2] = b; rwbuffer[3] = a; rwbuffer += 4; } } return 0; } void blank_screen(BOOL set) { if (set) { SendMessage(HWND_BROADCAST, WM_SYSCOMMAND, SC_MONITORPOWER, (LPARAM)2); } else { SendMessage(HWND_BROADCAST, WM_SYSCOMMAND, SC_MONITORPOWER, (LPARAM)-1); } } void AddRecentDocument(PCWSTR path) { SHAddToRecentDocs(SHARD_PATHW, path); } DWORD get_current_session(BOOL include_rdp) { auto rdp_or_console = WTSGetActiveConsoleSessionId(); if (!include_rdp) return rdp_or_console; PWTS_SESSION_INFOA pInfos; DWORD count; auto rdp = "rdp"; auto nrdp = strlen(rdp); if (WTSEnumerateSessionsA(WTS_CURRENT_SERVER_HANDLE, NULL, 1, &pInfos, &count)) { for (DWORD i = 0; i < count; i++) { auto info = pInfos[i]; if (info.State == WTSActive) { if (info.pWinStationName == NULL) continue; if (!stricmp(info.pWinStationName, "console")) { return info.SessionId; } if (!strnicmp(info.pWinStationName, rdp, nrdp)) { rdp_or_console = info.SessionId; } } } WTSFreeMemory(pInfos); } return rdp_or_console; } uint32_t get_active_user(PWSTR bufin, uint32_t nin, BOOL rdp) { uint32_t nout = 0; auto id = get_current_session(rdp); PWSTR buf = NULL; DWORD n = 0; if (WTSQuerySessionInformationW(WTS_CURRENT_SERVER_HANDLE, id, WTSUserName, &buf, &n)) { if (buf) { nout = min(nin, n); memcpy(bufin, buf, nout); WTSFreeMemory(buf); } } return nout; } BOOL has_rdp_service() { PWTS_SESSION_INFOA pInfos; DWORD count; auto rdp = "rdp"; auto nrdp = strlen(rdp); auto rdp_or_console = WTSGetActiveConsoleSessionId(); if (WTSEnumerateSessionsA(WTS_CURRENT_SERVER_HANDLE, NULL, 1, &pInfos, &count)) { for (DWORD i = 0; i < count; i++) { auto info = pInfos[i]; if (!strnicmp(info.pWinStationName, rdp, nrdp)) { return TRUE; } } WTSFreeMemory(pInfos); } return FALSE; } } // end of extern "C" // below copied from https://github.com/TigerVNC/tigervnc/blob/master/vncviewer/win32.c extern "C" { static HANDLE thread; static DWORD thread_id; static HHOOK hook = 0; static HWND target_wnd = 0; static HWND default_hook_wnd = 0; static bool win_down = false; static bool stop_system_key_propagate = false; bool is_win_down() { return win_down; } #define ARRAY_SIZE(a) (sizeof(a) / sizeof(*a)) static int is_system_hotkey(int vkCode, WPARAM wParam) { switch (vkCode) { case VK_LWIN: case VK_RWIN: win_down = wParam == WM_KEYDOWN; case VK_SNAPSHOT: return 1; case VK_TAB: if (GetAsyncKeyState(VK_MENU) & 0x8000) return 1; case VK_ESCAPE: if (GetAsyncKeyState(VK_MENU) & 0x8000) return 1; if (GetAsyncKeyState(VK_CONTROL) & 0x8000) return 1; } return 0; } static LRESULT CALLBACK keyboard_hook(int nCode, WPARAM wParam, LPARAM lParam) { if (nCode >= 0) { KBDLLHOOKSTRUCT *msgInfo = (KBDLLHOOKSTRUCT *)lParam; // Grabbing everything seems to mess up some keyboard state that // FLTK relies on, so just grab the keys that we normally cannot. if (stop_system_key_propagate && is_system_hotkey(msgInfo->vkCode, wParam)) { PostMessage(target_wnd, wParam, msgInfo->vkCode, (msgInfo->scanCode & 0xff) << 16 | (msgInfo->flags & 0xff) << 24); return 1; } } return CallNextHookEx(hook, nCode, wParam, lParam); } static DWORD WINAPI keyboard_thread(LPVOID data) { MSG msg; target_wnd = (HWND)data; // Make sure a message queue is created PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE | PM_NOYIELD); hook = SetWindowsHookEx(WH_KEYBOARD_LL, keyboard_hook, GetModuleHandle(0), 0); // If something goes wrong then there is not much we can do. // Just sit around and wait for WM_QUIT... while (GetMessage(&msg, NULL, 0, 0)) ; if (hook) UnhookWindowsHookEx(hook); target_wnd = 0; return 0; } int win32_enable_lowlevel_keyboard(HWND hwnd) { if (!default_hook_wnd) { default_hook_wnd = hwnd; } if (!hwnd) { hwnd = default_hook_wnd; } // Only one target at a time for now if (thread != NULL) { if (hwnd == target_wnd) return 0; return 1; } // We create a separate thread as it is crucial that hooks are processed // in a timely manner. thread = CreateThread(NULL, 0, keyboard_thread, hwnd, 0, &thread_id); if (thread == NULL) return 1; return 0; } void win32_disable_lowlevel_keyboard(HWND hwnd) { if (!hwnd) { hwnd = default_hook_wnd; } if (hwnd != target_wnd) return; PostThreadMessage(thread_id, WM_QUIT, 0, 0); CloseHandle(thread); thread = NULL; } void win_stop_system_key_propagate(bool v) { stop_system_key_propagate = v; } // https://stackoverflow.com/questions/4023586/correct-way-to-find-out-if-a-service-is-running-as-the-system-user BOOL is_local_system() { HANDLE hToken; UCHAR bTokenUser[sizeof(TOKEN_USER) + 8 + 4 * SID_MAX_SUB_AUTHORITIES]; PTOKEN_USER pTokenUser = (PTOKEN_USER)bTokenUser; ULONG cbTokenUser; SID_IDENTIFIER_AUTHORITY siaNT = SECURITY_NT_AUTHORITY; PSID pSystemSid; BOOL bSystem; // open process token if (!OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &hToken)) return FALSE; // retrieve user SID if (!GetTokenInformation(hToken, TokenUser, pTokenUser, sizeof(bTokenUser), &cbTokenUser)) { CloseHandle(hToken); return FALSE; } CloseHandle(hToken); // allocate LocalSystem well-known SID if (!AllocateAndInitializeSid(&siaNT, 1, SECURITY_LOCAL_SYSTEM_RID, 0, 0, 0, 0, 0, 0, 0, &pSystemSid)) return FALSE; // compare the user SID from the token with the LocalSystem SID bSystem = EqualSid(pTokenUser->User.Sid, pSystemSid); FreeSid(pSystemSid); return bSystem; } } // end of extern "C"