mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
346 lines
8.2 KiB
346 lines
8.2 KiB
//+--------------------------------------------------------------------------
|
|
// File: config.cpp
|
|
// Contents: CConfigStorage implements read/write to CA configuration data
|
|
// currently stored under HKLM\System\CCS\Services\Certsvc\
|
|
// Configuration
|
|
//---------------------------------------------------------------------------
|
|
#include <pch.cpp>
|
|
#include <config.h>
|
|
#pragma hdrstop
|
|
|
|
using namespace CertSrv;
|
|
|
|
HRESULT CConfigStorage::InitMachine(LPCWSTR pcwszMachine)
|
|
{
|
|
m_pwszMachine = new WCHAR[wcslen(pcwszMachine)+3];
|
|
if(!m_pwszMachine)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
m_pwszMachine[0] = L'\0';
|
|
|
|
if(pcwszMachine[0]!=L'\\' &&
|
|
pcwszMachine[1]!=L'\\')
|
|
{
|
|
wcscpy(m_pwszMachine, L"\\\\");
|
|
}
|
|
wcscat(m_pwszMachine, pcwszMachine);
|
|
return S_OK;
|
|
}
|
|
|
|
CConfigStorage::~CConfigStorage()
|
|
{
|
|
if(m_hRemoteHKLM)
|
|
RegCloseKey(m_hRemoteHKLM);
|
|
if(m_hRootConfigKey)
|
|
RegCloseKey(m_hRootConfigKey);
|
|
if(m_hCAKey)
|
|
RegCloseKey(m_hCAKey);
|
|
if(m_pwszMachine)
|
|
delete[] m_pwszMachine;
|
|
}
|
|
|
|
|
|
// Retrieve a CA configuration value. If no authority name is specified, the
|
|
// node path must be NULL and value is queried from the Configuration root.
|
|
// If an authority name is passed in, the value is retrieved from the authority
|
|
// node; if a node path is passed in, it is used relative to the authority node
|
|
// to read the value.
|
|
|
|
// For example, to read Configuration\DBDirectory, call:
|
|
//
|
|
// GetEntry(NULL, NULL, L"DBDirectory", &var)
|
|
//
|
|
// To read Configuration\MyCA\CAServerName, call:
|
|
//
|
|
// GetEntry(L"MyCA", NULL, L"CAServerName", &var)
|
|
//
|
|
// To read Configuration\MyCA\CSP\HashAlgorithm, call:
|
|
//
|
|
// GetEntry(L"MyCA", L"CSP", L"HashAlgorithm"
|
|
//
|
|
//
|
|
// If pcwszValue is null, getentry returns a VT_ARRAY|VT_BSTR with a list
|
|
// of subkey names.
|
|
|
|
HRESULT CConfigStorage::GetEntry(
|
|
LPCWSTR pcwszAuthorityName,
|
|
LPCWSTR pcwszRelativeNodePath,
|
|
LPCWSTR pcwszValue,
|
|
VARIANT *pVariant)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HKEY hKey = NULL;
|
|
BOOL fRet;
|
|
LPBYTE pData = NULL, pTmp;
|
|
DWORD cData = 0;
|
|
HKEY hKeyTmp = NULL;
|
|
DWORD dwType;
|
|
DWORD nIndex;
|
|
DWORD cName;
|
|
DWORD cKeys;
|
|
|
|
if(EmptyString(pcwszAuthorityName))
|
|
{
|
|
if(!EmptyString(pcwszRelativeNodePath))
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "CConfigStorage::GetEntry");
|
|
}
|
|
|
|
hr = InitRootKey();
|
|
_JumpIfError(hr, error, "CConfigStorage::InitRootKey");
|
|
|
|
hKey = m_hRootConfigKey;
|
|
}
|
|
else
|
|
{
|
|
hr = InitCAKey(pcwszAuthorityName);
|
|
_JumpIfError(hr, error, "CConfigStorage::InitCAKey");
|
|
|
|
hKey = m_hCAKey;
|
|
}
|
|
|
|
CSASSERT(hKey);
|
|
|
|
if(!EmptyString(pcwszRelativeNodePath))
|
|
{
|
|
hr = RegOpenKeyEx(
|
|
hKey,
|
|
pcwszRelativeNodePath,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hKeyTmp);
|
|
_JumpIfError(hr, error, "RegOpenKeyEx");
|
|
hKey = hKeyTmp;
|
|
}
|
|
|
|
if(EmptyString(pcwszValue))
|
|
{
|
|
dwType = REG_MULTI_SZ;
|
|
cData = 2;
|
|
|
|
hr = RegQueryInfoKey(
|
|
hKey,
|
|
NULL,NULL,NULL,
|
|
&cKeys,
|
|
&cName,
|
|
NULL,NULL,NULL,NULL,NULL,NULL);
|
|
_JumpIfError(hr, error, "RegQueryInfoKey");
|
|
|
|
cData = (cName+1)*cKeys*sizeof(WCHAR);
|
|
pData = (LPBYTE)LocalAlloc(LMEM_FIXED, cData);
|
|
if(!pData)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
|
|
pTmp = pData;
|
|
|
|
for(nIndex=0;nIndex<cKeys; nIndex++)
|
|
{
|
|
cName = cData;
|
|
hr = RegEnumKeyEx(
|
|
hKey,
|
|
nIndex,
|
|
(LPWSTR)pTmp,
|
|
&cName,
|
|
0, NULL, NULL, NULL);
|
|
_JumpIfError(hr, error, "RegEnumKeyEx");
|
|
pTmp = pTmp+(wcslen((LPWSTR)pTmp)+1)*sizeof(WCHAR);
|
|
}
|
|
|
|
*(LPWSTR)pTmp= L'\0';
|
|
|
|
hr = myRegValueToVariant(
|
|
dwType,
|
|
cData,
|
|
pData,
|
|
pVariant);
|
|
_JumpIfError(hr, error, "myRegValueToVariant");
|
|
}
|
|
else
|
|
{
|
|
hr = RegQueryValueEx(
|
|
hKey,
|
|
pcwszValue,
|
|
NULL,
|
|
&dwType,
|
|
NULL,
|
|
&cData);
|
|
_JumpIfError2(hr, error, "RegQueryValueEx", ERROR_FILE_NOT_FOUND);
|
|
|
|
pData = (LPBYTE)LocalAlloc(LMEM_FIXED, cData);
|
|
if(!pData)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
|
|
hr = RegQueryValueEx(
|
|
hKey,
|
|
pcwszValue,
|
|
NULL,
|
|
&dwType,
|
|
pData,
|
|
&cData);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
hr = myRegValueToVariant(
|
|
dwType,
|
|
cData,
|
|
pData,
|
|
pVariant);
|
|
_JumpIfError(hr, error, "myRegValueToVariant");
|
|
}
|
|
|
|
error:
|
|
if(hKeyTmp)
|
|
RegCloseKey(hKeyTmp);
|
|
if(pData)
|
|
LocalFree(pData);
|
|
return myHError(hr);
|
|
}
|
|
|
|
|
|
// If variant type is VT_EMPTY, SetEntry deletes the value. Otherwise it
|
|
// set the value, see myRegValueToVariant for supported types
|
|
HRESULT CConfigStorage::SetEntry(
|
|
LPCWSTR pcwszAuthorityName,
|
|
LPCWSTR pcwszRelativeNodePath,
|
|
LPCWSTR pcwszValue,
|
|
VARIANT *pVariant)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HKEY hKey = NULL;
|
|
BOOL fRet;
|
|
LPBYTE pData = NULL;
|
|
DWORD cData;
|
|
HKEY hKeyTmp = NULL;
|
|
DWORD dwType;
|
|
|
|
if(EmptyString(pcwszAuthorityName))
|
|
{
|
|
if(!EmptyString(pcwszRelativeNodePath))
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "CConfigStorage::GetEntry");
|
|
}
|
|
|
|
hr = InitRootKey();
|
|
_JumpIfError(hr, error, "CConfigStorage::InitRootKey");
|
|
|
|
hKey = m_hRootConfigKey;
|
|
}
|
|
else
|
|
{
|
|
hr = InitCAKey(pcwszAuthorityName);
|
|
_JumpIfError(hr, error, "CConfigStorage::InitCAKey");
|
|
|
|
hKey = m_hCAKey;
|
|
}
|
|
|
|
CSASSERT(hKey);
|
|
|
|
if(!EmptyString(pcwszRelativeNodePath))
|
|
{
|
|
hr = RegOpenKeyEx(
|
|
hKey,
|
|
pcwszRelativeNodePath,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hKeyTmp);
|
|
_JumpIfErrorStr(hr, error, "RegOpenKeyEx", pcwszRelativeNodePath);
|
|
hKey = hKeyTmp;
|
|
}
|
|
|
|
if(VT_EMPTY == V_VT(pVariant))
|
|
{
|
|
// delete value
|
|
hr = RegDeleteValue(
|
|
hKey,
|
|
pcwszValue);
|
|
_JumpIfErrorStr(hr, error, "RegDeleteValue", pcwszValue);
|
|
}
|
|
else
|
|
{
|
|
// set value
|
|
hr = myVariantToRegValue(
|
|
pVariant,
|
|
&dwType,
|
|
&cData,
|
|
&pData);
|
|
_JumpIfError(hr, error, "myVariantToRegValue");
|
|
|
|
hr = RegSetValueEx(
|
|
hKey,
|
|
pcwszValue,
|
|
NULL,
|
|
dwType,
|
|
pData,
|
|
cData);
|
|
_JumpIfErrorStr(hr, error, "RegSetValueEx", pcwszValue);
|
|
}
|
|
|
|
error:
|
|
if(hKeyTmp)
|
|
RegCloseKey(hKeyTmp);
|
|
if(pData)
|
|
LocalFree(pData);
|
|
|
|
return myHError(hr);
|
|
}
|
|
|
|
HRESULT CConfigStorage::InitRootKey()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if(!m_hRootConfigKey)
|
|
{
|
|
if(m_pwszMachine)
|
|
{
|
|
hr = RegConnectRegistry(
|
|
m_pwszMachine,
|
|
HKEY_LOCAL_MACHINE,
|
|
&m_hRemoteHKLM);
|
|
_JumpIfError(hr, error, "RegConnectRegistry");
|
|
|
|
|
|
}
|
|
|
|
hr = RegOpenKeyEx(
|
|
m_hRemoteHKLM?m_hRemoteHKLM:HKEY_LOCAL_MACHINE,
|
|
wszREGKEYCONFIGPATH,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&m_hRootConfigKey);
|
|
_JumpIfError(hr, error, "RegOpenKeyEx");
|
|
}
|
|
|
|
error:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CConfigStorage::InitCAKey(LPCWSTR pcwszAuthority)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if(!m_hCAKey)
|
|
{
|
|
hr = InitRootKey();
|
|
_JumpIfError(hr, error, "CConfigStorage::InitRootKey");
|
|
|
|
hr = RegOpenKeyEx(
|
|
m_hRootConfigKey,
|
|
pcwszAuthority,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&m_hCAKey);
|
|
_JumpIfError(hr, error, "RegOpenKeyEx");
|
|
}
|
|
|
|
error:
|
|
return hr;
|
|
}
|