/****************************************************************************** Copyright (C) 2023 by Lain Bailey 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, see . ******************************************************************************/ #include "util/windows/win-registry.h" #include "util/windows/win-version.h" #include "util/platform.h" #include "util/dstr.h" #include "obs.h" #include "obs-internal.h" #include #include #include static uint32_t win_ver = 0; static uint32_t win_build = 0; const char *get_module_extension(void) { return ".dll"; } static const char *module_bin[] = {"../../obs-plugins/64bit"}; static const char *module_data[] = {"../../data/obs-plugins/%module%"}; static const int module_patterns_size = sizeof(module_bin) / sizeof(module_bin[0]); void add_default_module_paths(void) { for (int i = 0; i < module_patterns_size; i++) obs_add_module_path(module_bin[i], module_data[i]); } /* on windows, points to [base directory]/data/libobs */ char *find_libobs_data_file(const char *file) { struct dstr path; dstr_init(&path); if (check_path(file, "../../data/libobs/", &path)) return path.array; dstr_free(&path); return NULL; } static void log_processor_info(void) { HKEY key; wchar_t data[1024]; char *str = NULL; DWORD size, speed; LSTATUS status; memset(data, 0, sizeof(data)); status = RegOpenKeyW( HKEY_LOCAL_MACHINE, L"HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0", &key); if (status != ERROR_SUCCESS) return; size = sizeof(data); status = RegQueryValueExW(key, L"ProcessorNameString", NULL, NULL, (LPBYTE)data, &size); if (status == ERROR_SUCCESS) { os_wcs_to_utf8_ptr(data, 0, &str); blog(LOG_INFO, "CPU Name: %s", str); bfree(str); } size = sizeof(speed); status = RegQueryValueExW(key, L"~MHz", NULL, NULL, (LPBYTE)&speed, &size); if (status == ERROR_SUCCESS) blog(LOG_INFO, "CPU Speed: %ldMHz", speed); RegCloseKey(key); } static void log_processor_cores(void) { blog(LOG_INFO, "Physical Cores: %d, Logical Cores: %d", os_get_physical_cores(), os_get_logical_cores()); } static void log_emulation_status(void) { if (os_get_emulation_status()) { blog(LOG_WARNING, "Windows ARM64: Running with x64 emulation"); } } static void log_available_memory(void) { MEMORYSTATUSEX ms; ms.dwLength = sizeof(ms); GlobalMemoryStatusEx(&ms); #ifdef _WIN64 const char *note = ""; #else const char *note = " (NOTE: 32bit programs cannot use more than 3gb)"; #endif blog(LOG_INFO, "Physical Memory: %luMB Total, %luMB Free%s", (DWORD)(ms.ullTotalPhys / 1048576), (DWORD)(ms.ullAvailPhys / 1048576), note); } static void log_lenovo_vantage(void) { SC_HANDLE manager = OpenSCManager(NULL, NULL, SC_MANAGER_CONNECT); if (!manager) return; SC_HANDLE service = OpenService(manager, L"FBNetFilter", SERVICE_QUERY_STATUS); if (service) { blog(LOG_WARNING, "Lenovo Vantage / Legion Edge is installed. The \"Network Boost\" " "feature must be disabled when streaming with OBS."); CloseServiceHandle(service); } CloseServiceHandle(manager); } static void log_conflicting_software(void) { log_lenovo_vantage(); } extern const char *get_win_release_id(); static void log_windows_version(void) { struct win_version_info ver; get_win_ver(&ver); const char *release_id = get_win_release_id(); bool b64 = is_64_bit_windows(); const char *windows_bitness = b64 ? "64" : "32"; bool arm64 = is_arm64_windows(); const char *arm64_windows = arm64 ? "ARM " : ""; blog(LOG_INFO, "Windows Version: %d.%d Build %d (release: %s; revision: %d; %s%s-bit)", ver.major, ver.minor, ver.build, release_id, ver.revis, arm64_windows, windows_bitness); } static void log_admin_status(void) { SID_IDENTIFIER_AUTHORITY auth = SECURITY_NT_AUTHORITY; PSID admin_group; BOOL success; success = AllocateAndInitializeSid(&auth, 2, SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &admin_group); if (success) { if (!CheckTokenMembership(NULL, admin_group, &success)) success = false; FreeSid(admin_group); } blog(LOG_INFO, "Running as administrator: %s", success ? "true" : "false"); } #define WIN10_GAME_BAR_REG_KEY \ L"Software\\Microsoft\\Windows\\CurrentVersion\\GameDVR" #define WIN10_GAME_DVR_POLICY_REG_KEY \ L"SOFTWARE\\Policies\\Microsoft\\Windows\\GameDVR" #define WIN10_GAME_DVR_REG_KEY L"System\\GameConfigStore" #define WIN10_GAME_MODE_REG_KEY L"Software\\Microsoft\\GameBar" static void log_gaming_features(void) { if (win_ver < 0xA00) return; struct reg_dword game_bar_enabled; struct reg_dword game_dvr_allowed; struct reg_dword game_dvr_enabled; struct reg_dword game_dvr_bg_recording; struct reg_dword game_mode_enabled; get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_BAR_REG_KEY, L"AppCaptureEnabled", &game_bar_enabled); get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_DVR_POLICY_REG_KEY, L"AllowGameDVR", &game_dvr_allowed); get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_DVR_REG_KEY, L"GameDVR_Enabled", &game_dvr_enabled); get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_BAR_REG_KEY, L"HistoricalCaptureEnabled", &game_dvr_bg_recording); get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_MODE_REG_KEY, L"AutoGameModeEnabled", &game_mode_enabled); if (game_mode_enabled.status != ERROR_SUCCESS) { get_reg_dword(HKEY_CURRENT_USER, WIN10_GAME_MODE_REG_KEY, L"AllowAutoGameMode", &game_mode_enabled); } blog(LOG_INFO, "Windows 10/11 Gaming Features:"); if (game_bar_enabled.status == ERROR_SUCCESS) { blog(LOG_INFO, "\tGame Bar: %s", (bool)game_bar_enabled.return_value ? "On" : "Off"); } if (game_dvr_allowed.status == ERROR_SUCCESS) { blog(LOG_INFO, "\tGame DVR Allowed: %s", (bool)game_dvr_allowed.return_value ? "Yes" : "No"); } if (game_dvr_enabled.status == ERROR_SUCCESS) { blog(LOG_INFO, "\tGame DVR: %s", (bool)game_dvr_enabled.return_value ? "On" : "Off"); } if (game_dvr_bg_recording.status == ERROR_SUCCESS) { blog(LOG_INFO, "\tGame DVR Background Recording: %s", (bool)game_dvr_bg_recording.return_value ? "On" : "Off"); } if (game_mode_enabled.status == ERROR_SUCCESS) { blog(LOG_INFO, "\tGame Mode: %s", (bool)game_mode_enabled.return_value ? "On" : "Off"); } else if (win_build >= 19042) { // On by default in newer Windows 10 builds (no registry key set) blog(LOG_INFO, "\tGame Mode: Probably On (no reg key set)"); } } static const char *get_str_for_state(int state) { switch (state) { case WSC_SECURITY_PRODUCT_STATE_ON: return "enabled"; case WSC_SECURITY_PRODUCT_STATE_OFF: return "disabled"; case WSC_SECURITY_PRODUCT_STATE_SNOOZED: return "temporarily disabled"; case WSC_SECURITY_PRODUCT_STATE_EXPIRED: return "expired"; default: return "unknown"; } } static const char *get_str_for_type(int type) { switch (type) { case WSC_SECURITY_PROVIDER_ANTIVIRUS: return "AV"; case WSC_SECURITY_PROVIDER_FIREWALL: return "FW"; case WSC_SECURITY_PROVIDER_ANTISPYWARE: return "ASW"; default: return "unknown"; } } static void log_security_products_by_type(IWSCProductList *prod_list, int type) { HRESULT hr; LONG count = 0; IWscProduct *prod; BSTR name; WSC_SECURITY_PRODUCT_STATE prod_state; hr = prod_list->lpVtbl->Initialize(prod_list, type); if (FAILED(hr)) return; hr = prod_list->lpVtbl->get_Count(prod_list, &count); if (FAILED(hr)) { prod_list->lpVtbl->Release(prod_list); return; } for (int i = 0; i < count; i++) { hr = prod_list->lpVtbl->get_Item(prod_list, i, &prod); if (FAILED(hr)) continue; hr = prod->lpVtbl->get_ProductName(prod, &name); if (FAILED(hr)) continue; hr = prod->lpVtbl->get_ProductState(prod, &prod_state); if (FAILED(hr)) { SysFreeString(name); continue; } char *product_name; os_wcs_to_utf8_ptr(name, 0, &product_name); blog(LOG_INFO, "\t%s: %s (%s)", product_name, get_str_for_state(prod_state), get_str_for_type(type)); bfree(product_name); SysFreeString(name); prod->lpVtbl->Release(prod); } prod_list->lpVtbl->Release(prod_list); } static void log_security_products(void) { IWSCProductList *prod_list = NULL; HMODULE h_wsc; HRESULT hr; /* We load the DLL rather than import wcsapi.lib because the clsid / * iid only exists on Windows 8 or higher. */ h_wsc = LoadLibraryW(L"wscapi.dll"); if (!h_wsc) return; const CLSID *prod_list_clsid = (const CLSID *)GetProcAddress(h_wsc, "CLSID_WSCProductList"); const IID *prod_list_iid = (const IID *)GetProcAddress(h_wsc, "IID_IWSCProductList"); if (prod_list_clsid && prod_list_iid) { blog(LOG_INFO, "Sec. Software Status:"); hr = CoCreateInstance(prod_list_clsid, NULL, CLSCTX_INPROC_SERVER, prod_list_iid, &prod_list); if (!FAILED(hr)) { log_security_products_by_type( prod_list, WSC_SECURITY_PROVIDER_ANTIVIRUS); } hr = CoCreateInstance(prod_list_clsid, NULL, CLSCTX_INPROC_SERVER, prod_list_iid, &prod_list); if (!FAILED(hr)) { log_security_products_by_type( prod_list, WSC_SECURITY_PROVIDER_FIREWALL); } hr = CoCreateInstance(prod_list_clsid, NULL, CLSCTX_INPROC_SERVER, prod_list_iid, &prod_list); if (!FAILED(hr)) { log_security_products_by_type( prod_list, WSC_SECURITY_PROVIDER_ANTISPYWARE); } } FreeLibrary(h_wsc); } void log_system_info(void) { struct win_version_info ver; get_win_ver(&ver); win_ver = (ver.major << 8) | ver.minor; win_build = ver.build; log_processor_info(); log_processor_cores(); log_available_memory(); log_windows_version(); log_emulation_status(); log_admin_status(); log_gaming_features(); log_security_products(); log_conflicting_software(); } struct obs_hotkeys_platform { int vk_codes[OBS_KEY_LAST_VALUE]; }; static int get_virtual_key(obs_key_t key) { switch (key) { case OBS_KEY_RETURN: return VK_RETURN; case OBS_KEY_ESCAPE: return VK_ESCAPE; case OBS_KEY_TAB: return VK_TAB; case OBS_KEY_BACKTAB: return VK_OEM_BACKTAB; case OBS_KEY_BACKSPACE: return VK_BACK; case OBS_KEY_INSERT: return VK_INSERT; case OBS_KEY_DELETE: return VK_DELETE; case OBS_KEY_PAUSE: return VK_PAUSE; case OBS_KEY_PRINT: return VK_SNAPSHOT; case OBS_KEY_CLEAR: return VK_CLEAR; case OBS_KEY_HOME: return VK_HOME; case OBS_KEY_END: return VK_END; case OBS_KEY_LEFT: return VK_LEFT; case OBS_KEY_UP: return VK_UP; case OBS_KEY_RIGHT: return VK_RIGHT; case OBS_KEY_DOWN: return VK_DOWN; case OBS_KEY_PAGEUP: return VK_PRIOR; case OBS_KEY_PAGEDOWN: return VK_NEXT; case OBS_KEY_SHIFT: return VK_SHIFT; case OBS_KEY_CONTROL: return VK_CONTROL; case OBS_KEY_ALT: return VK_MENU; case OBS_KEY_CAPSLOCK: return VK_CAPITAL; case OBS_KEY_NUMLOCK: return VK_NUMLOCK; case OBS_KEY_SCROLLLOCK: return VK_SCROLL; case OBS_KEY_F1: return VK_F1; case OBS_KEY_F2: return VK_F2; case OBS_KEY_F3: return VK_F3; case OBS_KEY_F4: return VK_F4; case OBS_KEY_F5: return VK_F5; case OBS_KEY_F6: return VK_F6; case OBS_KEY_F7: return VK_F7; case OBS_KEY_F8: return VK_F8; case OBS_KEY_F9: return VK_F9; case OBS_KEY_F10: return VK_F10; case OBS_KEY_F11: return VK_F11; case OBS_KEY_F12: return VK_F12; case OBS_KEY_F13: return VK_F13; case OBS_KEY_F14: return VK_F14; case OBS_KEY_F15: return VK_F15; case OBS_KEY_F16: return VK_F16; case OBS_KEY_F17: return VK_F17; case OBS_KEY_F18: return VK_F18; case OBS_KEY_F19: return VK_F19; case OBS_KEY_F20: return VK_F20; case OBS_KEY_F21: return VK_F21; case OBS_KEY_F22: return VK_F22; case OBS_KEY_F23: return VK_F23; case OBS_KEY_F24: return VK_F24; case OBS_KEY_SPACE: return VK_SPACE; case OBS_KEY_APOSTROPHE: return VK_OEM_7; case OBS_KEY_PLUS: return VK_OEM_PLUS; case OBS_KEY_COMMA: return VK_OEM_COMMA; case OBS_KEY_MINUS: return VK_OEM_MINUS; case OBS_KEY_PERIOD: return VK_OEM_PERIOD; case OBS_KEY_SLASH: return VK_OEM_2; case OBS_KEY_0: return '0'; case OBS_KEY_1: return '1'; case OBS_KEY_2: return '2'; case OBS_KEY_3: return '3'; case OBS_KEY_4: return '4'; case OBS_KEY_5: return '5'; case OBS_KEY_6: return '6'; case OBS_KEY_7: return '7'; case OBS_KEY_8: return '8'; case OBS_KEY_9: return '9'; case OBS_KEY_NUMASTERISK: return VK_MULTIPLY; case OBS_KEY_NUMPLUS: return VK_ADD; case OBS_KEY_NUMMINUS: return VK_SUBTRACT; case OBS_KEY_NUMPERIOD: return VK_DECIMAL; case OBS_KEY_NUMSLASH: return VK_DIVIDE; case OBS_KEY_NUM0: return VK_NUMPAD0; case OBS_KEY_NUM1: return VK_NUMPAD1; case OBS_KEY_NUM2: return VK_NUMPAD2; case OBS_KEY_NUM3: return VK_NUMPAD3; case OBS_KEY_NUM4: return VK_NUMPAD4; case OBS_KEY_NUM5: return VK_NUMPAD5; case OBS_KEY_NUM6: return VK_NUMPAD6; case OBS_KEY_NUM7: return VK_NUMPAD7; case OBS_KEY_NUM8: return VK_NUMPAD8; case OBS_KEY_NUM9: return VK_NUMPAD9; case OBS_KEY_SEMICOLON: return VK_OEM_1; case OBS_KEY_A: return 'A'; case OBS_KEY_B: return 'B'; case OBS_KEY_C: return 'C'; case OBS_KEY_D: return 'D'; case OBS_KEY_E: return 'E'; case OBS_KEY_F: return 'F'; case OBS_KEY_G: return 'G'; case OBS_KEY_H: return 'H'; case OBS_KEY_I: return 'I'; case OBS_KEY_J: return 'J'; case OBS_KEY_K: return 'K'; case OBS_KEY_L: return 'L'; case OBS_KEY_M: return 'M'; case OBS_KEY_N: return 'N'; case OBS_KEY_O: return 'O'; case OBS_KEY_P: return 'P'; case OBS_KEY_Q: return 'Q'; case OBS_KEY_R: return 'R'; case OBS_KEY_S: return 'S'; case OBS_KEY_T: return 'T'; case OBS_KEY_U: return 'U'; case OBS_KEY_V: return 'V'; case OBS_KEY_W: return 'W'; case OBS_KEY_X: return 'X'; case OBS_KEY_Y: return 'Y'; case OBS_KEY_Z: return 'Z'; case OBS_KEY_BRACKETLEFT: return VK_OEM_4; case OBS_KEY_BACKSLASH: return VK_OEM_5; case OBS_KEY_BRACKETRIGHT: return VK_OEM_6; case OBS_KEY_ASCIITILDE: return VK_OEM_3; case OBS_KEY_HENKAN: return VK_CONVERT; case OBS_KEY_MUHENKAN: return VK_NONCONVERT; case OBS_KEY_KANJI: return VK_KANJI; case OBS_KEY_TOUROKU: return VK_OEM_FJ_TOUROKU; case OBS_KEY_MASSYO: return VK_OEM_FJ_MASSHOU; case OBS_KEY_HANGUL: return VK_HANGUL; case OBS_KEY_BACKSLASH_RT102: return VK_OEM_102; case OBS_KEY_MOUSE1: return VK_LBUTTON; case OBS_KEY_MOUSE2: return VK_RBUTTON; case OBS_KEY_MOUSE3: return VK_MBUTTON; case OBS_KEY_MOUSE4: return VK_XBUTTON1; case OBS_KEY_MOUSE5: return VK_XBUTTON2; case OBS_KEY_VK_CANCEL: return VK_CANCEL; case OBS_KEY_0x07: return 0x07; case OBS_KEY_0x0A: return 0x0A; case OBS_KEY_0x0B: return 0x0B; case OBS_KEY_0x0E: return 0x0E; case OBS_KEY_0x0F: return 0x0F; case OBS_KEY_0x16: return 0x16; case OBS_KEY_VK_JUNJA: return VK_JUNJA; case OBS_KEY_VK_FINAL: return VK_FINAL; case OBS_KEY_0x1A: return 0x1A; case OBS_KEY_VK_ACCEPT: return VK_ACCEPT; case OBS_KEY_VK_MODECHANGE: return VK_MODECHANGE; case OBS_KEY_VK_SELECT: return VK_SELECT; case OBS_KEY_VK_PRINT: return VK_PRINT; case OBS_KEY_VK_EXECUTE: return VK_EXECUTE; case OBS_KEY_VK_HELP: return VK_HELP; case OBS_KEY_0x30: return 0x30; case OBS_KEY_0x31: return 0x31; case OBS_KEY_0x32: return 0x32; case OBS_KEY_0x33: return 0x33; case OBS_KEY_0x34: return 0x34; case OBS_KEY_0x35: return 0x35; case OBS_KEY_0x36: return 0x36; case OBS_KEY_0x37: return 0x37; case OBS_KEY_0x38: return 0x38; case OBS_KEY_0x39: return 0x39; case OBS_KEY_0x3A: return 0x3A; case OBS_KEY_0x3B: return 0x3B; case OBS_KEY_0x3C: return 0x3C; case OBS_KEY_0x3D: return 0x3D; case OBS_KEY_0x3E: return 0x3E; case OBS_KEY_0x3F: return 0x3F; case OBS_KEY_0x40: return 0x40; case OBS_KEY_0x41: return 0x41; case OBS_KEY_0x42: return 0x42; case OBS_KEY_0x43: return 0x43; case OBS_KEY_0x44: return 0x44; case OBS_KEY_0x45: return 0x45; case OBS_KEY_0x46: return 0x46; case OBS_KEY_0x47: return 0x47; case OBS_KEY_0x48: return 0x48; case OBS_KEY_0x49: return 0x49; case OBS_KEY_0x4A: return 0x4A; case OBS_KEY_0x4B: return 0x4B; case OBS_KEY_0x4C: return 0x4C; case OBS_KEY_0x4D: return 0x4D; case OBS_KEY_0x4E: return 0x4E; case OBS_KEY_0x4F: return 0x4F; case OBS_KEY_0x50: return 0x50; case OBS_KEY_0x51: return 0x51; case OBS_KEY_0x52: return 0x52; case OBS_KEY_0x53: return 0x53; case OBS_KEY_0x54: return 0x54; case OBS_KEY_0x55: return 0x55; case OBS_KEY_0x56: return 0x56; case OBS_KEY_0x57: return 0x57; case OBS_KEY_0x58: return 0x58; case OBS_KEY_0x59: return 0x59; case OBS_KEY_0x5A: return 0x5A; case OBS_KEY_VK_LWIN: return VK_LWIN; case OBS_KEY_VK_RWIN: return VK_RWIN; case OBS_KEY_VK_APPS: return VK_APPS; case OBS_KEY_0x5E: return 0x5E; case OBS_KEY_VK_SLEEP: return VK_SLEEP; case OBS_KEY_VK_SEPARATOR: return VK_SEPARATOR; case OBS_KEY_0x88: return 0x88; case OBS_KEY_0x89: return 0x89; case OBS_KEY_0x8A: return 0x8A; case OBS_KEY_0x8B: return 0x8B; case OBS_KEY_0x8C: return 0x8C; case OBS_KEY_0x8D: return 0x8D; case OBS_KEY_0x8E: return 0x8E; case OBS_KEY_0x8F: return 0x8F; case OBS_KEY_VK_OEM_FJ_JISHO: return VK_OEM_FJ_JISHO; case OBS_KEY_VK_OEM_FJ_LOYA: return VK_OEM_FJ_LOYA; case OBS_KEY_VK_OEM_FJ_ROYA: return VK_OEM_FJ_ROYA; case OBS_KEY_0x97: return 0x97; case OBS_KEY_0x98: return 0x98; case OBS_KEY_0x99: return 0x99; case OBS_KEY_0x9A: return 0x9A; case OBS_KEY_0x9B: return 0x9B; case OBS_KEY_0x9C: return 0x9C; case OBS_KEY_0x9D: return 0x9D; case OBS_KEY_0x9E: return 0x9E; case OBS_KEY_0x9F: return 0x9F; case OBS_KEY_VK_LSHIFT: return VK_LSHIFT; case OBS_KEY_VK_RSHIFT: return VK_RSHIFT; case OBS_KEY_VK_LCONTROL: return VK_LCONTROL; case OBS_KEY_VK_RCONTROL: return VK_RCONTROL; case OBS_KEY_VK_LMENU: return VK_LMENU; case OBS_KEY_VK_RMENU: return VK_RMENU; case OBS_KEY_VK_BROWSER_BACK: return VK_BROWSER_BACK; case OBS_KEY_VK_BROWSER_FORWARD: return VK_BROWSER_FORWARD; case OBS_KEY_VK_BROWSER_REFRESH: return VK_BROWSER_REFRESH; case OBS_KEY_VK_BROWSER_STOP: return VK_BROWSER_STOP; case OBS_KEY_VK_BROWSER_SEARCH: return VK_BROWSER_SEARCH; case OBS_KEY_VK_BROWSER_FAVORITES: return VK_BROWSER_FAVORITES; case OBS_KEY_VK_BROWSER_HOME: return VK_BROWSER_HOME; case OBS_KEY_VK_VOLUME_MUTE: return VK_VOLUME_MUTE; case OBS_KEY_VK_VOLUME_DOWN: return VK_VOLUME_DOWN; case OBS_KEY_VK_VOLUME_UP: return VK_VOLUME_UP; case OBS_KEY_VK_MEDIA_NEXT_TRACK: return VK_MEDIA_NEXT_TRACK; case OBS_KEY_VK_MEDIA_PREV_TRACK: return VK_MEDIA_PREV_TRACK; case OBS_KEY_VK_MEDIA_STOP: return VK_MEDIA_STOP; case OBS_KEY_VK_MEDIA_PLAY_PAUSE: return VK_MEDIA_PLAY_PAUSE; case OBS_KEY_VK_LAUNCH_MAIL: return VK_LAUNCH_MAIL; case OBS_KEY_VK_LAUNCH_MEDIA_SELECT: return VK_LAUNCH_MEDIA_SELECT; case OBS_KEY_VK_LAUNCH_APP1: return VK_LAUNCH_APP1; case OBS_KEY_VK_LAUNCH_APP2: return VK_LAUNCH_APP2; case OBS_KEY_0xB8: return 0xB8; case OBS_KEY_0xB9: return 0xB9; case OBS_KEY_0xC1: return 0xC1; case OBS_KEY_0xC2: return 0xC2; case OBS_KEY_0xC3: return 0xC3; case OBS_KEY_0xC4: return 0xC4; case OBS_KEY_0xC5: return 0xC5; case OBS_KEY_0xC6: return 0xC6; case OBS_KEY_0xC7: return 0xC7; case OBS_KEY_0xC8: return 0xC8; case OBS_KEY_0xC9: return 0xC9; case OBS_KEY_0xCA: return 0xCA; case OBS_KEY_0xCB: return 0xCB; case OBS_KEY_0xCC: return 0xCC; case OBS_KEY_0xCD: return 0xCD; case OBS_KEY_0xCE: return 0xCE; case OBS_KEY_0xCF: return 0xCF; case OBS_KEY_0xD0: return 0xD0; case OBS_KEY_0xD1: return 0xD1; case OBS_KEY_0xD2: return 0xD2; case OBS_KEY_0xD3: return 0xD3; case OBS_KEY_0xD4: return 0xD4; case OBS_KEY_0xD5: return 0xD5; case OBS_KEY_0xD6: return 0xD6; case OBS_KEY_0xD7: return 0xD7; case OBS_KEY_0xD8: return 0xD8; case OBS_KEY_0xD9: return 0xD9; case OBS_KEY_0xDA: return 0xDA; case OBS_KEY_VK_OEM_8: return VK_OEM_8; case OBS_KEY_0xE0: return 0xE0; case OBS_KEY_VK_OEM_AX: return VK_OEM_AX; case OBS_KEY_VK_ICO_HELP: return VK_ICO_HELP; case OBS_KEY_VK_ICO_00: return VK_ICO_00; case OBS_KEY_VK_PROCESSKEY: return VK_PROCESSKEY; case OBS_KEY_VK_ICO_CLEAR: return VK_ICO_CLEAR; case OBS_KEY_VK_PACKET: return VK_PACKET; case OBS_KEY_0xE8: return 0xE8; case OBS_KEY_VK_OEM_RESET: return VK_OEM_RESET; case OBS_KEY_VK_OEM_JUMP: return VK_OEM_JUMP; case OBS_KEY_VK_OEM_PA1: return VK_OEM_PA1; case OBS_KEY_VK_OEM_PA2: return VK_OEM_PA2; case OBS_KEY_VK_OEM_PA3: return VK_OEM_PA3; case OBS_KEY_VK_OEM_WSCTRL: return VK_OEM_WSCTRL; case OBS_KEY_VK_OEM_CUSEL: return VK_OEM_CUSEL; case OBS_KEY_VK_OEM_ATTN: return VK_OEM_ATTN; case OBS_KEY_VK_OEM_FINISH: return VK_OEM_FINISH; case OBS_KEY_VK_OEM_COPY: return VK_OEM_COPY; case OBS_KEY_VK_OEM_AUTO: return VK_OEM_AUTO; case OBS_KEY_VK_OEM_ENLW: return VK_OEM_ENLW; case OBS_KEY_VK_ATTN: return VK_ATTN; case OBS_KEY_VK_CRSEL: return VK_CRSEL; case OBS_KEY_VK_EXSEL: return VK_EXSEL; case OBS_KEY_VK_EREOF: return VK_EREOF; case OBS_KEY_VK_PLAY: return VK_PLAY; case OBS_KEY_VK_ZOOM: return VK_ZOOM; case OBS_KEY_VK_NONAME: return VK_NONAME; case OBS_KEY_VK_PA1: return VK_PA1; case OBS_KEY_VK_OEM_CLEAR: return VK_OEM_CLEAR; /* TODO: Implement keys for non-US keyboards */ default:; } return 0; } bool obs_hotkeys_platform_init(struct obs_core_hotkeys *hotkeys) { hotkeys->platform_context = bzalloc(sizeof(obs_hotkeys_platform_t)); for (size_t i = 0; i < OBS_KEY_LAST_VALUE; i++) hotkeys->platform_context->vk_codes[i] = get_virtual_key(i); return true; } void obs_hotkeys_platform_free(struct obs_core_hotkeys *hotkeys) { bfree(hotkeys->platform_context); hotkeys->platform_context = NULL; } static bool vk_down(DWORD vk) { short state = GetAsyncKeyState(vk); bool down = (state & 0x8000) != 0; return down; } bool obs_hotkeys_platform_is_pressed(obs_hotkeys_platform_t *context, obs_key_t key) { if (key == OBS_KEY_META) { return vk_down(VK_LWIN) || vk_down(VK_RWIN); } UNUSED_PARAMETER(context); return vk_down(obs_key_to_virtual_key(key)); } void obs_key_to_str(obs_key_t key, struct dstr *str) { wchar_t name[128] = L""; UINT scan_code; int vk; if (key == OBS_KEY_NONE) { return; } else if (key >= OBS_KEY_F13 && key <= OBS_KEY_F24) { dstr_printf(str, "F%d", (int)(key - OBS_KEY_F13 + 13)); return; } else if (key >= OBS_KEY_MOUSE1 && key <= OBS_KEY_MOUSE29) { if (obs->hotkeys.translations[key]) { dstr_copy(str, obs->hotkeys.translations[key]); } else { dstr_printf(str, "Mouse %d", (int)(key - OBS_KEY_MOUSE1 + 1)); } return; } if (key == OBS_KEY_PAUSE) { dstr_copy(str, obs_get_hotkey_translation(key, "Pause")); return; } else if (key == OBS_KEY_META) { dstr_copy(str, obs_get_hotkey_translation(key, "Windows")); return; } vk = obs_key_to_virtual_key(key); scan_code = MapVirtualKey(vk, 0) << 16; switch (vk) { case VK_HOME: case VK_END: case VK_LEFT: case VK_UP: case VK_RIGHT: case VK_DOWN: case VK_PRIOR: case VK_NEXT: case VK_INSERT: case VK_DELETE: case VK_NUMLOCK: scan_code |= 0x01000000; } if ((key < OBS_KEY_VK_CANCEL || key > OBS_KEY_VK_OEM_CLEAR) && scan_code != 0 && GetKeyNameTextW(scan_code, name, 128) != 0) { dstr_from_wcs(str, name); } else if (key != OBS_KEY_NONE) { dstr_copy(str, obs_key_to_name(key)); } } obs_key_t obs_key_from_virtual_key(int code) { obs_hotkeys_platform_t *platform = obs->hotkeys.platform_context; for (size_t i = 0; i < OBS_KEY_LAST_VALUE; i++) { if (platform->vk_codes[i] == code) { return (obs_key_t)i; } } return OBS_KEY_NONE; } int obs_key_to_virtual_key(obs_key_t key) { if (key == OBS_KEY_META) return VK_LWIN; return obs->hotkeys.platform_context->vk_codes[(int)key]; } static inline void add_combo_key(obs_key_t key, struct dstr *str) { struct dstr key_str = {0}; obs_key_to_str(key, &key_str); if (!dstr_is_empty(&key_str)) { if (!dstr_is_empty(str)) { dstr_cat(str, " + "); } dstr_cat_dstr(str, &key_str); } dstr_free(&key_str); } void obs_key_combination_to_str(obs_key_combination_t combination, struct dstr *str) { if ((combination.modifiers & INTERACT_CONTROL_KEY) != 0) { add_combo_key(OBS_KEY_CONTROL, str); } if ((combination.modifiers & INTERACT_COMMAND_KEY) != 0) { add_combo_key(OBS_KEY_META, str); } if ((combination.modifiers & INTERACT_ALT_KEY) != 0) { add_combo_key(OBS_KEY_ALT, str); } if ((combination.modifiers & INTERACT_SHIFT_KEY) != 0) { add_combo_key(OBS_KEY_SHIFT, str); } if (combination.key != OBS_KEY_NONE) { add_combo_key(combination.key, str); } } bool sym_initialize_called = false; void reset_win32_symbol_paths(void) { static BOOL(WINAPI * sym_initialize_w)(HANDLE, const wchar_t *, BOOL); static BOOL(WINAPI * sym_set_search_path_w)(HANDLE, const wchar_t *); static bool funcs_initialized = false; static bool initialize_success = false; struct obs_module *module = obs->first_module; struct dstr path_str = {0}; DARRAY(char *) paths; wchar_t *path_str_w = NULL; char *abspath; da_init(paths); if (!funcs_initialized) { HMODULE mod; funcs_initialized = true; mod = LoadLibraryW(L"DbgHelp"); if (!mod) return; sym_initialize_w = (void *)GetProcAddress(mod, "SymInitializeW"); sym_set_search_path_w = (void *)GetProcAddress(mod, "SymSetSearchPathW"); if (!sym_initialize_w || !sym_set_search_path_w) { FreeLibrary(mod); return; } initialize_success = true; // Leaks 'mod' once. } if (!initialize_success) return; abspath = os_get_abs_path_ptr("."); if (abspath) da_push_back(paths, &abspath); while (module) { bool found = false; struct dstr path = {0}; char *path_end; dstr_copy(&path, module->bin_path); dstr_replace(&path, "/", "\\"); path_end = strrchr(path.array, '\\'); if (!path_end) { module = module->next; dstr_free(&path); continue; } *path_end = 0; abspath = os_get_abs_path_ptr(path.array); if (abspath) { for (size_t i = 0; i < paths.num; i++) { const char *existing_path = paths.array[i]; if (astrcmpi(abspath, existing_path) == 0) { found = true; break; } } if (!found) { da_push_back(paths, &abspath); } else { bfree(abspath); } } dstr_free(&path); module = module->next; } for (size_t i = 0; i < paths.num; i++) { const char *path = paths.array[i]; if (path && *path) { if (i != 0) dstr_cat(&path_str, ";"); dstr_cat(&path_str, paths.array[i]); } } if (path_str.array) { os_utf8_to_wcs_ptr(path_str.array, path_str.len, &path_str_w); if (path_str_w) { if (!sym_initialize_called) { sym_initialize_w(GetCurrentProcess(), path_str_w, false); sym_initialize_called = true; } else { sym_set_search_path_w(GetCurrentProcess(), path_str_w); } bfree(path_str_w); } } for (size_t i = 0; i < paths.num; i++) bfree(paths.array[i]); dstr_free(&path_str); da_free(paths); } extern void initialize_crash_handler(void); void obs_init_win32_crash_handler(void) { initialize_crash_handler(); } bool initialize_com(void) { const HRESULT hr = CoInitializeEx(0, COINIT_APARTMENTTHREADED); const bool success = SUCCEEDED(hr); if (!success) blog(LOG_ERROR, "CoInitializeEx failed: 0x%08X", hr); return success; } void uninitialize_com(void) { CoUninitialize(); }