windows-nt/Source/XPSP1/NT/ds/security/gina/snapins/gpedit/registry.cpp

1142 lines
29 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
#include "main.h"
const WCHAR g_cOpenBracket = L'[';
const WCHAR g_cCloseBracket = L']';
const WCHAR g_cSemiColon = L';';
#define TEMP_LOCATION TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Group Policy Objects")
#define REG_EVENT_NAME TEXT("Group Policy registry event name for ")
///////////////////////////////////////////////////////////////////////////////
// //
// CRegistryHive object implementation //
// //
///////////////////////////////////////////////////////////////////////////////
CRegistryHive::CRegistryHive()
{
m_cRef = 1;
InterlockedIncrement(&g_cRefThisDll);
m_hKey = NULL;
m_hEvent = NULL;
m_lpFileName = m_lpKeyName = m_lpEventName = NULL;
}
CRegistryHive::~CRegistryHive()
{
InterlockedDecrement(&g_cRefThisDll);
}
///////////////////////////////////////////////////////////////////////////////
// //
// CRegistryHive object implementation (IUnknown) //
// //
///////////////////////////////////////////////////////////////////////////////
HRESULT CRegistryHive::QueryInterface (REFIID riid, void **ppv)
{
if (IsEqualIID(riid, IID_IUnknown))
{
*ppv = (LPUNKNOWN)this;
m_cRef++;
return S_OK;
}
else
{
*ppv = NULL;
return E_NOINTERFACE;
}
}
ULONG CRegistryHive::AddRef (void)
{
return ++m_cRef;
}
ULONG CRegistryHive::Release (void)
{
if (--m_cRef == 0) {
if (m_hKey)
{
RegCloseKey (m_hKey);
if (m_lpKeyName)
{
BOOL bCleanRegistry = TRUE;
//
// We need to decide if the registry keys can be deleted or if
// they are in use by another copy of GPE. To do this, close
// the event we created during initialization and then try to
// open the event again. If the event is successfully opened,
// then another process is using the same registry key and it
// should not be deleted.
//
if (m_hEvent && m_lpEventName)
{
CloseHandle (m_hEvent);
m_hEvent = OpenEvent (SYNCHRONIZE, FALSE, m_lpEventName);
if (m_hEvent)
{
bCleanRegistry = FALSE;
}
}
if (bCleanRegistry)
{
RegDelnode (HKEY_CURRENT_USER, m_lpKeyName);
RegDeleteKey (HKEY_CURRENT_USER, TEMP_LOCATION);
}
}
}
if (m_lpKeyName)
{
LocalFree (m_lpKeyName);
}
if (m_lpFileName)
{
LocalFree (m_lpFileName);
}
if (m_lpEventName)
{
LocalFree (m_lpEventName);
}
if (m_hEvent)
{
CloseHandle (m_hEvent);
}
delete this;
return 0;
}
return m_cRef;
}
///////////////////////////////////////////////////////////////////////////////
// //
// CRegistryHive object implementation (Public functions) //
// //
///////////////////////////////////////////////////////////////////////////////
STDMETHODIMP CRegistryHive::Initialize(LPTSTR lpFileName, LPTSTR lpKeyName)
{
TCHAR szBuffer[300];
INT i;
LONG lResult;
DWORD dwDisp;
HRESULT hr;
//
// Check for null pointer
//
if (!lpFileName || !(*lpFileName))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Null filename")));
return E_INVALIDARG;
}
//
// Make sure this object hasn't been initialized already
//
if (m_hKey || m_lpFileName || m_lpKeyName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Object already initialized")));
return (E_UNEXPECTED);
}
//
// Find a temporary registry key to work with
//
lstrcpy (szBuffer, TEMP_LOCATION);
lstrcat (szBuffer, TEXT("\\"));
lstrcat (szBuffer, lpKeyName);
lResult = RegCreateKeyEx (HKEY_CURRENT_USER, szBuffer, 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS,
NULL, &m_hKey, &dwDisp);
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Failed to open registry key with %d"), lResult));
return (HRESULT_FROM_WIN32(lResult));
}
//
// Store the keyname
//
m_lpKeyName = (LPTSTR) LocalAlloc (LPTR, (lstrlen(szBuffer) + 1) * sizeof(TCHAR));
if (!m_lpKeyName)
{
RegCloseKey (m_hKey);
m_hKey = NULL;
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Failed to allocate memory for KeyName")));
return (HRESULT_FROM_WIN32(GetLastError()));
}
lstrcpy (m_lpKeyName, szBuffer);
//
// Store the filename
//
m_lpFileName = (LPTSTR) LocalAlloc (LPTR, (lstrlen(lpFileName) + 1) * sizeof(TCHAR));
if (!m_lpFileName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Failed to allocate memory for filename")));
return (HRESULT_FROM_WIN32(GetLastError()));
}
lstrcpy (m_lpFileName, lpFileName);
//
// Store the event name
//
lstrcpy (szBuffer, REG_EVENT_NAME);
lstrcat (szBuffer, lpKeyName);
m_lpEventName = (LPTSTR) LocalAlloc (LPTR, (lstrlen(szBuffer) + 1) * sizeof(TCHAR));
if (!m_lpEventName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Failed to allocate memory for filename")));
return (HRESULT_FROM_WIN32(GetLastError()));
}
lstrcpy (m_lpEventName, szBuffer);
//
// Load the file if it exists
//
hr = Load();
if (FAILED(hr))
{
return hr;
}
//
// Create the registry event
//
m_hEvent = CreateEvent (NULL, FALSE, FALSE, m_lpEventName);
if (!m_hEvent)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Initialize: Failed to create registry event with %d"),
GetLastError()));
return (HRESULT_FROM_WIN32(GetLastError()));
}
return (S_OK);
}
STDMETHODIMP CRegistryHive::GetHKey(HKEY *hKey)
{
HRESULT hr = E_FAIL;
*hKey = NULL;
if (m_hKey)
{
if (DuplicateHandle (GetCurrentProcess(),
(HANDLE)m_hKey,
GetCurrentProcess(),
(LPHANDLE) hKey, 0,
TRUE, DUPLICATE_SAME_ACCESS))
{
hr = S_OK;
}
else
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
}
return hr;
}
STDMETHODIMP CRegistryHive::Save(VOID)
{
HANDLE hFile;
HRESULT hr;
DWORD dwTemp, dwBytesWritten;
LPWSTR lpKeyName;
//
// Check parameters
//
if (!m_hKey || !m_lpFileName || !m_lpKeyName)
{
return E_INVALIDARG;
}
//
// Allocate a buffer to hold the keyname
//
lpKeyName = (LPWSTR) LocalAlloc (LPTR, MAX_KEYNAME_SIZE * sizeof(WCHAR));
if (!lpKeyName)
{
return (HRESULT_FROM_WIN32(GetLastError()));
}
//
// Create the output file
//
hFile = CreateFile (m_lpFileName, GENERIC_WRITE, 0, NULL,
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
NULL);
if (hFile == INVALID_HANDLE_VALUE) {
LocalFree (lpKeyName);
return (HRESULT_FROM_WIN32(GetLastError()));
}
//
// Write the header block
//
// 2 DWORDS, signature (PReg) and version number and 2 newlines
//
dwTemp = REGFILE_SIGNATURE;
if (!WriteFile (hFile, &dwTemp, sizeof(dwTemp), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(dwTemp))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Save: Failed to write signature with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
dwTemp = REGISTRY_FILE_VERSION;
if (!WriteFile (hFile, &dwTemp, sizeof(dwTemp), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(dwTemp))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Save: Failed to write version number with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Exports the values / keys
//
hr = ExportKey (m_hKey, hFile, lpKeyName);
Exit:
//
// Finished
//
CloseHandle (hFile);
LocalFree (lpKeyName);
return hr;
}
STDMETHODIMP CRegistryHive::ExportKey(HKEY hKey, HANDLE hFile, LPWSTR lpKeyName)
{
HRESULT hr = S_OK;
DWORD dwBytesWritten, dwNameSize, dwDataSize, dwKeySize;
DWORD dwIndex, dwTemp1, dwTemp2, dwType, dwKeyCount = 0;
LONG lResult;
HKEY hSubKey;
LPWSTR lpValueName = NULL;
LPWSTR lpSubKeyName = NULL;
LPBYTE lpValueData = NULL;
LPWSTR lpEnd;
//
// Gather information about this key
//
lResult = RegQueryInfoKey (hKey, NULL, NULL, NULL, &dwKeyCount, &dwKeySize, NULL,
NULL, &dwNameSize, &dwDataSize, NULL, NULL);
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: Failed to query registry key information with %d"),
lResult));
return HRESULT_FROM_WIN32(lResult);
}
//
// Allocate buffers to work with
//
lpValueName = (LPWSTR) LocalAlloc (LPTR, (dwNameSize + 1) * sizeof(WCHAR));
if (!lpValueName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: Failed to alloc memory with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
lpValueData = (LPBYTE) LocalAlloc (LPTR, dwDataSize);
if (!lpValueData)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: Failed to alloc memory with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Enumerate the values and write them to the file
//
dwIndex = 0;
while (TRUE)
{
dwTemp1 = dwNameSize + 1;
dwTemp2 = dwDataSize;
*lpValueName = L'\0';
lResult = RegEnumValueW (hKey, dwIndex, lpValueName, &dwTemp1, NULL,
&dwType, lpValueData, &dwTemp2);
if (lResult == ERROR_NO_MORE_ITEMS)
break;
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: RegEnumValue failed with %d"),
lResult));
hr = HRESULT_FROM_WIN32(lResult);
goto Exit;
}
hr = WriteValue(hFile, lpKeyName, lpValueName, dwType, dwTemp2, lpValueData);
if (hr != S_OK)
goto Exit;
dwIndex++;
}
//
// If dwIndex is 0, this is an empty key. We need to special case this
// so the empty key is entered into the registry file when there are
// no subkeys under it.
//
if ((dwIndex == 0) && (dwKeyCount == 0) && (*lpKeyName))
{
hr = WriteValue(hFile, lpKeyName, lpValueName, REG_NONE, 0, lpValueData);
if (hr != S_OK)
goto Exit;
}
LocalFree (lpValueName);
lpValueName = NULL;
LocalFree (lpValueData);
lpValueData = NULL;
//
// Now process the sub keys
//
lpSubKeyName = (LPWSTR) LocalAlloc (LPTR, (dwKeySize + 1) * sizeof(WCHAR));
if (!lpSubKeyName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: Failed to alloc memory with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
dwIndex = 0;
if (*lpKeyName)
lpEnd = CheckSlash (lpKeyName);
else
lpEnd = lpKeyName;
while (TRUE)
{
dwTemp1 = dwKeySize + 1;
lResult = RegEnumKeyEx (hKey, dwIndex, lpSubKeyName, &dwTemp1,
NULL, NULL, NULL, NULL);
if (lResult == ERROR_NO_MORE_ITEMS)
break;
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: RegEnumKeyEx failed with %d"),
lResult));
hr = HRESULT_FROM_WIN32(lResult);
goto Exit;
}
lstrcpy (lpEnd, lpSubKeyName);
lResult = RegOpenKeyEx (hKey, lpSubKeyName, 0, KEY_READ, &hSubKey);
if (lResult == ERROR_SUCCESS)
{
hr = ExportKey (hSubKey, hFile, lpKeyName);
RegCloseKey (hSubKey);
if (hr != S_OK)
break;
}
else
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::ExportKey: RegOpenKeyEx failed with %d"),
lResult));
}
dwIndex++;
}
Exit:
if (lpValueName)
LocalFree (lpValueName);
if (lpValueData)
LocalFree (lpValueData);
if (lpSubKeyName)
LocalFree (lpSubKeyName);
return hr;
}
STDMETHODIMP CRegistryHive::WriteValue(HANDLE hFile, LPWSTR lpKeyName,
LPWSTR lpValueName, DWORD dwType,
DWORD dwDataLength, LPBYTE lpData)
{
HRESULT hr = S_OK;
DWORD dwBytesWritten;
DWORD dwTemp;
//
// Write the entry to the text file.
//
// Format:
//
// [keyname;valuename;type;datalength;data]
//
// open bracket
if (!WriteFile (hFile, &g_cOpenBracket, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write open bracket with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// key name
dwTemp = (lstrlen (lpKeyName) + 1) * sizeof (WCHAR);
if (!WriteFile (hFile, lpKeyName, dwTemp, &dwBytesWritten, NULL) ||
dwBytesWritten != dwTemp)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write key name with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// semicolon
if (!WriteFile (hFile, &g_cSemiColon, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// value name
dwTemp = (lstrlen (lpValueName) + 1) * sizeof (WCHAR);
if (!WriteFile (hFile, lpValueName, dwTemp, &dwBytesWritten, NULL) ||
dwBytesWritten != dwTemp)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write value name with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// semicolon
if (!WriteFile (hFile, &g_cSemiColon, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// type
if (!WriteFile (hFile, &dwType, sizeof(DWORD), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(DWORD))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write data type with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// semicolon
if (!WriteFile (hFile, &g_cSemiColon, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// data length
if (!WriteFile (hFile, &dwDataLength, sizeof(DWORD), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(DWORD))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write data type with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// semicolon
if (!WriteFile (hFile, &g_cSemiColon, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// data
if (!WriteFile (hFile, lpData, dwDataLength, &dwBytesWritten, NULL) ||
dwBytesWritten != dwDataLength)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write data with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
// close bracket
if (!WriteFile (hFile, &g_cCloseBracket, sizeof(WCHAR), &dwBytesWritten, NULL) ||
dwBytesWritten != sizeof(WCHAR))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::WriteValue: Failed to write close bracket with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
DebugMsg((DM_VERBOSE, TEXT("CRegistryHive::WriteValue: Successfully wrote: %s\\%s"),
lpKeyName, lpValueName));
Exit:
return hr;
}
STDMETHODIMP CRegistryHive::Load(VOID)
{
HANDLE hFile;
HRESULT hr = S_OK;
DWORD dwTemp, dwBytesRead, dwType, dwDataLength, dwDisp;
LPWSTR lpKeyName, lpValueName, lpTemp;
LPBYTE lpData;
WCHAR chTemp;
HKEY hSubKey;
LONG lResult;
//
// Check parameters
//
if (!m_hKey || !m_lpFileName || !m_lpKeyName)
{
return E_INVALIDARG;
}
//
// Open the registry file
//
hFile = CreateFile (m_lpFileName, GENERIC_READ, FILE_SHARE_READ, NULL,
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN,
NULL);
if (hFile == INVALID_HANDLE_VALUE) {
if (GetLastError() == ERROR_FILE_NOT_FOUND)
{
return S_OK;
}
else
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: CreateFile failed for <%s> with %d"),
m_lpFileName, GetLastError()));
return (HRESULT_FROM_WIN32(GetLastError()));
}
}
//
// Allocate buffers to hold the keyname, valuename, and data
//
lpKeyName = (LPWSTR) LocalAlloc (LPTR, MAX_KEYNAME_SIZE * sizeof(WCHAR));
if (!lpKeyName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to allocate memory with %d"),
GetLastError()));
CloseHandle (hFile);
return (HRESULT_FROM_WIN32(GetLastError()));
}
lpValueName = (LPWSTR) LocalAlloc (LPTR, MAX_VALUENAME_SIZE * sizeof(WCHAR));
if (!lpValueName)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to allocate memory with %d"),
GetLastError()));
LocalFree (lpKeyName);
CloseHandle (hFile);
return (HRESULT_FROM_WIN32(GetLastError()));
}
//
// Read the header block
//
// 2 DWORDS, signature (PReg) and version number and 2 newlines
//
if (!ReadFile (hFile, &dwTemp, sizeof(dwTemp), &dwBytesRead, NULL) ||
dwBytesRead != sizeof(dwTemp))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read signature with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
if (dwTemp != REGFILE_SIGNATURE)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Invalid file signature")));
hr = E_FAIL;
goto Exit;
}
if (!ReadFile (hFile, &dwTemp, sizeof(dwTemp), &dwBytesRead, NULL) ||
dwBytesRead != sizeof(dwTemp))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read version number with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
if (dwTemp != REGISTRY_FILE_VERSION)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Invalid file version")));
hr = E_FAIL;
goto Exit;
}
//
// Read the data
//
while (TRUE)
{
//
// Read the first character. It will either be a [ or the end
// of the file.
//
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
if (GetLastError() != ERROR_HANDLE_EOF)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read first character with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
}
goto Exit;
}
if ((dwBytesRead == 0) || (chTemp != L'['))
{
goto Exit;
}
//
// Read the keyname
//
lpTemp = lpKeyName;
while (TRUE)
{
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read keyname character with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
*lpTemp++ = chTemp;
if (chTemp == TEXT('\0'))
break;
}
//
// Read the semi-colon
//
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
if (GetLastError() != ERROR_HANDLE_EOF)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read first character with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
}
goto Exit;
}
if ((dwBytesRead == 0) || (chTemp != L';'))
{
goto Exit;
}
//
// Read the valuename
//
lpTemp = lpValueName;
while (TRUE)
{
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read valuename character with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
*lpTemp++ = chTemp;
if (chTemp == TEXT('\0'))
break;
}
//
// Read the semi-colon
//
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
if (GetLastError() != ERROR_HANDLE_EOF)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read first character with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
}
goto Exit;
}
if ((dwBytesRead == 0) || (chTemp != L';'))
{
goto Exit;
}
//
// Read the type
//
if (!ReadFile (hFile, &dwType, sizeof(DWORD), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read type with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Skip semicolon
//
if (!ReadFile (hFile, &dwTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to skip semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Read the data length
//
if (!ReadFile (hFile, &dwDataLength, sizeof(DWORD), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to data length with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Skip semicolon
//
if (!ReadFile (hFile, &dwTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to skip semicolon with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Allocate memory for data
//
lpData = (LPBYTE) LocalAlloc (LPTR, dwDataLength);
if (!lpData)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to allocate memory for data with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Read data
//
if (!ReadFile (hFile, lpData, dwDataLength, &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to read data with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
//
// Skip closing bracket
//
if (!ReadFile (hFile, &chTemp, sizeof(WCHAR), &dwBytesRead, NULL))
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to skip closing bracket with %d"),
GetLastError()));
hr = HRESULT_FROM_WIN32(GetLastError());
goto Exit;
}
if (chTemp != L']')
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Expected to find ], but found %c"),
chTemp));
hr = E_FAIL;
goto Exit;
}
//
// Save registry value
//
lResult = RegCreateKeyEx (m_hKey, lpKeyName, 0, NULL, REG_OPTION_NON_VOLATILE,
KEY_WRITE, NULL, &hSubKey, &dwDisp);
if (lResult == ERROR_SUCCESS)
{
if ((dwType == REG_NONE) && (dwDataLength == 0) &&
(*lpValueName == L'\0'))
{
lResult = ERROR_SUCCESS;
}
else
{
lResult = RegSetValueEx (hSubKey, lpValueName, 0, dwType,
lpData, dwDataLength);
}
RegCloseKey (hSubKey);
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to set value <%s> with %d"),
lpValueName, lResult));
hr = HRESULT_FROM_WIN32(lResult);
LocalFree (lpData);
goto Exit;
}
}
else
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::Load: Failed to open key <%s> with %d"),
lpKeyName, lResult));
hr = HRESULT_FROM_WIN32(lResult);
LocalFree (lpData);
goto Exit;
}
LocalFree (lpData);
}
Exit:
//
// Finished
//
CloseHandle (hFile);
LocalFree (lpKeyName);
LocalFree (lpValueName);
return hr;
}
STDMETHODIMP CRegistryHive::IsRegistryEmpty(BOOL *bEmpty)
{
HRESULT hr = ERROR_SUCCESS;
DWORD dwKeys, dwValues;
LONG lResult;
if (!m_hKey)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::IsRegistryEmpty: registry key not open yet")));
return HRESULT_FROM_WIN32(ERROR_NOT_READY);
}
lResult = RegQueryInfoKey(m_hKey, NULL, NULL, NULL, &dwKeys, NULL, NULL, &dwValues,
NULL, NULL, NULL, NULL);
if (lResult != ERROR_SUCCESS)
{
DebugMsg((DM_WARNING, TEXT("CRegistryHive::IsRegistryEmpty: RegQueryInfoKey failed with %d"),
lResult));
return HRESULT_FROM_WIN32(lResult);
}
if ((dwKeys == 0) && (dwValues == 0))
{
*bEmpty = TRUE;
DebugMsg((DM_VERBOSE, TEXT("CRegistryHive::IsRegistryEmpty: registry key is empty")));
}
else
{
*bEmpty = FALSE;
DebugMsg((DM_VERBOSE, TEXT("CRegistryHive::IsRegistryEmpty: registry key is not empty. Keys = %d, Values = %d"),
dwKeys, dwValues));
}
return S_OK;
}