Windows2003-3790/windows/appcompat/shims/verifier/registrychecks.cpp
2020-09-30 16:53:55 +02:00

1493 lines
34 KiB
C++

/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
RegistryChecks.cpp
Abstract:
Warn the app when it's trying to read from or write to inappropriate
places in the registry.
Notes:
This is a general purpose shim.
History:
03/09/2001 maonis Created
09/04/2001 maonis Since none of the paths we compare with exceed MAX_PATH - 1
characters, we only examine at most that many characters of
the key paths to make sure there's no buffer overflow in
the paths of open keys.
02/20/2002 rparsons Implemented strsafe functions.
02/25/2002 rparsons Modified critical section code to be thread safe.
--*/
#include "precomp.h"
IMPLEMENT_SHIM_BEGIN(RegistryChecks)
#include "ShimHookMacro.h"
#include "RegistryChecks.h"
//
// verifier log entries
//
BEGIN_DEFINE_VERIFIER_LOG(RegistryChecks)
VERIFIER_LOG_ENTRY(VLOG_HKCU_Console_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_ControlPanel_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_Environment_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_Identities_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_KeyboardLayout_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_Printers_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_RemoteAccess_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_SessionInformation_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_UNICODEProgramGroups_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_VolatileEnvironment_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCU_Windows31MigrationStatus_READ)
VERIFIER_LOG_ENTRY(VLOG_HKLM_HARDWARE_READ)
VERIFIER_LOG_ENTRY(VLOG_HKLM_SAM_READ)
VERIFIER_LOG_ENTRY(VLOG_HKLM_SECURITY_READ)
VERIFIER_LOG_ENTRY(VLOG_HKLM_SYSTEM_READ)
VERIFIER_LOG_ENTRY(VLOG_HKCC_READ)
VERIFIER_LOG_ENTRY(VLOG_HKUS_READ)
VERIFIER_LOG_ENTRY(VLOG_NON_HKCU_WRITE)
END_DEFINE_VERIFIER_LOG(RegistryChecks)
INIT_VERIFIER_LOG(RegistryChecks);
const RCWARNING g_warnNoDirectRead[] =
{
{HKCU_Console_STR, VLOG_HKCU_Console_READ, NUM_OF_CHAR(HKCU_Console_STR)},
{HKCU_ControlPanel_STR, VLOG_HKCU_ControlPanel_READ, NUM_OF_CHAR(HKCU_ControlPanel_STR)},
{HKCU_Environment_STR, VLOG_HKCU_Environment_READ, NUM_OF_CHAR(HKCU_Environment_STR)},
{HKCU_Identities_STR, VLOG_HKCU_Identities_READ, NUM_OF_CHAR(HKCU_Identities_STR)},
{HKCU_KeyboardLayout_STR, VLOG_HKCU_KeyboardLayout_READ, NUM_OF_CHAR(HKCU_KeyboardLayout_STR)},
{HKCU_Printers_STR, VLOG_HKCU_Printers_READ, NUM_OF_CHAR(HKCU_Printers_STR)},
{HKCU_RemoteAccess_STR, VLOG_HKCU_RemoteAccess_READ, NUM_OF_CHAR(HKCU_RemoteAccess_STR)},
{HKCU_SessionInformation_STR, VLOG_HKCU_SessionInformation_READ, NUM_OF_CHAR(HKCU_SessionInformation_STR)},
{HKCU_UNICODEProgramGroups_STR, VLOG_HKCU_UNICODEProgramGroups_READ, NUM_OF_CHAR(HKCU_UNICODEProgramGroups_STR)},
{HKCU_VolatileEnvironment_STR, VLOG_HKCU_VolatileEnvironment_READ, NUM_OF_CHAR(HKCU_VolatileEnvironment_STR)},
{HKCU_Windows31MigrationStatus_STR, VLOG_HKCU_Windows31MigrationStatus_READ,NUM_OF_CHAR(HKCU_Windows31MigrationStatus_STR)},
{HKLM_HARDWARE_STR, VLOG_HKLM_HARDWARE_READ, NUM_OF_CHAR(HKLM_HARDWARE_STR)},
{HKLM_SAM_STR, VLOG_HKLM_SAM_READ, NUM_OF_CHAR(HKLM_SAM_STR)},
{HKLM_SECURITY_STR, VLOG_HKLM_SECURITY_READ, NUM_OF_CHAR(HKLM_SECURITY_STR)},
{HKLM_SYSTEM_STR, VLOG_HKLM_SYSTEM_READ, NUM_OF_CHAR(HKLM_SYSTEM_STR)},
{HKCC_STR, VLOG_HKCC_READ, NUM_OF_CHAR(HKCC_STR)},
{HKUS_STR, VLOG_HKUS_READ, NUM_OF_CHAR(HKUS_STR)},
};
const UINT g_cWarnNDirectRead = sizeof(g_warnNoDirectRead) / sizeof(RCWARNING);
//
// Critical section that keeps us safe while using linked-lists, etc.
//
CCriticalSection g_csCritSec;
VOID
MakePathW(
IN RCOPENKEY* key,
IN HKEY hKey,
IN LPCWSTR lpSubKey,
IN OUT LPWSTR lpPath
)
{
if (key) {
if (key->wszPath[0]) {
//
// We only care about at most MAX_PATH - 1 characters.
//
wcsncpy(lpPath, key->wszPath, MAX_PATH - 1);
}
} else {
if (hKey == HKEY_CLASSES_ROOT) {
StringCchCopy(lpPath, MAX_PATH - 1, L"HKCR");
} else if (hKey == HKEY_CURRENT_CONFIG) {
StringCchCopy(lpPath, MAX_PATH - 1, L"HKCC");
} else if (hKey == HKEY_CURRENT_USER) {
StringCchCopy(lpPath, MAX_PATH - 1, L"HKCU");
} else if (hKey == HKEY_LOCAL_MACHINE) {
StringCchCopy(lpPath, MAX_PATH - 1, L"HKLM");
} else if (hKey == HKEY_USERS) {
StringCchCopy(lpPath, MAX_PATH - 1, L"HKUS");
} else {
StringCchCopy(lpPath, MAX_PATH - 1, L"Not recongized");
}
}
if (lpSubKey && *lpSubKey) {
DWORD cLen = wcslen(lpPath);
//
// We only care about at most MAX_PATH - 1 characters.
//
if (cLen < MAX_PATH - 1) {
lpPath[cLen] = L'\\';
wcsncpy(lpPath + cLen + 1, lpSubKey, MAX_PATH - cLen - 2);
}
}
lpPath[MAX_PATH - 1] = L'\0';
}
VOID CheckReading(
IN LPCWSTR pwszPath
)
{
RCWARNING warn;
for (UINT ui = 0; ui < g_cWarnNDirectRead; ++ui) {
warn = g_warnNoDirectRead[ui];
if (!_wcsnicmp(pwszPath, warn.wszPath, warn.cLen)) {
VLOG(VLOG_LEVEL_ERROR, warn.dwAVStatus,
"Read from dangerous registry entry '%ls'.",
pwszPath);
}
}
}
// We warn for every tempt of writing to anything besides keys under HKCU.
// Note this applies to both Users and Admins/Power Users because when an
// app is running it shouldn't write anything to non HKCU keys, which should
// be done during the installation time.
VOID CheckWriting(
IN REGSAM samDesired,
IN LPCWSTR pwszPath
)
{
if ((samDesired &~ STANDARD_RIGHTS_WRITE) & KEY_WRITE) {
if (_wcsnicmp(pwszPath, L"HKCU", 4)) {
VLOG(VLOG_LEVEL_ERROR, VLOG_NON_HKCU_WRITE, "Write to non-HKCU registry entry '%ls'.", pwszPath);
}
}
}
//
// Implementation of the CRegistryChecks class.
//
RCOPENKEY*
CRegistryChecks::FindKey(
HKEY hKey
)
{
RCOPENKEY* key = keys;
while (key) {
if (key->hkBase == hKey) {
return key;
}
key = key->next;
}
return NULL;
}
// We add the key to the front of the list because the most
// recently added keys are usually used/deleted first.
BOOL
CRegistryChecks::AddKey(
HKEY hKey,
LPCWSTR pwszPath
)
{
RCOPENKEY* key = new RCOPENKEY;
if (!key) {
return FALSE;
}
key->hkBase = hKey;
//
// None of the key paths we need to check exceed MAX_PATH - 1 characters so
// we only need to copy at most that many characters.
//
wcsncpy(key->wszPath, pwszPath, MAX_PATH - 1);
key->wszPath[MAX_PATH - 1] = L'\0';
key->next = keys;
keys = key;
return TRUE;
}
VOID
CRegistryChecks::Check(
HKEY hKey,
LPCSTR lpSubKey,
BOOL fCheckRead,
BOOL fCheckWrite,
REGSAM samDesired
)
{
LPWSTR pwszSubKey = NULL;
if (pwszSubKey = ToUnicode(lpSubKey)) {
Check(hKey, pwszSubKey, fCheckRead, fCheckWrite);
free(pwszSubKey);
} else {
DPFN(eDbgLevelError, "Failed to convert %s to unicode", lpSubKey);
}
}
VOID
CRegistryChecks::Check(
HKEY hKey,
LPCWSTR lpSubKey,
BOOL fCheckRead,
BOOL fCheckWrite,
REGSAM samDesired
)
{
RCOPENKEY* key = FindKey(hKey);
WCHAR wszPath[MAX_PATH] = L"";
MakePathW(key, hKey, lpSubKey, wszPath);
if (fCheckRead) {
CheckReading(wszPath);
}
if (fCheckWrite) {
CheckWriting(samDesired, wszPath);
}
}
LONG
CRegistryChecks::OpenKeyExOriginalW(
HKEY hKey,
LPCWSTR lpSubKey,
LPWSTR lpClass,
DWORD dwOptions,
REGSAM samDesired,
LPSECURITY_ATTRIBUTES lpSecurityAttributes,
PHKEY phkResult,
LPDWORD lpdwDisposition,
BOOL bCreate
)
{
if (bCreate) {
return ORIGINAL_API(RegCreateKeyExW)(
hKey,
lpSubKey,
0,
lpClass,
dwOptions,
samDesired,
lpSecurityAttributes,
phkResult,
lpdwDisposition);
} else {
return ORIGINAL_API(RegOpenKeyExW)(
hKey,
lpSubKey,
0,
samDesired,
phkResult);
}
}
LONG
CRegistryChecks::OpenKeyExA(
HKEY hKey,
LPCSTR lpSubKey,
LPSTR lpClass,
DWORD dwOptions,
REGSAM samDesired,
LPSECURITY_ATTRIBUTES lpSecurityAttributes,
PHKEY phkResult,
LPDWORD lpdwDisposition,
BOOL bCreate
)
{
LONG lRet;
LPWSTR pwszSubKey = NULL;
LPWSTR pwszClass = NULL;
if (lpSubKey) {
if (!(pwszSubKey = ToUnicode(lpSubKey))) {
DPFN(eDbgLevelError, "Failed to convert %s to unicode", lpSubKey);
return ERROR_NOT_ENOUGH_MEMORY;
}
}
if (lpClass) {
if (!(pwszClass = ToUnicode(lpClass)))
{
free(pwszSubKey);
DPFN(eDbgLevelError, "Failed to convert %s to unicode", lpClass);
return ERROR_NOT_ENOUGH_MEMORY;
}
}
lRet = OpenKeyExW(
hKey,
pwszSubKey,
pwszClass,
dwOptions,
samDesired,
lpSecurityAttributes,
phkResult,
lpdwDisposition,
bCreate);
free(pwszSubKey);
free(pwszClass);
return lRet;
}
LONG
CRegistryChecks::OpenKeyExW(
HKEY hKey,
LPCWSTR lpSubKey,
LPWSTR lpClass,
DWORD dwOptions,
REGSAM samDesired,
LPSECURITY_ATTRIBUTES lpSecurityAttributes,
PHKEY phkResult,
LPDWORD lpdwDisposition,
BOOL bCreate
)
{
RCOPENKEY* key = FindKey(hKey);
WCHAR wszPath[MAX_PATH] = L"";
MakePathW(key, hKey, lpSubKey, wszPath);
CheckReading(wszPath);
CheckWriting(samDesired, wszPath);
LONG lRes = OpenKeyExOriginalW(
hKey,
lpSubKey,
lpClass,
dwOptions,
samDesired,
lpSecurityAttributes,
phkResult,
lpdwDisposition,
bCreate);
if (lRes == ERROR_SUCCESS) {
if (AddKey(*phkResult, wszPath)) {
DPFN(eDbgLevelInfo, "[OpenKeyExW] success - adding key 0x%08X", *phkResult);
} else {
lRes = ERROR_INVALID_HANDLE;
}
}
return lRes;
}
LONG
CRegistryChecks::QueryValueA(
HKEY hKey,
LPCSTR lpSubKey,
LPSTR lpValue,
PLONG lpcbValue
)
{
Check(hKey, lpSubKey, TRUE, FALSE);
return ORIGINAL_API(RegQueryValueA)(
hKey,
lpSubKey,
lpValue,
lpcbValue);
}
LONG
CRegistryChecks::QueryValueW(
HKEY hKey,
LPCWSTR lpSubKey,
LPWSTR lpValue,
PLONG lpcbValue
)
{
Check(hKey, lpSubKey, TRUE, FALSE);
return ORIGINAL_API(RegQueryValueW)(
hKey,
lpSubKey,
lpValue,
lpcbValue);
}
LONG
CRegistryChecks::QueryValueExA(
HKEY hKey,
LPCSTR lpValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegQueryValueExA)(
hKey,
lpValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
CRegistryChecks::QueryValueExW(
HKEY hKey,
LPCWSTR lpValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegQueryValueExW)(
hKey,
lpValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
CRegistryChecks::QueryInfoKeyA(
HKEY hKey,
LPSTR lpClass,
LPDWORD lpcbClass,
LPDWORD lpReserved,
LPDWORD lpcSubKeys,
LPDWORD lpcbMaxSubKeyLen,
LPDWORD lpcbMaxClassLen,
LPDWORD lpcValues,
LPDWORD lpcbMaxValueNameLen,
LPDWORD lpcbMaxValueLen,
LPDWORD lpcbSecurityDescriptor,
PFILETIME lpftLastWriteTime
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegQueryInfoKeyA)(
hKey,
lpClass,
lpcbClass,
lpReserved,
lpcSubKeys,
lpcbMaxSubKeyLen,
lpcbMaxClassLen,
lpcValues,
lpcbMaxValueNameLen,
lpcbMaxValueLen,
lpcbSecurityDescriptor,
lpftLastWriteTime);
}
LONG
CRegistryChecks::QueryInfoKeyW(
HKEY hKey,
LPWSTR lpClass,
LPDWORD lpcbClass,
LPDWORD lpReserved,
LPDWORD lpcSubKeys,
LPDWORD lpcbMaxSubKeyLen,
LPDWORD lpcbMaxClassLen,
LPDWORD lpcValues,
LPDWORD lpcbMaxValueNameLen,
LPDWORD lpcbMaxValueLen,
LPDWORD lpcbSecurityDescriptor,
PFILETIME lpftLastWriteTime
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegQueryInfoKeyW)(
hKey,
lpClass,
lpcbClass,
lpReserved,
lpcSubKeys,
lpcbMaxSubKeyLen,
lpcbMaxClassLen,
lpcValues,
lpcbMaxValueNameLen,
lpcbMaxValueLen,
lpcbSecurityDescriptor,
lpftLastWriteTime);
}
LONG
CRegistryChecks::SetValueA(
HKEY hKey,
LPCSTR lpSubKey,
DWORD dwType,
LPCSTR lpData,
DWORD cbData
)
{
Check(hKey, lpSubKey, FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegSetValueA)(
hKey,
lpSubKey,
dwType,
lpData,
cbData);
}
LONG
CRegistryChecks::SetValueW(
HKEY hKey,
LPCWSTR lpSubKey,
DWORD dwType,
LPCWSTR lpData,
DWORD cbData
)
{
Check(hKey, lpSubKey, FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegSetValueW)(
hKey,
lpSubKey,
dwType,
lpData,
cbData);
}
LONG
CRegistryChecks::SetValueExA(
HKEY hKey,
LPCSTR lpValueName,
DWORD Reserved,
DWORD dwType,
CONST BYTE * lpData,
DWORD cbData
)
{
Check(hKey, L"", FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegSetValueExA)(
hKey,
lpValueName,
Reserved,
dwType,
lpData,
cbData);
}
LONG
CRegistryChecks::SetValueExW(
HKEY hKey,
LPCWSTR lpValueName,
DWORD Reserved,
DWORD dwType,
CONST BYTE * lpData,
DWORD cbData
)
{
Check(hKey, L"", FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegSetValueExW)(
hKey,
lpValueName,
Reserved,
dwType,
lpData,
cbData);
}
LONG
CRegistryChecks::EnumValueA(
HKEY hKey,
DWORD dwIndex,
LPSTR lpValueName,
LPDWORD lpcbValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegEnumValueA)(
hKey,
dwIndex,
lpValueName,
lpcbValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
// If the key was not originated from HKCU,
// we enum at the original location.
LONG
CRegistryChecks::EnumValueW(
HKEY hKey,
DWORD dwIndex,
LPWSTR lpValueName,
LPDWORD lpcbValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegEnumValueW)(
hKey,
dwIndex,
lpValueName,
lpcbValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
CRegistryChecks::EnumKeyExA(
HKEY hKey,
DWORD dwIndex,
LPSTR lpName,
LPDWORD lpcbName,
LPDWORD lpReserved,
LPSTR lpClass,
LPDWORD lpcbClass,
PFILETIME lpftLastWriteTime
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegEnumKeyExA)(
hKey,
dwIndex,
lpName,
lpcbName,
lpReserved,
lpClass,
lpcbClass,
lpftLastWriteTime);
}
// If the key was not originated from HKCU,
// we enum at the original location.
LONG
CRegistryChecks::EnumKeyExW(
HKEY hKey,
DWORD dwIndex,
LPWSTR lpName,
LPDWORD lpcbName,
LPDWORD lpReserved,
LPWSTR lpClass,
LPDWORD lpcbClass,
PFILETIME lpftLastWriteTime
)
{
Check(hKey, L"", TRUE, FALSE);
return ORIGINAL_API(RegEnumKeyExW)(
hKey,
dwIndex,
lpName,
lpcbName,
lpReserved,
lpClass,
lpcbClass,
lpftLastWriteTime);
}
// Remove the key from the list.
LONG
CRegistryChecks::CloseKey(
HKEY hKey
)
{
RCOPENKEY* key = keys;
RCOPENKEY* last = NULL;
while (key) {
if (key->hkBase == hKey) {
if (last) {
last->next = key->next;
} else {
keys = key->next;
}
delete key;
break;
}
last = key;
key = key->next;
}
DPFN(eDbgLevelInfo, "[CloseKey] closing key 0x%08X", hKey);
return ORIGINAL_API(RegCloseKey)(hKey);
}
LONG
CRegistryChecks::DeleteKeyA(
HKEY hKey,
LPCSTR lpSubKey
)
{
Check(hKey, lpSubKey, FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegDeleteKeyA)(
hKey,
lpSubKey);
}
LONG
CRegistryChecks::DeleteKeyW(
HKEY hKey,
LPCWSTR lpSubKey
)
{
Check(hKey, lpSubKey, FALSE, TRUE, KEY_WRITE);
return ORIGINAL_API(RegDeleteKeyW)(
hKey,
lpSubKey);
}
CRegistryChecks RRegistry;
//
// Hook APIs.
//
LONG
APIHOOK(RegOpenKeyA)(
HKEY hKey,
LPSTR lpSubKey,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExA(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
MAXIMUM_ALLOWED,
NULL,
phkResult,
NULL,
FALSE);
}
LONG
APIHOOK(RegOpenKeyW)(
HKEY hKey,
LPWSTR lpSubKey,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExW(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
MAXIMUM_ALLOWED,
NULL,
phkResult,
NULL,
FALSE);
}
LONG
APIHOOK(RegOpenKeyExA)(
HKEY hKey,
LPCSTR lpSubKey,
DWORD ulOptions,
REGSAM samDesired,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExA(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
samDesired,
NULL,
phkResult,
NULL,
FALSE);
}
LONG
APIHOOK(RegOpenKeyExW)(
HKEY hKey,
LPCWSTR lpSubKey,
DWORD ulOptions,
REGSAM samDesired,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExW(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
samDesired,
NULL,
phkResult,
NULL,
FALSE);
}
LONG
APIHOOK(RegCreateKeyA)(
HKEY hKey,
LPCSTR lpSubKey,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExA(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
MAXIMUM_ALLOWED,
NULL,
phkResult,
NULL,
TRUE);
}
LONG
APIHOOK(RegCreateKeyW)(
HKEY hKey,
LPCWSTR lpSubKey,
PHKEY phkResult
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExW(
hKey,
lpSubKey,
0,
REG_OPTION_NON_VOLATILE,
MAXIMUM_ALLOWED,
NULL,
phkResult,
NULL,
TRUE);
}
LONG
APIHOOK(RegCreateKeyExA)(
HKEY hKey,
LPCSTR lpSubKey,
DWORD Reserved,
LPSTR lpClass,
DWORD dwOptions,
REGSAM samDesired,
LPSECURITY_ATTRIBUTES lpSecurityAttributes,
PHKEY phkResult,
LPDWORD lpdwDisposition
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExA(
hKey,
lpSubKey,
lpClass,
dwOptions,
samDesired,
lpSecurityAttributes,
phkResult,
lpdwDisposition,
TRUE);
}
LONG
APIHOOK(RegCreateKeyExW)(
HKEY hKey,
LPCWSTR lpSubKey,
DWORD Reserved,
LPWSTR lpClass,
DWORD dwOptions,
REGSAM samDesired,
LPSECURITY_ATTRIBUTES lpSecurityAttributes,
PHKEY phkResult,
LPDWORD lpdwDisposition
)
{
CLock cLock(g_csCritSec);
return RRegistry.OpenKeyExW(
hKey,
lpSubKey,
lpClass,
dwOptions,
samDesired,
lpSecurityAttributes,
phkResult,
lpdwDisposition,
TRUE);
}
LONG
APIHOOK(RegQueryValueA)(
HKEY hKey,
LPCSTR lpSubKey,
LPSTR lpValue,
PLONG lpcbValue
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryValueA(
hKey,
lpSubKey,
lpValue,
lpcbValue);
}
LONG
APIHOOK(RegQueryValueW)(
HKEY hKey,
LPCWSTR lpSubKey,
LPWSTR lpValue,
PLONG lpcbValue
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryValueW(
hKey,
lpSubKey,
lpValue,
lpcbValue);
}
LONG
APIHOOK(RegQueryValueExA)(
HKEY hKey,
LPCSTR lpValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryValueExA(
hKey,
lpValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
APIHOOK(RegQueryValueExW)(
HKEY hKey,
LPCWSTR lpValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryValueExW(
hKey,
lpValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
APIHOOK(RegQueryInfoKeyA)(
HKEY hKey,
LPSTR lpClass,
LPDWORD lpcbClass,
LPDWORD lpReserved,
LPDWORD lpcSubKeys,
LPDWORD lpcbMaxSubKeyLen,
LPDWORD lpcbMaxClassLen,
LPDWORD lpcValues,
LPDWORD lpcbMaxValueNameLen,
LPDWORD lpcbMaxValueLen,
LPDWORD lpcbSecurityDescriptor,
PFILETIME lpftLastWriteTime
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryInfoKeyA(
hKey,
lpClass,
lpcbClass,
lpReserved,
lpcSubKeys,
lpcbMaxSubKeyLen,
lpcbMaxClassLen,
lpcValues,
lpcbMaxValueNameLen,
lpcbMaxValueLen,
lpcbSecurityDescriptor,
lpftLastWriteTime);
}
LONG
APIHOOK(RegQueryInfoKeyW)(
HKEY hKey,
LPWSTR lpClass,
LPDWORD lpcbClass,
LPDWORD lpReserved,
LPDWORD lpcSubKeys,
LPDWORD lpcbMaxSubKeyLen,
LPDWORD lpcbMaxClassLen,
LPDWORD lpcValues,
LPDWORD lpcbMaxValueNameLen,
LPDWORD lpcbMaxValueLen,
LPDWORD lpcbSecurityDescriptor,
PFILETIME lpftLastWriteTime
)
{
CLock cLock(g_csCritSec);
return RRegistry.QueryInfoKeyW(
hKey,
lpClass,
lpcbClass,
lpReserved,
lpcSubKeys,
lpcbMaxSubKeyLen,
lpcbMaxClassLen,
lpcValues,
lpcbMaxValueNameLen,
lpcbMaxValueLen,
lpcbSecurityDescriptor,
lpftLastWriteTime);
}
LONG
APIHOOK(RegSetValueA)(
HKEY hKey,
LPCSTR lpSubKey,
DWORD dwType,
LPCSTR lpData,
DWORD cbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.SetValueA(
hKey,
lpSubKey,
dwType,
lpData,
cbData);
}
LONG
APIHOOK(RegSetValueW)(
HKEY hKey,
LPCWSTR lpSubKey,
DWORD dwType,
LPCWSTR lpData,
DWORD cbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.SetValueW(
hKey,
lpSubKey,
dwType,
lpData,
cbData);
}
LONG
APIHOOK(RegSetValueExA)(
HKEY hKey,
LPCSTR lpSubKey,
DWORD Reserved,
DWORD dwType,
CONST BYTE * lpData,
DWORD cbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.SetValueExA(
hKey,
lpSubKey,
Reserved,
dwType,
lpData,
cbData);
}
LONG
APIHOOK(RegSetValueExW)(
HKEY hKey,
LPCWSTR lpSubKey,
DWORD Reserved,
DWORD dwType,
CONST BYTE * lpData,
DWORD cbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.SetValueExW(
hKey,
lpSubKey,
Reserved,
dwType,
lpData,
cbData);
}
LONG
APIHOOK(RegEnumValueA)(
HKEY hKey,
DWORD dwIndex,
LPSTR lpValueName,
LPDWORD lpcbValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumValueA(
hKey,
dwIndex,
lpValueName,
lpcbValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
APIHOOK(RegEnumValueW)(
HKEY hKey,
DWORD dwIndex,
LPWSTR lpValueName,
LPDWORD lpcbValueName,
LPDWORD lpReserved,
LPDWORD lpType,
LPBYTE lpData,
LPDWORD lpcbData
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumValueW(
hKey,
dwIndex,
lpValueName,
lpcbValueName,
lpReserved,
lpType,
lpData,
lpcbData);
}
LONG
APIHOOK(RegEnumKeyA)(
HKEY hKey,
DWORD dwIndex,
LPSTR lpName,
DWORD cbName
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumKeyExA(
hKey,
dwIndex,
lpName,
&cbName,
NULL,
NULL,
NULL,
NULL); // can this be null???
}
LONG
APIHOOK(RegEnumKeyW)(
HKEY hKey,
DWORD dwIndex,
LPWSTR lpName,
DWORD cbName
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumKeyExW(
hKey,
dwIndex,
lpName,
&cbName,
NULL,
NULL,
NULL,
NULL);
}
LONG
APIHOOK(RegEnumKeyExA)(
HKEY hKey,
DWORD dwIndex,
LPSTR lpName,
LPDWORD lpcbName,
LPDWORD lpReserved,
LPSTR lpClass,
LPDWORD lpcbClass,
PFILETIME lpftLastWriteTime
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumKeyExA(
hKey,
dwIndex,
lpName,
lpcbName,
lpReserved,
lpClass,
lpcbClass,
lpftLastWriteTime);
}
LONG
APIHOOK(RegEnumKeyExW)(
HKEY hKey,
DWORD dwIndex,
LPWSTR lpName,
LPDWORD lpcbName,
LPDWORD lpReserved,
LPWSTR lpClass,
LPDWORD lpcbClass,
PFILETIME lpftLastWriteTime
)
{
CLock cLock(g_csCritSec);
return RRegistry.EnumKeyExW(
hKey,
dwIndex,
lpName,
lpcbName,
lpReserved,
lpClass,
lpcbClass,
lpftLastWriteTime);
}
LONG
APIHOOK(RegCloseKey)(HKEY hKey)
{
CLock cLock(g_csCritSec);
return RRegistry.CloseKey(hKey);
}
LONG
APIHOOK(RegDeleteKeyA)(
HKEY hKey,
LPCSTR lpSubKey
)
{
CLock cLock(g_csCritSec);
return RRegistry.DeleteKeyA(hKey, lpSubKey);
}
LONG
APIHOOK(RegDeleteKeyW)(
HKEY hKey,
LPCWSTR lpSubKey
)
{
CLock cLock(g_csCritSec);
return RRegistry.DeleteKeyW(hKey, lpSubKey);
}
SHIM_INFO_BEGIN()
SHIM_INFO_DESCRIPTION(AVS_REGISTRYCHECKS_DESC)
SHIM_INFO_FRIENDLY_NAME(AVS_REGISTRYCHECKS_FRIENDLY)
SHIM_INFO_VERSION(1, 3)
SHIM_INFO_FLAGS(AVRF_FLAG_EXTERNAL_ONLY)
SHIM_INFO_INCLUDE_EXCLUDE("E:msi.dll sxs.dll comctl32.dll ole32.dll oleaut32.dll")
SHIM_INFO_END()
/*++
Register hooked functions
Note we purposely ignore the cleanup because some apps call registry functions
during process detach.
--*/
HOOK_BEGIN
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_Console_READ,
AVS_HKCU_Console_READ,
AVS_HKCU_Console_READ_R,
AVS_HKCU_Console_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_ControlPanel_READ,
AVS_HKCU_ControlPanel_READ,
AVS_HKCU_ControlPanel_READ_R,
AVS_HKCU_ControlPanel_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_Environment_READ,
AVS_HKCU_Environment_READ,
AVS_HKCU_Environment_READ_R,
AVS_HKCU_Environment_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_Identities_READ,
AVS_HKCU_Identities_READ,
AVS_HKCU_Identities_READ_R,
AVS_HKCU_Identities_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_KeyboardLayout_READ,
AVS_HKCU_KeyboardLayout_READ,
AVS_HKCU_KeyboardLayout_READ_R,
AVS_HKCU_KeyboardLayout_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_Printers_READ,
AVS_HKCU_Printers_READ,
AVS_HKCU_Printers_READ_R,
AVS_HKCU_Printers_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_RemoteAccess_READ,
AVS_HKCU_RemoteAccess_READ,
AVS_HKCU_RemoteAccess_READ_R,
AVS_HKCU_RemoteAccess_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_SessionInformation_READ,
AVS_HKCU_SessionInformation_READ,
AVS_HKCU_SessionInformation_READ_R,
AVS_HKCU_SessionInformation_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_UNICODEProgramGroups_READ,
AVS_HKCU_UNICODEProgramGroups_READ,
AVS_HKCU_UNICODEProgramGroups_READ_R,
AVS_HKCU_UNICODEProgramGroups_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_VolatileEnvironment_READ,
AVS_HKCU_VolatileEnvironment_READ,
AVS_HKCU_VolatileEnvironment_READ_R,
AVS_HKCU_VolatileEnvironment_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCU_Windows31MigrationStatus_READ,
AVS_HKCU_Windows31MigrationStatus_READ,
AVS_HKCU_Windows31MigrationStatus_READ_R,
AVS_HKCU_Windows31MigrationStatus_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKLM_HARDWARE_READ,
AVS_HKLM_HARDWARE_READ,
AVS_HKLM_HARDWARE_READ_R,
AVS_HKLM_HARDWARE_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKLM_SAM_READ,
AVS_HKLM_SAM_READ,
AVS_HKLM_SAM_READ_R,
AVS_HKLM_SAM_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKLM_SECURITY_READ,
AVS_HKLM_SECURITY_READ,
AVS_HKLM_SECURITY_READ_R,
AVS_HKLM_SECURITY_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKLM_SYSTEM_READ,
AVS_HKLM_SYSTEM_READ,
AVS_HKLM_SYSTEM_READ_R,
AVS_HKLM_SYSTEM_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKCC_READ,
AVS_HKCC_READ,
AVS_HKCC_READ_R,
AVS_HKCC_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_HKUS_READ,
AVS_HKUS_READ,
AVS_HKUS_READ_R,
AVS_HKUS_READ_URL)
DUMP_VERIFIER_LOG_ENTRY(VLOG_NON_HKCU_WRITE,
AVS_NON_HKCU_WRITE,
AVS_NON_HKCU_WRITE_R,
AVS_NON_HKCU_WRITE_URL)
APIHOOK_ENTRY(ADVAPI32.DLL, RegOpenKeyA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegOpenKeyW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegOpenKeyExA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegOpenKeyExW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegCreateKeyA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegCreateKeyW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegCreateKeyExA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegCreateKeyExW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegCloseKey)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryValueA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryValueW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryValueExA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryValueExW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryInfoKeyA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegQueryInfoKeyW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegSetValueA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegSetValueW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegSetValueExA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegSetValueExW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumValueA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumValueW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumKeyA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumKeyW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumKeyExA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegEnumKeyExW)
APIHOOK_ENTRY(ADVAPI32.DLL, RegDeleteKeyA)
APIHOOK_ENTRY(ADVAPI32.DLL, RegDeleteKeyW)
HOOK_END
IMPLEMENT_SHIM_END