obs-studio/plugins/win-capture/game-capture-file-init.c

334 lines
8.2 KiB
C
Raw Normal View History

#include <windows.h>
#include <strsafe.h>
#include <shlobj.h>
#include <aclapi.h>
#include <sddl.h>
#include <obs-module.h>
#include <util/windows/win-version.h>
#include <util/platform.h>
#include <util/c99defs.h>
#include <util/base.h>
/* ------------------------------------------------------------------------- */
/* helper funcs */
static bool has_elevation_internal()
{
SID_IDENTIFIER_AUTHORITY sia = SECURITY_NT_AUTHORITY;
PSID sid = NULL;
BOOL elevated = false;
BOOL success;
success = AllocateAndInitializeSid(&sia, 2, SECURITY_BUILTIN_DOMAIN_RID,
DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0,
0, 0, &sid);
if (success && sid) {
CheckTokenMembership(NULL, sid, &elevated);
FreeSid(sid);
}
return elevated;
}
static bool has_elevation()
{
static bool elevated = false;
static bool initialized = false;
if (!initialized) {
elevated = has_elevation_internal();
initialized = true;
}
return elevated;
}
static bool add_aap_perms(const wchar_t *dir)
{
PSECURITY_DESCRIPTOR sd = NULL;
SID *aap_sid = NULL;
SID *bu_sid = NULL;
PACL new_dacl1 = NULL;
PACL new_dacl2 = NULL;
bool success = false;
PACL dacl;
if (GetNamedSecurityInfoW(dir, SE_FILE_OBJECT,
DACL_SECURITY_INFORMATION, NULL, NULL, &dacl,
NULL, &sd) != ERROR_SUCCESS) {
goto fail;
}
EXPLICIT_ACCESSW ea = {0};
ea.grfAccessPermissions = GENERIC_READ | GENERIC_EXECUTE;
ea.grfAccessMode = GRANT_ACCESS;
ea.grfInheritance = SUB_CONTAINERS_AND_OBJECTS_INHERIT;
ea.Trustee.TrusteeForm = TRUSTEE_IS_SID;
/* ALL_APP_PACKAGES */
ConvertStringSidToSidW(L"S-1-15-2-1", &aap_sid);
ea.Trustee.ptstrName = (wchar_t *)aap_sid;
if (SetEntriesInAclW(1, &ea, dacl, &new_dacl1) != ERROR_SUCCESS) {
goto fail;
}
ea.grfAccessPermissions = GENERIC_READ | GENERIC_WRITE |
GENERIC_EXECUTE;
/* BUILTIN_USERS */
ConvertStringSidToSidW(L"S-1-5-32-545", &bu_sid);
ea.Trustee.ptstrName = (wchar_t *)bu_sid;
DWORD s = SetEntriesInAclW(1, &ea, new_dacl1, &new_dacl2);
if (s != ERROR_SUCCESS) {
goto fail;
}
if (SetNamedSecurityInfoW((wchar_t *)dir, SE_FILE_OBJECT,
DACL_SECURITY_INFORMATION, NULL, NULL,
new_dacl2, NULL) != ERROR_SUCCESS) {
goto fail;
}
success = true;
fail:
if (sd)
LocalFree(sd);
if (new_dacl1)
LocalFree(new_dacl1);
if (new_dacl2)
LocalFree(new_dacl2);
if (aap_sid)
LocalFree(aap_sid);
if (bu_sid)
LocalFree(bu_sid);
return success;
}
static inline bool file_exists(const wchar_t *path)
{
WIN32_FIND_DATAW wfd;
HANDLE h = FindFirstFileW(path, &wfd);
if (h == INVALID_HANDLE_VALUE)
return false;
FindClose(h);
return true;
}
static LSTATUS get_reg(HKEY hkey, LPCWSTR sub_key, LPCWSTR value_name, bool b64)
{
HKEY key;
LSTATUS status;
DWORD flags = b64 ? KEY_WOW64_64KEY : KEY_WOW64_32KEY;
DWORD size = sizeof(DWORD);
DWORD val;
status = RegOpenKeyEx(hkey, sub_key, 0, KEY_READ | flags, &key);
if (status == ERROR_SUCCESS) {
status = RegQueryValueExW(key, value_name, NULL, NULL,
(LPBYTE)&val, &size);
RegCloseKey(key);
}
return status;
}
#define get_programdata_path(path, subpath) \
do { \
SHGetFolderPathW(NULL, CSIDL_COMMON_APPDATA, NULL, \
SHGFP_TYPE_CURRENT, path); \
StringCbCatW(path, sizeof(path), L"\\"); \
StringCbCatW(path, sizeof(path), subpath); \
} while (false)
#define make_filename(str, name, ext) \
do { \
StringCbCatW(str, sizeof(str), name); \
StringCbCatW(str, sizeof(str), b64 ? L"64" : L"32"); \
StringCbCatW(str, sizeof(str), ext); \
} while (false)
/* ------------------------------------------------------------------------- */
/* function to get the path to the hook */
static bool programdata64_hook_exists = false;
static bool programdata32_hook_exists = false;
char *get_hook_path(bool b64)
{
wchar_t path[MAX_PATH];
get_programdata_path(path, L"obs-studio-hook\\");
make_filename(path, L"graphics-hook", L".dll");
if ((b64 && programdata64_hook_exists) ||
(!b64 && programdata32_hook_exists)) {
char *path_utf8 = NULL;
os_wcs_to_utf8_ptr(path, 0, &path_utf8);
return path_utf8;
}
return obs_module_file(b64 ? "graphics-hook64.dll"
: "graphics-hook32.dll");
}
/* ------------------------------------------------------------------------- */
/* initialization */
#define IMPLICIT_LAYERS L"SOFTWARE\\Khronos\\Vulkan\\ImplicitLayers"
static bool update_hook_file(bool b64)
{
wchar_t temp[MAX_PATH];
wchar_t src[MAX_PATH];
wchar_t dst[MAX_PATH];
wchar_t src_json[MAX_PATH];
wchar_t dst_json[MAX_PATH];
StringCbCopyW(temp, sizeof(temp),
L"..\\..\\data\\obs-plugins\\"
L"win-capture\\");
make_filename(temp, L"obs-vulkan", L".json");
if (_wfullpath(src_json, temp, MAX_PATH) == NULL)
return false;
StringCbCopyW(temp, sizeof(temp),
L"..\\..\\data\\obs-plugins\\"
L"win-capture\\");
make_filename(temp, L"graphics-hook", L".dll");
if (_wfullpath(src, temp, MAX_PATH) == NULL)
return false;
get_programdata_path(temp, L"obs-studio-hook\\");
StringCbCopyW(dst_json, sizeof(dst_json), temp);
StringCbCopyW(dst, sizeof(dst), temp);
make_filename(dst_json, L"obs-vulkan", L".json");
make_filename(dst, L"graphics-hook", L".dll");
if (!file_exists(src)) {
return false;
}
if (!file_exists(src_json)) {
return false;
}
if (!file_exists(dst) || !file_exists(dst_json)) {
CreateDirectoryW(temp, NULL);
if (has_elevation())
add_aap_perms(temp);
if (!CopyFileW(src_json, dst_json, false))
return false;
if (!CopyFileW(src, dst, false))
return false;
return true;
}
if (has_elevation())
add_aap_perms(temp);
struct win_version_info ver_src = {0};
struct win_version_info ver_dst = {0};
if (!get_dll_ver(src, &ver_src))
return false;
#ifndef _DEBUG
if (!get_dll_ver(dst, &ver_dst))
return false;
#endif
/* if source is greater than dst, overwrite new file */
while (win_version_compare(&ver_dst, &ver_src) < 0) {
if (!CopyFileW(src_json, dst_json, false))
return false;
if (!CopyFileW(src, dst, false))
return false;
if (!get_dll_ver(dst, &ver_dst))
return false;
}
/* do not use if major version incremented in target compared to
* ours */
if (ver_dst.major > ver_src.major) {
return false;
}
return true;
}
#define warn(format, ...) \
blog(LOG_WARNING, "%s: " format, "[Vulkan Capture Init]", ##__VA_ARGS__)
/* Sets vulkan layer registry if it doesn't already exist */
static void init_vulkan_registry(bool b64)
{
DWORD flags = b64 ? KEY_WOW64_64KEY : KEY_WOW64_32KEY;
HKEY key = NULL;
LSTATUS s;
wchar_t path[MAX_PATH];
get_programdata_path(path, L"obs-studio-hook\\");
make_filename(path, L"obs-vulkan", L".json");
s = get_reg(HKEY_LOCAL_MACHINE, IMPLICIT_LAYERS, path, b64);
if (s == ERROR_FILE_NOT_FOUND) {
s = get_reg(HKEY_CURRENT_USER, IMPLICIT_LAYERS, path, b64);
if (s != ERROR_FILE_NOT_FOUND && s != ERROR_SUCCESS) {
warn("Failed to query registry keys: %d", (int)s);
goto finish;
}
if (s == ERROR_SUCCESS && has_elevation()) {
s = RegOpenKeyEx(HKEY_CURRENT_USER, IMPLICIT_LAYERS, 0,
KEY_WRITE | flags, &key);
if (s == ERROR_SUCCESS) {
RegDeleteValueW(key, path);
RegCloseKey(key);
s = -1;
key = NULL;
}
}
} else if (s != ERROR_SUCCESS) {
warn("Failed to query registry keys: %d", (int)s);
goto finish;
}
if (s == ERROR_SUCCESS) {
goto finish;
}
HKEY type = has_elevation() ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
DWORD temp;
s = RegCreateKeyExW(type, IMPLICIT_LAYERS, 0, NULL, 0,
KEY_WRITE | flags, NULL, &key, &temp);
if (s != ERROR_SUCCESS) {
warn("Failed to create registry key");
goto finish;
}
DWORD zero = 0;
s = RegSetValueExW(key, path, 0, REG_DWORD, (const BYTE *)&zero,
sizeof(zero));
if (s != ERROR_SUCCESS) {
warn("Failed to set registry value");
}
finish:
if (key)
RegCloseKey(key);
}
void init_hook_files()
{
if (update_hook_file(true)) {
programdata64_hook_exists = true;
init_vulkan_registry(true);
}
if (update_hook_file(false)) {
programdata32_hook_exists = true;
init_vulkan_registry(false);
}
}