Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

7544 lines
203 KiB

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1997 - 1999
//
// File: setuput.cpp
//
// Contents: Utility functions for OCM based setup.
//
// History: 04/20/97 JerryK Created
//
//-------------------------------------------------------------------------
#include "pch.cpp"
#pragma hdrstop
// ** C Runtime Includes
#include <sys/types.h>
#include <sys/stat.h>
// ** System Includes **
#include <lmaccess.h>
#include <lmapibuf.h>
#include "csdisp.h"
#include <shlobj.h>
#include <userenv.h>
#include <dsgetdc.h>
#include <sddl.h>
#include <winldap.h>
#include <autoenr.h>
#include <userenvp.h> // CreateLinkFile API
// ** security includes **
#include <aclapi.h>
// ** Application Includes **
#include "initcert.h"
#include "cscsp.h"
#include "cspenum.h"
#include "csldap.h"
#include "wizpage.h"
#include "websetup.h"
#include "certsrvd.h"
#include "regd.h"
#include "usecert.h"
#include "certmsg.h"
#include "dssetup.h"
#include "progress.h"
#include <certca.h>
#include "csprop.h"
#include "setupids.h"
#define __dwFILE__ __dwFILE_OCMSETUP_SETUPUT_CPP__
EXTERN_C const IID IID_IGPEInformation;
EXTERN_C const CLSID CLSID_GPESnapIn;
#define CERT_HALF_SECOND 500 // # of milliseconds in half second
#define CERT_MAX_ATTEMPT 2 * 60 * 2 // # of half seconds in 2 minutes
#define wszREQUESTVERINDPROGID L"CertSrv.Request"
#define wszREQUESTPROGID L"CertSrv.Request.1"
#define wszADMINVERINDPROGID L"CertSrv.Admin"
#define wszADMINPROGID L"CertSrv.Admin.1"
#define wszREQUESTFRIENDLYNAME L"CertSrv Request"
#define wszADMINFRIENDLYNAME L"CertSrv Admin"
#define wszCERTSRV L"CertSrv"
#define wszREGW3SCRIPTMAP L"System\\CurrentControlSet\\Services\\W3SVC\\Parameters\\Script Map"
#define wszHTTPS L"https://"
#define wszASPEXT L".asp"
#define wszHTTP L"http://"
#define wszDOTCERTEXT L".cer"
#define wszNEWLINE L"\n"
#define wszFILESC L"file://\\\\"
#define SZ_REGSVR32 L"regsvr32.exe"
#define SZ_REGSVR32_CERTCLI L"/i:i /n /s certcli.dll"
#define SZ_VERB_OPEN L"open"
// hardcoded shares
#define wszCERTENROLLURLPATH L"/CertEnroll/"
#define wszzREGSUBJECTTEMPLATEVALUE \
wszPROPEMAIL L"\0" \
wszPROPCOMMONNAME L"\0" \
wszATTRORGUNIT4 L"\0" \
wszPROPORGANIZATION L"\0" \
wszPROPLOCALITY L"\0" \
wszPROPSTATE L"\0" \
wszPROPDOMAINCOMPONENT L"\0" \
wszPROPCOUNTRY L"\0"
// Whistler SMIME extension (or any other CSP):
// SMIME Capabilities
// [1]SMIME Capability
// Object ID=1.2.840.113549.3.2 szOID_RSA_RC2CBC, 128 bit
// Parameters=02 02 00 80
// [2]SMIME Capability
// Object ID=1.2.840.113549.3.4 szOID_RSA_RC4, 128 bit
// Parameters=02 02 00 80
// [3]SMIME Capability
// Object ID=1.3.14.3.2.7 szOID_OIWSEC_desCBC
// [4]SMIME Capability
// Object ID=1.2.840.113549.3.7 szOID_RSA_DES_EDE3_CBC
//
#define wszzREGVALUEDEFAULTSMIME \
TEXT(szOID_RSA_RC2CBC) L",128" L"\0" \
TEXT(szOID_RSA_RC4) L",128" L"\0" \
TEXT(szOID_OIWSEC_desCBC) L"\0" \
TEXT(szOID_RSA_DES_EDE3_CBC) L"\0"
#ifdef CERTSRV_ENABLE_ALL_REGISTRY_DEFAULTS
# define wszREGSUBJECTALTNAMEVALUE L"EMail"
# define wszREGSUBJECTALTNAME2VALUE L"EMail"
#else
# define wszREGSUBJECTALTNAMEVALUE \
L"DISABLED: Set to EMail to set SubjectAltName extension to the email address"
# define wszREGSUBJECTALTNAME2VALUE \
L"DISABLED: Set to EMail to set SubjectAltName2 extension to the email address"
#endif
#define szNULL_SESSION_REG_LOCATION "System\\CurrentControlSet\\Services\\LanmanServer\\Parameters"
#define szNULL_SESSION_VALUE "NullSessionPipes"
#define wszDEFAULTSHAREDFOLDER L"\\CAConfig"
// globals
WCHAR *g_pwszArgvPath = NULL; // for installing from local directory
WCHAR *g_pwszNoService = NULL; // skip CreateService
WCHAR *g_pwszSanitizedChar = NULL; // take first char for sanitizing test
#if DBG_CERTSRV
WCHAR *g_pwszDumpStrings = NULL; // dump resource strings
#endif
BOOL g_fW3SvcRunning = FALSE;
WCHAR g_wszServicePath[MAX_PATH];
// Version-independent ProgID
// ProgID
WCHAR const g_wszCertAdmDotDll[] = L"certadm.dll";
WCHAR const g_wszCertCliDotDll[] = L"certcli.dll";
WCHAR const g_wszcertEncDotDll[] = L"certenc.dll";
WCHAR const g_wszCertXDSDotDll[] = L"certxds.dll";
WCHAR const g_wszCertIfDotDll[] = L"certif.dll";
WCHAR const g_wszCertPDefDotDll[] = L"certpdef.dll";
WCHAR const g_wszCertMMCDotDll[] = L"certmmc.dll";
WCHAR const g_wszCertSrvDotMsc[] = L"certsrv.msc";
WCHAR const g_wszSCrdEnrlDotDll[] = L"scrdenrl.dll";
WCHAR const g_wszCertReqDotExe[] = L"certreq.exe";
WCHAR const g_wszCertUtilDotExe[] = L"certutil.exe";
WCHAR const g_wszCertDBDotDll[] = L"certdb.dll";
WCHAR const g_wszCertViewDotDll[] = L"certview.dll";
WCHAR const g_wszCSBullDotGif[] = L"csbull.gif";
WCHAR const g_wszCSBackDotGif[] = L"csback.gif";
WCHAR const g_wszCSLogoDotGif[] = L"cslogo.gif";
CHAR const * const aszRegisterServer[] = {
"DllRegisterServer",
"DllUnregisterServer",
};
typedef struct _REGISTERDLL
{
WCHAR const *pwszDllName;
DWORD Flags;
} REGISTERDLL;
#define RD_SERVER 0x00000001 // Register on server
#define RD_CLIENT 0x00000002 // Register on client
#define RD_UNREGISTER 0x00000004 // Unegister on client & server
#define RD_WHISTLER 0x00000008 // Register must succeed on Whistler only
#define RD_SKIPUNREGPOLICY 0x00000010 // not unreg custom policy during upgrade
#define RD_SKIPUNREGEXIT 0x00000020 // not unreg custom exit during upgrade
#define RD_SKIPUNREGMMC 0x00000040 // bug# 38876
REGISTERDLL const g_aRegisterDll[] = {
{ g_wszCertAdmDotDll, RD_SERVER | RD_CLIENT },
{ g_wszCertCliDotDll, RD_SERVER | RD_CLIENT },
{ g_wszcertEncDotDll, RD_SERVER | RD_CLIENT | RD_UNREGISTER },
{ g_wszCertXDSDotDll, RD_SERVER | RD_UNREGISTER | RD_SKIPUNREGEXIT},
{ g_wszCertIfDotDll, RD_UNREGISTER },
{ g_wszCertPDefDotDll, RD_SERVER | RD_UNREGISTER | RD_SKIPUNREGPOLICY },
{ g_wszCertMMCDotDll, RD_SERVER | RD_UNREGISTER | RD_SKIPUNREGMMC },
{ g_wszSCrdEnrlDotDll, RD_SERVER | RD_CLIENT | RD_UNREGISTER | RD_WHISTLER },
{ g_wszCertDBDotDll, RD_SERVER | RD_UNREGISTER },
{ g_wszCertViewDotDll, RD_UNREGISTER },
{ NULL, 0 }
};
typedef struct _PROGRAMENTRY
{
UINT uiLinkName;
UINT uiGroupName;
UINT uiDescription;
DWORD csidl; // special folder index
WCHAR const *pwszExeName;
WCHAR const *pwszClientArgs;
WCHAR const *pwszServerArgs;
DWORD Flags;
} PROGRAMENTRY;
#define PE_SERVER 0x00000001 // Install on server
#define PE_CLIENT 0x00000002 // Install on client
#define PE_DELETEONLY 0x00000004 // Always delete
PROGRAMENTRY const g_aProgramEntry[] = {
{
IDS_STARTMENU_NEWCRL_LINKNAME, // uiLinkName
IDS_STARTMENU_CERTSERVER, // uiGroupName
0, // uiDescription
CSIDL_COMMON_PROGRAMS, // "All Users\Start Menu\Programs"
g_wszCertUtilDotExe, // pwszExeName
NULL, // pwszClientArgs
L"-crl -", // pwszServerArgs
PE_DELETEONLY | PE_SERVER, // Flags
},
{
IDS_STARTMENU_CERTHIER_LINKNAME, // uiLinkName
IDS_STARTMENU_CERTSERVER, // uiGroupName
0, // uiDescription
CSIDL_COMMON_PROGRAMS, // "All Users\Start Menu\Programs"
L"certhier.exe", // pwszExeName
NULL, // pwszClientArgs
NULL, // pwszServerArgs
PE_DELETEONLY | PE_SERVER, // Flags
},
{
IDS_STARTMENU_CERTREQ_LINKNAME, // uiLinkName
IDS_STARTMENU_CERTSERVER, // uiGroupName
0, // uiDescription
CSIDL_COMMON_PROGRAMS, // "All Users\Start Menu\Programs"
g_wszCertReqDotExe, // pwszExeName
NULL, // pwszClientArgs
NULL, // pwszServerArgs
PE_DELETEONLY | PE_CLIENT | PE_SERVER, // Flags
},
};
#define CPROGRAMENTRY ARRAYSIZE(g_aProgramEntry)
static char rgcCERT_NULL_SESSION[] = {0x43, 0x45, 0x52, 0x54, 0x00, 0x00};
// ** Prototypes **
HRESULT
UpgradeServerRegEntries(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp);
HRESULT
CreateServerRegEntries(
IN HWND hwnd,
IN BOOL fUpgrade,
IN PER_COMPONENT_DATA *pComp);
HRESULT
CreateWebClientRegEntries(
BOOL fUpgrade,
PER_COMPONENT_DATA *pComp);
HRESULT
UpgradeWebClientRegEntries(
PER_COMPONENT_DATA *pComp);
HRESULT
GetServerNames(
IN HWND hwnd,
IN HINSTANCE hInstance,
IN BOOL fUnattended,
OUT WCHAR **ppwszServerName,
OUT WCHAR **ppwszServerNameOld);
HRESULT
UpdateDomainAndUserName(
IN HWND hwnd,
IN OUT PER_COMPONENT_DATA *pComp);
HRESULT
RegisterAndUnRegisterDLLs(
IN DWORD Flags,
IN PER_COMPONENT_DATA *pComp,
IN HWND hwnd);
HRESULT RenameMiscTargets(HWND hwnd, PER_COMPONENT_DATA *pComp, BOOL fServer);
HRESULT DeleteProgramGroups(IN BOOL fAll, IN PER_COMPONENT_DATA *pComp);
HRESULT CreateCertificateService(PER_COMPONENT_DATA *pComp, HWND hwnd);
HRESULT DeleteCertificates(const WCHAR *, BOOL fRoot);
HRESULT TriggerAutoenrollment();
//endproto
#ifdef DBG_OCM_TRACE
VOID
CaptureStackBackTrace(
EXCEPTION_POINTERS *pep,
ULONG cSkip,
ULONG cFrames,
ULONG *aeip)
{
ZeroMemory(aeip, cFrames * sizeof(aeip[0]));
#if i386 == 1
ULONG ieip, *pebp;
ULONG *pebpMax = (ULONG *) MAXLONG; // 2 * 1024 * 1024 * 1024; // 2 gig - 1
ULONG *pebpMin = (ULONG *) (64 * 1024); // 64k
if (pep == NULL)
{
ieip = 0;
cSkip++; // always skip current frame
pebp = ((ULONG *) &pep) - 2;
}
else
{
ieip = 1;
CSASSERT(cSkip == 0);
aeip[0] = pep->ContextRecord->Eip;
pebp = (ULONG *) pep->ContextRecord->Ebp;
}
if (pebp >= pebpMin && pebp < pebpMax)
{
__try
{
for ( ; ieip < cSkip + cFrames; ieip++)
{
if (ieip >= cSkip)
{
aeip[ieip - cSkip] = *(pebp + 1); // save an eip
}
ULONG *pebpNext = (ULONG *) *pebp;
if (pebpNext < pebp + 2 || pebpNext >= pebpMax - 1)
{
break;
}
pebp = pebpNext;
}
}
__except(EXCEPTION_EXECUTE_HANDLER)
{
;
}
}
#endif // i386 == 1
}
#endif // DBG_OCM_TRACE
VOID
DumpBackTrace(char const *pszName)
{
#ifdef DBG_OCM_TRACE
ULONG aeip[10];
DBGPRINT((MAXDWORD, "%hs: BackTrace:\n", pszName));
CaptureStackBackTrace(NULL, 1, ARRAYSIZE(aeip), aeip);
for (int i = 0; i < ARRAYSIZE(aeip); i++)
{
if (NULL == aeip[i])
{
break;
}
DBGPRINT((MAXDWORD, "ln %x;", aeip[i]));
}
DBGPRINT((MAXDWORD, "\n"));
#endif // DBG_OCM_TRACE
}
__inline VOID
AppendBackSlash(
IN OUT WCHAR *pwszOut)
{
DWORD cwc = wcslen(pwszOut);
if (0 == cwc || L'\\' != pwszOut[cwc - 1])
{
pwszOut[cwc++] = L'\\';
pwszOut[cwc] = L'\0';
}
}
__inline VOID
StripBackSlash(
IN OUT WCHAR *pwszOut)
{
DWORD cwc = wcslen(pwszOut);
if (0 < cwc && L'\\' == pwszOut[cwc - 1])
{
pwszOut[cwc] = L'\0';
}
}
VOID
BuildPath(
OUT WCHAR *pwszOut,
IN DWORD cwcOut,
IN WCHAR const *pwszDir,
IN WCHAR const *pwszFile)
{
wcscpy(pwszOut, pwszDir);
AppendBackSlash(pwszOut);
wcscat(pwszOut, pwszFile);
StripBackSlash(pwszOut);
CSASSERT(wcslen(pwszOut) < cwcOut);
DBGPRINT((DBG_SS_CERTOCMI, "BuildPath(%ws, %ws) -> %ws\n", pwszDir, pwszFile, pwszOut));
}
VOID
FreeCARequestInfo(CASERVERSETUPINFO *pServer)
{
if (NULL != pServer->pwszRequestFile)
{
LocalFree(pServer->pwszRequestFile);
}
if (NULL != pServer->pwszParentCAMachine)
{
LocalFree(pServer->pwszParentCAMachine);
}
if (NULL != pServer->pwszParentCAName)
{
LocalFree(pServer->pwszParentCAName);
}
}
VOID
FreeCAStoreInfo(CASERVERSETUPINFO *pServer)
{
if (NULL != pServer->pwszSharedFolder)
{
LocalFree(pServer->pwszSharedFolder);
}
if (NULL != pServer->pwszDBDirectory)
{
LocalFree(pServer->pwszDBDirectory);
}
if (NULL != pServer->pwszLogDirectory)
{
LocalFree(pServer->pwszLogDirectory);
}
}
VOID
FreeCAServerAdvanceInfo(CASERVERSETUPINFO *pServer)
{
if (NULL != pServer->pCSPInfoList)
{
FreeCSPInfoList(pServer->pCSPInfoList);
}
if (NULL != pServer->pKeyList)
{
csiFreeKeyList(pServer->pKeyList);
}
if (NULL != pServer->pDefaultCSPInfo)
{
freeCSPInfo(pServer->pDefaultCSPInfo);
}
if (NULL != pServer->pwszDesanitizedKeyContainerName)
{
LocalFree(pServer->pwszDesanitizedKeyContainerName);
}
if (NULL != pServer->pccExistingCert)
{
ClearExistingCertToUse(pServer);
}
if (NULL != pServer->pccUpgradeCert)
{
CertFreeCertificateContext(pServer->pccUpgradeCert);
}
if (NULL != pServer->pwszValidityPeriodCount)
{
LocalFree(pServer->pwszValidityPeriodCount);
}
if (NULL != pServer->pszAlgId)
{
LocalFree(pServer->pszAlgId);
}
if (NULL != pServer->hMyStore)
{
CertCloseStore(pServer->hMyStore, 0);
}
// don't free following because they are just pointers
// pServer->pCSPInfo
// pServer->pHashInfo
}
VOID
FreeCAServerIdInfo(
CASERVERSETUPINFO *pServer)
{
if (NULL != pServer->pwszCACommonName)
{
LocalFree(pServer->pwszCACommonName);
pServer->pwszCACommonName = NULL;
}
}
VOID
FreeCAServerInfo(CASERVERSETUPINFO *pServer)
{
FreeCAServerIdInfo(pServer);
FreeCAServerAdvanceInfo(pServer);
FreeCAStoreInfo(pServer);
FreeCARequestInfo(pServer);
if (NULL != pServer->pwszSanitizedName)
{
LocalFree(pServer->pwszSanitizedName);
}
if (NULL != pServer->pwszDNSuffix)
{
LocalFree(pServer->pwszDNSuffix);
}
if (NULL != pServer->pwszFullCADN)
{
LocalFree(pServer->pwszFullCADN);
}
if (NULL != pServer->pwszKeyContainerName)
{
LocalFree(pServer->pwszKeyContainerName);
}
if (NULL != pServer->pwszCACertFile)
{
LocalFree(pServer->pwszCACertFile);
}
if (NULL != pServer->pwszUseExistingCert)
{
LocalFree(pServer->pwszUseExistingCert);
}
if (NULL != pServer->pwszPreserveDB)
{
LocalFree(pServer->pwszPreserveDB);
}
if (NULL != pServer->pwszCustomPolicy)
{
LocalFree(pServer->pwszCustomPolicy);
}
if (NULL != pServer->pwszzCustomExit)
{
LocalFree(pServer->pwszzCustomExit);
}
}
VOID
FreeCAClientInfo(CAWEBCLIENTSETUPINFO *pClient)
{
if (NULL != pClient)
{
if (NULL != pClient->pwszWebCAMachine)
{
LocalFree(pClient->pwszWebCAMachine);
}
if (NULL != pClient->pwszWebCAName)
{
LocalFree(pClient->pwszWebCAName);
}
if (NULL != pClient->pwszSanitizedWebCAName)
{
LocalFree(pClient->pwszSanitizedWebCAName);
}
if (NULL != pClient->pwszSharedFolder)
{
LocalFree(pClient->pwszSharedFolder);
}
}
}
VOID
FreeCAInfo(CASETUPINFO *pCA)
{
if (NULL != pCA->pServer)
{
FreeCAServerInfo(pCA->pServer);
LocalFree(pCA->pServer);
pCA->pServer = NULL;
}
if (NULL != pCA->pClient)
{
FreeCAClientInfo(pCA->pClient);
LocalFree(pCA->pClient);
pCA->pClient = NULL;
}
}
VOID
FreeCAComponentInfo(PER_COMPONENT_DATA *pComp)
{
if (NULL != pComp->pwszCustomMessage)
{
LocalFree(pComp->pwszCustomMessage);
}
if (NULL != pComp->pwszComponent)
{
LocalFree(pComp->pwszComponent);
}
if (NULL != pComp->pwszUnattendedFile)
{
LocalFree(pComp->pwszUnattendedFile);
}
if (NULL != pComp->pwszServerName)
{
LocalFree(pComp->pwszServerName);
}
if (NULL != pComp->pwszServerNameOld)
{
LocalFree(pComp->pwszServerNameOld);
}
if (NULL != pComp->pwszSystem32)
{
LocalFree(pComp->pwszSystem32);
}
FreeCAInfo(&(pComp->CA));
}
VOID
FreeCAGlobals(VOID)
{
if (NULL != g_pwszArgvPath)
{
LocalFree(g_pwszArgvPath);
}
if (NULL != g_pwszNoService)
{
LocalFree(g_pwszNoService);
}
if (NULL != g_pwszSanitizedChar)
{
LocalFree(g_pwszSanitizedChar);
}
#if DBG_CERTSRV
if (NULL != g_pwszDumpStrings)
{
LocalFree(g_pwszDumpStrings);
}
#endif
}
VOID
SaveCustomMessage(
IN OUT PER_COMPONENT_DATA *pComp,
OPTIONAL IN WCHAR const *pwszCustomMessage)
{
HRESULT hr;
if (NULL != pwszCustomMessage)
{
if (NULL != pComp->pwszCustomMessage)
{
LocalFree(pComp->pwszCustomMessage);
pComp->pwszCustomMessage = NULL;
}
hr = myDupString(pwszCustomMessage, &pComp->pwszCustomMessage);
_JumpIfError(hr, error, "myDupString");
}
error:
;
}
HRESULT
LoadDefaultCAIDAttributes(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
DWORD i;
// free existing Id info before load default
FreeCAServerIdInfo(pServer);
// load default from resource
if (NULL != g_pwszSanitizedChar)
{
if (NULL != pServer->pwszCACommonName)
{
LocalFree(pServer->pwszCACommonName);
}
// replace with the env var
pServer->pwszCACommonName = (WCHAR*)LocalAlloc(LMEM_FIXED,
(wcslen(g_pwszSanitizedChar) + 1) * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pServer->pwszCACommonName);
wcscpy(pServer->pwszCACommonName, g_pwszSanitizedChar);
}
// default validity
pServer->enumValidityPeriod = dwVALIDITYPERIODENUMDEFAULT;
pServer->dwValidityPeriodCount = dwVALIDITYPERIODCOUNTDEFAULT_ROOT;
GetSystemTimeAsFileTime(&pServer->NotBefore);
pServer->NotAfter = pServer->NotBefore;
myMakeExprDateTime(
&pServer->NotAfter,
pServer->dwValidityPeriodCount,
pServer->enumValidityPeriod);
hr = S_OK;
error:
return(hr);
}
HRESULT
GetDefaultDBDirectory(
IN PER_COMPONENT_DATA *pComp,
OUT WCHAR **ppwszDir)
{
HRESULT hr;
DWORD cwc;
*ppwszDir = NULL;
cwc = wcslen(pComp->pwszSystem32) +
wcslen(wszLOGPATH) +
1;
*ppwszDir = (WCHAR *) LocalAlloc(LMEM_FIXED, cwc * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, *ppwszDir);
// default
wcscpy(*ppwszDir, pComp->pwszSystem32);
wcscat(*ppwszDir, wszLOGPATH);
CSASSERT(cwc == (DWORD) (wcslen(*ppwszDir) + 1));
hr = S_OK;
error:
return(hr);
}
HRESULT
LoadDefaultDBDirAttributes(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
if (NULL != pServer->pwszDBDirectory)
{
LocalFree(pServer->pwszDBDirectory);
pServer->pwszDBDirectory = NULL;
}
hr = GetDefaultDBDirectory(pComp, &pServer->pwszDBDirectory);
_JumpIfError(hr, error, "GetDefaultDBDirectory");
// default log dir is the same as db
if (NULL != pServer->pwszLogDirectory)
{
LocalFree(pServer->pwszLogDirectory);
}
pServer->pwszLogDirectory = (WCHAR *) LocalAlloc(
LMEM_FIXED,
(wcslen(pServer->pwszDBDirectory) + 1) * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pServer->pwszLogDirectory);
wcscpy(pServer->pwszLogDirectory, pServer->pwszDBDirectory);
pServer->fPreserveDB = FALSE;
hr = S_OK;
error:
return(hr);
}
HRESULT
LoadDefaultAdvanceAttributes(
IN OUT CASERVERSETUPINFO* pServer)
{
HRESULT hr;
// load default csp, ms base csp
pServer->fAdvance = FALSE;
if (NULL == pServer->pDefaultCSPInfo)
{
pServer->pDefaultCSPInfo = newCSPInfo(PROV_RSA_FULL, wszBASECSP);
if (NULL == pServer->pDefaultCSPInfo && !IsWhistler())
{
pServer->pDefaultCSPInfo = newCSPInfo(PROV_RSA_FULL, MS_DEF_PROV_W);
}
_JumpIfOutOfMemory(hr, error, pServer->pDefaultCSPInfo);
}
// determine default hash, sha1
pServer->pDefaultHashInfo = pServer->pDefaultCSPInfo->pHashList;
while (NULL != pServer->pDefaultHashInfo)
{
if (pServer->pDefaultHashInfo->idAlg == CALG_SHA1)
{
//got default
break;
}
pServer->pDefaultHashInfo = pServer->pDefaultHashInfo->next;
}
// If we have not just created a default key, reset the key container name.
if (pServer->pCSPInfo != pServer->pDefaultCSPInfo ||
(pServer->dwKeyLength != CA_DEFAULT_KEY_LENGTH_ROOT &&
pServer->dwKeyLength != CA_DEFAULT_KEY_LENGTH_SUB) ||
!pServer->fDeletableNewKey) {
ClearKeyContainerName(pServer);
}
// ok, point to defaults
pServer->pCSPInfo = pServer->pDefaultCSPInfo;
pServer->pHashInfo = pServer->pDefaultHashInfo;
// some other related defaults
pServer->dwKeyLength = IsRootCA(pServer->CAType)?
CA_DEFAULT_KEY_LENGTH_ROOT:
CA_DEFAULT_KEY_LENGTH_SUB;
pServer->dwKeyLenMin = 0;
pServer->dwKeyLenMax = 0;
// update hash oid
if (NULL != pServer->pszAlgId)
{
// free old
LocalFree(pServer->pszAlgId);
}
hr = myGetSigningOID(
NULL, // hProv
pServer->pCSPInfo->pwszProvName,
pServer->pCSPInfo->dwProvType,
pServer->pHashInfo->idAlg,
&(pServer->pszAlgId));
_JumpIfError(hr, error, "myGetSigningOID");
error:
return(hr);
}
HRESULT
LoadDefaultCAClientAttributes(
IN HWND hwnd,
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CAWEBCLIENTSETUPINFO *pClient = pComp->CA.pClient;
if (NULL != pClient)
{
// free existing client setup info
FreeCAClientInfo(pClient);
LocalFree(pClient);
pComp->CA.pClient = NULL;
}
pComp->CA.pClient = (CAWEBCLIENTSETUPINFO *) LocalAlloc(
LMEM_FIXED | LMEM_ZEROINIT,
sizeof(CAWEBCLIENTSETUPINFO));
_JumpIfOutOfMemory(hr, error, pComp->CA.pClient);
pComp->CA.pClient->WebCAType = ENUM_UNKNOWN_CA;
hr = S_OK;
error:
return(hr);
}
HRESULT
GetDefaultSharedFolder(
OUT WCHAR **ppwszSharedFolder)
{
HRESULT hr = S_OK;
WCHAR *pwszSysDrive = NULL;
*ppwszSharedFolder = NULL;
hr = myGetEnvString(&pwszSysDrive, L"SystemDrive");
if (S_OK == hr)
{
*ppwszSharedFolder = (WCHAR *) LocalAlloc(
LMEM_FIXED,
(wcslen(pwszSysDrive) + wcslen(wszDEFAULTSHAREDFOLDER) + 1) *
sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, *ppwszSharedFolder);
wcscpy(*ppwszSharedFolder, pwszSysDrive);
wcscat(*ppwszSharedFolder, wszDEFAULTSHAREDFOLDER);
}
error:
if (NULL != pwszSysDrive)
{
LocalFree(pwszSysDrive);
}
return hr;
}
HRESULT
LoadDefaultCAServerAttributes(
IN HWND hwnd,
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
BOOL fDSCA = FALSE;
WCHAR *pwszSysDrive = NULL;
bool fIsDomainMember;
bool fUserCanInstallCA;
if (NULL != pComp->CA.pServer)
{
// free existing server setup info
FreeCAServerInfo(pComp->CA.pServer);
LocalFree(pComp->CA.pServer);
}
// allocate server info buffer
pComp->CA.pServer = (CASERVERSETUPINFO *) LocalAlloc(
LMEM_FIXED | LMEM_ZEROINIT,
sizeof(CASERVERSETUPINFO));
_JumpIfOutOfMemory(hr, error, pComp->CA.pServer);
hr = LoadDefaultCAIDAttributes(pComp);
_JumpIfError(hr, error, "LoadDefaultCAIDAttributes");
hr = LoadDefaultAdvanceAttributes(pComp->CA.pServer);
_JumpIfError(hr, error, "LoadDefaultAdvanceAttributes");
hr = LoadDefaultDBDirAttributes(pComp);
_JumpIfError(hr, error, "LoadDefaultDBDirAttributes");
// decide default using DS
// xtan, the following call should be replaced with HasDSWritePermission()
// remove DisableEnterpriseCAs()
pComp->CA.pServer->fUseDS = FALSE;
hr = LocalMachineIsDomainMember(&fIsDomainMember);
_JumpIfError(hr, error, "LocalMachineIsDomainMember");
if (fIsDomainMember)
{
if(IsDSAvailable())
{
hr = CurrentUserCanInstallCA(fUserCanInstallCA);
_JumpIfError(hr, error, "CurrentUserCanInstallCA");
if(fUserCanInstallCA)
{
pComp->CA.pServer->fUseDS = TRUE;
fDSCA = csiIsAnyDSCAAvailable();
}
}
}
// alway free and null old shared folder
if (NULL != pComp->CA.pServer->pwszSharedFolder)
{
LocalFree(pComp->CA.pServer->pwszSharedFolder);
pComp->CA.pServer->pwszSharedFolder = NULL;
}
// decide default CA type and default shared folder
pComp->CA.pServer->CAType = ENUM_STANDALONE_ROOTCA;
if (pComp->CA.pServer->fUseDS)
{
if (fDSCA)
{
pComp->CA.pServer->CAType = ENUM_ENTERPRISE_SUBCA;
}
else
{
pComp->CA.pServer->CAType = ENUM_ENTERPRISE_ROOTCA;
}
}
if (pComp->fUnattended || !pComp->CA.pServer->fUseDS)
{
BOOL fChangeToDefault = FALSE;
// try reg load first
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGDIRECTORY,
&pComp->CA.pServer->pwszSharedFolder);
if (S_OK != hr && HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegStrValue", wszREGDIRECTORY);
}
if (S_OK == hr)
{
if (L'\0' == *pComp->CA.pServer->pwszSharedFolder)
{
// this mast be empty string
fChangeToDefault = TRUE;
}
else
{
//got something, make sure unc path exist
DWORD dwPathFlag;
if (!myIsFullPath(pComp->CA.pServer->pwszSharedFolder,
&dwPathFlag))
{
// somehow register an invalid path, don't use it
fChangeToDefault = TRUE;
}
else
{
if (UNC_PATH == dwPathFlag &&
DE_DIREXISTS != DirExists(pComp->CA.pServer->pwszSharedFolder))
{
// this unc path doesn't exist any more
// not making any sense to use it
fChangeToDefault = TRUE;
pComp->CA.pServer->fUNCPathNotFound = TRUE;
}
}
}
}
else
{
//must be not found
fChangeToDefault = TRUE;
}
if (fChangeToDefault)
{
//free 1st
if (NULL != pComp->CA.pServer->pwszSharedFolder)
{
LocalFree(pComp->CA.pServer->pwszSharedFolder);
}
// load default
hr = GetDefaultSharedFolder(&pComp->CA.pServer->pwszSharedFolder);
_JumpIfError(hr, error, "GetDefaultSharedFolder");
}
}
pComp->CA.pServer->fSaveRequestAsFile = FALSE;
pComp->CA.pServer->pwszRequestFile = NULL;
pComp->CA.pServer->pwszParentCAMachine = NULL;
pComp->CA.pServer->pwszParentCAName = NULL;
hr = S_OK;
error:
return hr;
}
HRESULT
InitCASetup(
IN HWND hwnd,
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
UINT ui;
bool fIsAdmin = false;
hr = GetServerNames(
hwnd,
pComp->hInstance,
pComp->fUnattended,
&pComp->pwszServerName,
&pComp->pwszServerNameOld);
_JumpIfError(hr, error, "GetServerNames");
DBGPRINT((
DBG_SS_CERTOCMI,
"InitCASetup:GetServerNames:%ws,%ws\n",
pComp->pwszServerName,
pComp->pwszServerNameOld));
DumpBackTrace("InitCASetup");
hr = IsCurrentUserBuiltinAdmin(&fIsAdmin);
_JumpIfError(hr, error, "IsCurrentUserBuiltinAdmin");
if (!fIsAdmin)
{
hr = E_ACCESSDENIED;
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_NOT_ADM,
0,
NULL);
_JumpError(hr, error, "IsCurrentUserBuiltinAdmin");
}
// load some of environment variables
hr = myGetEnvString(&g_pwszArgvPath, L"CertSrv_BinDir");
hr = myGetEnvString(&g_pwszNoService, L"CertSrv_NoService");
hr = myGetEnvString(&g_pwszSanitizedChar, L"CertSrv_Sanitize");
#if DBG_CERTSRV
myGetEnvString(&g_pwszDumpStrings, L"CertSrv_DumpStrings");
#endif
// figure out where the system root directory is (build path to x:\\winnt\system32\)
ui = GetSystemDirectory(NULL, 0); // returns chars neccessary to hold path (incl null)
if (ui == 0)
{
hr = myHLastError();
_JumpError(hr, error, "GetSystemDirectory");
}
pComp->pwszSystem32 = (LPWSTR)LocalAlloc(LMEM_FIXED, (ui+1)*sizeof(WCHAR));
if (NULL == pComp->pwszSystem32)
{
hr = E_OUTOFMEMORY;
_JumpError(hr, error, "LocalAlloc");
}
if (0 == GetSystemDirectory(pComp->pwszSystem32, ui))
{
hr = myHLastError();
_JumpError(hr, error, "GetSystemDirectory");
}
wcscat(pComp->pwszSystem32, L"\\");
// load default atrributes anyway
hr = LoadDefaultCAServerAttributes(hwnd, pComp);
_JumpIfError(hr, error, "LoadDefaultCAServerAttributes");
hr = LoadDefaultCAClientAttributes(hwnd, pComp);
_JumpIfError(hr, error, "LoadDefaultCAClientAttributes");
if (pComp->fUnattended)
{
// hook unattended data
hr = HookUnattendedServerAttributes(pComp,
LookupSubComponent(cscServer));
_JumpIfError(hr, error, "HookUnattendedServerAttributes");
hr = HookUnattendedClientAttributes(pComp,
LookupSubComponent(cscClient));
_JumpIfError(hr, error, "HookUnattendedClientAttributes");
}
hr = S_OK;
error:
return(hr);
}
HRESULT
CreateInitialCertificateRequest(
IN HCRYPTPROV hProv,
IN CASERVERSETUPINFO *pServer,
IN PER_COMPONENT_DATA *pComp,
IN HWND hwnd,
OUT BYTE **ppbEncode,
OUT DWORD *pcbEncode)
{
HRESULT hr;
BYTE *pbSubjectEncoded = NULL;
DWORD cbSubjectEncoded;
HINF hInf = INVALID_HANDLE_VALUE;
DWORD ErrorLine;
LPCWSTR pszErrorPtr;
hr = AddCNAndEncode(
pServer->pwszCACommonName,
pServer->pwszDNSuffix,
&pbSubjectEncoded,
&cbSubjectEncoded);
_JumpIfError(hr, error, "AddCNAndEncodeCertStrToName");
hr = myInfOpenFile(NULL, &hInf, &ErrorLine);
_PrintIfError2(
hr,
"myInfOpenFile",
HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND));
hr = csiBuildRequest(
hInf,
NULL,
pbSubjectEncoded,
cbSubjectEncoded,
pServer->pszAlgId,
TRUE, // fNewKey
CANAMEIDTOICERT(pServer->dwCertNameId),
CANAMEIDTOIKEY(pServer->dwCertNameId),
hProv,
hwnd,
pComp->hInstance,
pComp->fUnattended,
ppbEncode,
pcbEncode);
_JumpIfError(hr, error, "csiBuildRequest");
error:
if (INVALID_HANDLE_VALUE != hInf)
{
myInfCloseFile(hInf);
}
if (NULL != pbSubjectEncoded)
{
myFree(pbSubjectEncoded, CERTLIB_USE_LOCALALLOC);
}
CSILOG(hr, IDS_LOG_CREATE_REQUEST, NULL, NULL, NULL);
return(hr);
}
HRESULT
BuildCAHierarchy(
HCRYPTPROV hProv,
PER_COMPONENT_DATA *pComp,
CRYPT_KEY_PROV_INFO const *pKeyProvInfo,
HWND hwnd)
{
HRESULT hr;
BYTE *pbRequest = NULL;
DWORD cbRequest;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
BSTR bStrChain = NULL;
if (!pServer->fSaveRequestAsFile)
{
// online case
if (NULL == pServer->pwszParentCAMachine ||
NULL == pServer->pwszParentCAName)
{
hr = E_POINTER;
_JumpError(hr, error, "Empty machine name or parent ca name");
}
}
// create request 1st
hr = CreateInitialCertificateRequest(
hProv,
pServer,
pComp,
hwnd,
&pbRequest,
&cbRequest);
if (S_OK != hr)
{
pComp->iErrMsg = IDS_ERR_BUILDCERTREQUEST;
_JumpError(hr, error, "CreateInitialCertificateRequest");
}
// save it to a file always
hr = EncodeToFileW(
pServer->pwszRequestFile,
pbRequest,
cbRequest,
DECF_FORCEOVERWRITE | CRYPT_STRING_BASE64REQUESTHEADER);
_JumpIfError(hr, error, "EncodeToFileW");
// register request file name always
hr = mySetCARegFileNameTemplate(
wszREGREQUESTFILENAME,
pComp->pwszServerName,
pServer->pwszSanitizedName,
pServer->pwszRequestFile);
_JumpIfErrorStr(hr, error, "mySetCARegFileNameTemplate", wszREGREQUESTFILENAME);
if (pServer->fSaveRequestAsFile)
{
// mark it as request file
hr = SetSetupStatus(
pServer->pwszSanitizedName,
SETUP_SUSPEND_FLAG | SETUP_REQUEST_FLAG,
TRUE);
_JumpIfError(hr, error, "SetSetupStatus");
// done if save as request file
goto done;
}
hr = csiSubmitCARequest(
pComp->hInstance,
pComp->fUnattended,
hwnd,
FALSE, // fRenew
FALSE, // fRetrievePending
pServer->pwszSanitizedName,
pServer->pwszParentCAMachine,
pServer->pwszParentCAName,
pbRequest,
cbRequest,
&bStrChain);
// in any case, you can finish setup from mmc
_JumpIfError(hr, done, "csiSubmitCARequest");
hr = csiFinishInstallationFromPKCS7(
pComp->hInstance,
pComp->fUnattended,
hwnd,
pServer->pwszSanitizedName,
pServer->pwszCACommonName,
pKeyProvInfo,
pServer->CAType,
CANAMEIDTOICERT(pServer->dwCertNameId),
CANAMEIDTOIKEY(pServer->dwCertNameId),
pServer->fUseDS,
FALSE, // fRenew
pComp->pwszServerName,
(BYTE *) bStrChain,
SysStringByteLen(bStrChain),
pServer->pwszCACertFile);
_JumpIfError(hr, error, "csiFinishInstallationFromPKCS7");
done:
hr = S_OK;
error:
if (NULL != pbRequest)
{
myFree(pbRequest, CERTLIB_USE_LOCALALLOC);
}
if (NULL != bStrChain)
{
SysFreeString(bStrChain);
}
return(hr);
}
// Find the newest CA cert that:
// - matches the passed Subject DN,
// - matches the passed cert index,
// - expires prior the next newer cert (compare to pNotAfter)
// - expires latest of all that match the above
// - has KeyProvInfo
// - key and cert can be used together to sign
HRESULT
SetCARegOldCertHashByIndex(
IN WCHAR const *pwszSanitizedName,
IN HCERTSTORE hStore,
IN WCHAR const *pwszDN,
IN DWORD iCert,
IN OUT FILETIME *pNotAfter)
{
HRESULT hr;
CERT_CONTEXT const *pCert = NULL;
CERT_CONTEXT const *pCertNewest = NULL;
WCHAR *pwszDNT = NULL;
DWORD dwProvType;
WCHAR *pwszProvName = NULL;
ALG_ID idAlg;
BOOL fMachineKeyset;
DWORD dwNameId;
DWORD cbKey;
CRYPT_KEY_PROV_INFO *pKey = NULL;
hr = myGetCertSrvCSP(
FALSE, // fEncryptionCSP
pwszSanitizedName,
&dwProvType,
&pwszProvName,
&idAlg,
&fMachineKeyset,
NULL); // pdwKeySize
_JumpIfError(hr, error, "myGetCertSrvCSP");
while (TRUE)
{
if (NULL != pKey)
{
LocalFree(pKey);
pKey = NULL;
}
if (NULL != pwszDNT)
{
LocalFree(pwszDNT);
pwszDNT = NULL;
}
pCert = CertEnumCertificatesInStore(hStore, pCert);
if (NULL == pCert)
{
break;
}
hr = myCertNameToStr(
X509_ASN_ENCODING,
&pCert->pCertInfo->Subject,
CERT_X500_NAME_STR | CERT_NAME_STR_REVERSE_FLAG,
&pwszDNT);
if (S_OK != hr)
{
_PrintError(hr, "myCertNameToStr");
continue;
}
if (0 != lstrcmp(pwszDN, pwszDNT))
{
DBGPRINT((DBG_SS_CERTOCM, "Skipping cert: %ws\n", pwszDNT));
continue;
}
hr = myGetNameId(pCert, &dwNameId);
if (S_OK != hr ||
MAXDWORD == dwNameId ||
CANAMEIDTOICERT(dwNameId) != iCert)
{
DBGPRINT((DBG_SS_CERTOCM, "Skipping cert: NameId=%x\n", dwNameId));
continue;
}
DBGPRINT((DBG_SS_CERTOCM, "NameId=%x\n", dwNameId));
if (0 < CompareFileTime(&pCert->pCertInfo->NotAfter, pNotAfter))
{
DBGPRINT((DBG_SS_CERTOCM, "Skipping cert: too new\n"));
continue;
}
if (!myCertGetCertificateContextProperty(
pCert,
CERT_KEY_PROV_INFO_PROP_ID,
CERTLIB_USE_LOCALALLOC,
(VOID **) &pKey,
&cbKey))
{
hr = myHLastError();
_PrintError(hr, "CertGetCertificateContextProperty");
continue;
}
hr = myValidateHashForSigning(
pKey->pwszContainerName,
pwszProvName,
dwProvType,
fMachineKeyset,
&pCert->pCertInfo->SubjectPublicKeyInfo,
idAlg);
if (S_OK != hr)
{
_PrintError(hr, "myValidateHashForSigning");
continue;
}
if (NULL != pCertNewest)
{
if (0 > CompareFileTime(
&pCert->pCertInfo->NotAfter,
&pCertNewest->pCertInfo->NotAfter))
{
DBGPRINT((DBG_SS_CERTOCM, "Skipping cert: not newest\n"));
continue;
}
CertFreeCertificateContext(pCertNewest);
pCertNewest = NULL;
}
pCertNewest = CertDuplicateCertificateContext(pCert);
if (NULL == pCertNewest)
{
hr = myHLastError();
_JumpError(hr, error, "CertDuplicateCertificate");
}
}
if (NULL == pCertNewest)
{
hr = CRYPT_E_NOT_FOUND;
_JumpError(hr, error, "CertEnumCertificatesInStore");
}
// mark as unarchived:
CertSetCertificateContextProperty(
pCertNewest,
CERT_ARCHIVED_PROP_ID,
0,
NULL);
hr = mySetCARegHash(pwszSanitizedName, CSRH_CASIGCERT, iCert, pCertNewest);
_JumpIfError(hr, error, "mySetCARegHash");
*pNotAfter = pCertNewest->pCertInfo->NotAfter;
error:
if (NULL != pKey)
{
LocalFree(pKey);
}
if (NULL != pwszDNT)
{
LocalFree(pwszDNT);
}
if (NULL != pwszProvName)
{
LocalFree(pwszProvName);
}
if (NULL != pCertNewest)
{
CertFreeCertificateContext(pCertNewest);
}
if (NULL != pCert)
{
CertFreeCertificateContext(pCert);
}
return(hr);
}
HRESULT
SetCARegOldCertHashes(
IN WCHAR const *pwszSanitizedName,
IN DWORD cCertOld,
IN CERT_CONTEXT const *pccCA)
{
HRESULT hr;
HCERTSTORE hMyStore = NULL;
DWORD i;
WCHAR *pwszDN = NULL;
FILETIME NotAfter;
if (0 != cCertOld)
{
hr = myCertNameToStr(
X509_ASN_ENCODING,
&pccCA->pCertInfo->Subject,
CERT_X500_NAME_STR | CERT_NAME_STR_REVERSE_FLAG,
&pwszDN);
_JumpIfError(hr, error, "myCertNameToStr");
// open my store
hMyStore = CertOpenStore(
CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING,
NULL, // hProv
CERT_SYSTEM_STORE_LOCAL_MACHINE |
CERT_STORE_ENUM_ARCHIVED_FLAG,
wszMY_CERTSTORE);
if (NULL == hMyStore)
{
hr = myHLastError();
_JumpError(hr, error, "CertOpenStore");
}
NotAfter = pccCA->pCertInfo->NotAfter;
for (i = cCertOld; i > 0; i--)
{
hr = SetCARegOldCertHashByIndex(
pwszSanitizedName,
hMyStore,
pwszDN,
i - 1,
&NotAfter);
_PrintIfError(hr, "SetCARegOldCertHashByIndex");
}
}
hr = S_OK;
error:
if (NULL != pwszDN)
{
LocalFree(pwszDN);
}
if (NULL != hMyStore)
{
CertCloseStore(hMyStore, 0);
}
return(hr);
}
HRESULT
CreateCertificates(
IN OUT PER_COMPONENT_DATA *pComp,
IN HWND hwnd)
{
HRESULT hr;
HCRYPTPROV hCryptProv = NULL;
HCERTSTORE hStore = NULL;
CRYPT_KEY_PROV_INFO keyProvInfo;
WCHAR wszEnrollPath[MAX_PATH];
CERT_CONTEXT const *pccCA = NULL;
BYTE *pbEncoded = NULL;
WCHAR *pwszEnrollPath = NULL;
WCHAR *pwszDir = NULL;
WCHAR *pwszFolderPath = NULL;
DWORD dwSize;
wszEnrollPath[0] = L'\0';
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
ZeroMemory(&keyProvInfo, sizeof(keyProvInfo));
if (NULL == pServer->pwszKeyContainerName && pComp->fUnattended)
{
// create a new key if unattended
hr = csiGenerateCAKeys(
pServer->pwszSanitizedName,
pServer->pCSPInfo->pwszProvName,
pServer->pCSPInfo->dwProvType,
pServer->pCSPInfo->fMachineKeyset,
pServer->dwKeyLength,
pComp->hInstance,
pComp->fUnattended,
hwnd,
&pComp->CA.pServer->fKeyGenFailed);
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_FATAL_GENKEY,
hr,
pServer->pwszSanitizedName);
_JumpIfError(hr, error, "csiGenerateCAKeys");
}
// now set this as the existing key
hr = SetKeyContainerName(pServer, pServer->pwszSanitizedName);
_JumpIfError(hr, error, "SetKeyContainerName");
}
hr = csiFillKeyProvInfo(
pServer->pwszKeyContainerName,
pServer->pCSPInfo->pwszProvName,
pServer->pCSPInfo->dwProvType,
pServer->pCSPInfo->fMachineKeyset,
&keyProvInfo);
_JumpIfError(hr, error, "csiFillKeyProvInfo");
// get csp handle
if (!myCertSrvCryptAcquireContext(
&hCryptProv,
pServer->pwszKeyContainerName,
pServer->pCSPInfo->pwszProvName,
pServer->pCSPInfo->dwProvType,
pComp->fUnattended? CRYPT_SILENT : 0, // query
pServer->pCSPInfo->fMachineKeyset))
{
hr = myHLastError();
_JumpError(hr, error, "myCertSrvCryptAcquireContext");
}
if (hCryptProv == NULL)
{
hr = E_HANDLE;
_JumpError(hr, error, "myCertSrvCryptAcquireContext");
}
// open certificate store
if (NULL == pServer->hMyStore)
{
pServer->hMyStore = CertOpenStore(
CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING,
NULL, // hProv
CERT_SYSTEM_STORE_LOCAL_MACHINE |
CERT_STORE_ENUM_ARCHIVED_FLAG,
wszMY_CERTSTORE);
if (NULL == pServer->hMyStore)
{
// no store exists, done
hr = myHLastError();
_JumpIfError(hr, error, "CertOpenStore");
}
}
if (NULL != pServer->pccExistingCert)
{
// reuse cert, mark unarchived
CertSetCertificateContextProperty(
pServer->pccExistingCert,
CERT_ARCHIVED_PROP_ID,
0,
NULL);
}
if (IsSubordinateCA(pServer->CAType) && NULL == pServer->pccExistingCert)
{
hr = BuildCAHierarchy(hCryptProv, pComp, &keyProvInfo, hwnd);
_JumpIfError(hr, error, "BuildCAHierarchy");
}
else
{
WCHAR const *pwszCertName;
DWORD cwc;
BuildPath(
wszEnrollPath,
ARRAYSIZE(wszEnrollPath),
pComp->pwszSystem32,
wszCERTENROLLSHAREPATH);
hr = csiBuildFileName(
wszEnrollPath,
pServer->pwszSanitizedName,
L".crt",
CANAMEIDTOICERT(pServer->dwCertNameId),
&pwszEnrollPath,
pComp->hInstance,
pComp->fUnattended,
NULL);
_JumpIfError(hr, error, "csiBuildFileName");
CSASSERT(NULL != pServer->pwszCACertFile);
pwszCertName = wcsrchr(pServer->pwszCACertFile, L'\\');
CSASSERT(NULL != pwszCertName);
cwc = SAFE_SUBTRACT_POINTERS(pwszCertName, pServer->pwszCACertFile);
pwszDir = (WCHAR *) LocalAlloc(LMEM_FIXED, (cwc + 1) * sizeof(WCHAR));
if (NULL == pwszDir)
{
hr = E_OUTOFMEMORY;
_JumpError(hr, error, "LocalAlloc");
}
CopyMemory(pwszDir, pServer->pwszCACertFile, cwc * sizeof(WCHAR));
pwszDir[cwc] = L'\0';
hr = csiBuildFileName(
pwszDir,
pServer->pwszSanitizedName,
L".crt",
CANAMEIDTOICERT(pServer->dwCertNameId),
&pwszFolderPath,
pComp->hInstance,
pComp->fUnattended,
NULL);
_JumpIfError(hr, error, "csiBuildFileName");
// create and save a selfsigned root cert
hr = csiBuildAndWriteCert(
hCryptProv,
pServer,
pwszFolderPath,
pwszEnrollPath,
pServer->pccExistingCert, // if NULL, we will build a new cert
&pccCA,
wszCERTTYPE_CA,
pComp->hInstance,
pComp->fUnattended,
hwnd);
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_BUILDCERT,
hr,
NULL);
_JumpError(hr, error, "csiBuildAndWriteCert");
}
hr = SetCARegOldCertHashes(
pServer->pwszSanitizedName,
CANAMEIDTOICERT(pServer->dwCertNameId),
pccCA);
_JumpIfError(hr, error, "SetCARegOldCertHashes");
hr = mySetCARegHash(
pServer->pwszSanitizedName,
CSRH_CASIGCERT,
CANAMEIDTOICERT(pServer->dwCertNameId),
pccCA);
_JumpIfError(hr, error, "mySetCARegHash");
hr = csiSaveCertAndKeys(pccCA, NULL, &keyProvInfo, pServer->CAType);
_JumpIfError(hr, error, "csiSaveCertAndKeys");
if (pServer->fUseDS)
{
hr = csiSetupCAInDS(
hwnd,
pComp->pwszServerName,
pServer->pwszSanitizedName,
pServer->pwszCACommonName,
NULL,
pServer->CAType,
CANAMEIDTOICERT(pServer->dwCertNameId),
CANAMEIDTOIKEY(pServer->dwCertNameId),
FALSE, // fRenew
pccCA);
_PrintIfError(hr, "csiSetupCAInDS");
if (hr == S_OK)
pServer->fSavedCAInDS = TRUE;
}
}
hr = S_OK;
error:
csiFreeKeyProvInfo(&keyProvInfo);
if (NULL != pbEncoded)
{
LocalFree(pbEncoded);
}
if (NULL != pwszDir)
{
LocalFree(pwszDir);
}
if (NULL != pwszFolderPath)
{
LocalFree(pwszFolderPath);
}
if (NULL != pwszEnrollPath)
{
LocalFree(pwszEnrollPath);
}
if (NULL != pccCA)
{
CertFreeCertificateContext(pccCA);
}
if (NULL != hCryptProv)
{
CryptReleaseContext(hCryptProv, 0);
}
CSILOG(hr, IDS_LOG_CREATE_CERTIFICATE, NULL, NULL, NULL);
return(hr);
}
HRESULT
StartCertsrvService(BOOL fSilent)
{
HRESULT hr;
SC_HANDLE hSCManager = NULL;
SC_HANDLE hSCCertsvc = NULL;
SERVICE_STATUS status;
DWORD dwAttempt;
BOOL fSawPending;
WCHAR const *apwszSilentArg[1] = {L"-s"};
hSCManager = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
if (NULL == hSCManager)
{
hr = myHLastError();
_JumpError(hr, error, "OpenSCManager");
}
hSCCertsvc = OpenService(
hSCManager,
wszSERVICE_NAME,
SERVICE_ALL_ACCESS);
if (NULL == hSCCertsvc)
{
hr = myHLastError();
_JumpErrorStr(hr, error, "OpenService", wszSERVICE_NAME);
}
// START the service
if (!StartService(hSCCertsvc,
fSilent ? 1 : 0,
fSilent ? apwszSilentArg : NULL))
{
hr = myHLastError();
_JumpErrorStr(hr, error, "StartService", wszSERVICE_NAME);
}
// get out after it is really started
fSawPending = FALSE;
for (dwAttempt = 0; dwAttempt < CERT_MAX_ATTEMPT; dwAttempt++)
{
DBGCODE(status.dwCurrentState = -1);
if (!QueryServiceStatus(hSCCertsvc, &status))
{
// query failed, ignore error
hr = S_OK;
_JumpErrorStr(
myHLastError(), // Display ignored error
error,
"QueryServiceStatus",
wszSERVICE_NAME);
}
if (SERVICE_START_PENDING != status.dwCurrentState &&
SERVICE_STOPPED != status.dwCurrentState)
{
// it was started already
break;
}
DBGPRINT((
DBG_SS_CERTOCM,
"Starting %ws service: current state=%d\n",
wszSERVICE_NAME,
status.dwCurrentState));
if (fSawPending && SERVICE_STOPPED == status.dwCurrentState)
{
hr = HRESULT_FROM_WIN32(ERROR_SERVICE_NEVER_STARTED);
_JumpErrorStr(
hr,
error,
"Service won't start",
wszSERVICE_NAME);
}
if (SERVICE_START_PENDING == status.dwCurrentState)
{
fSawPending = TRUE;
}
Sleep(CERT_HALF_SECOND);
}
if (dwAttempt >= CERT_MAX_ATTEMPT)
{
DBGPRINT((
DBG_SS_CERTOCM,
"Timeout starting %ws service: current state=%d\n",
wszSERVICE_NAME,
status.dwCurrentState));
}
else
{
DBGPRINT((
DBG_SS_CERTOCM,
"Started %ws service\n",
wszSERVICE_NAME));
}
hr = S_OK;
error:
if (NULL != hSCCertsvc)
{
CloseServiceHandle(hSCCertsvc);
}
if (NULL != hSCManager)
{
CloseServiceHandle(hSCManager);
}
CSILOG(hr, IDS_LOG_START_SERVICE, NULL, NULL, NULL);
return(hr);
}
HRESULT
EnforceCertFileExtensions(
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
WCHAR *pwszTmp = NULL;
WCHAR *pwszSuffix;
BOOL fAppendExtension = TRUE;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
if (NULL == pServer->pwszCACertFile)
{
// no ca cert file
goto done;
}
// make enough to hold extra extension crt
pwszTmp = (WCHAR *) LocalAlloc(
LMEM_FIXED,
(wcslen(pServer->pwszCACertFile) + 5) * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pwszTmp);
wcscpy(pwszTmp, pServer->pwszCACertFile);
// check to make sure our self-signed file has the right extension
// Is there an extension?
pwszSuffix = wcsrchr(pwszTmp, L'.');
if (NULL != pwszSuffix)
{
// Is the stuff after the '.' already a 'crt' extension?
if (0 == lstrcmpi(pwszSuffix, L".crt"))
{
fAppendExtension = FALSE;
}
else if (pwszSuffix[1] == L'\0') // Is '.' last character?
{
while (pwszSuffix >= pwszTmp && *pwszSuffix == L'.')
{
*pwszSuffix-- = L'\0';
}
}
}
if (fAppendExtension)
{
// Apply the extension
wcscat(pwszTmp, L".crt");
// free old one
LocalFree(pServer->pwszCACertFile);
pServer->pwszCACertFile = pwszTmp;
pwszTmp = NULL;
}
done:
hr = S_OK;
error:
if (NULL != pwszTmp)
{
LocalFree(pwszTmp);
}
return(hr);
}
HRESULT
PrepareEDBDirectory(
HWND hwnd,
PER_COMPONENT_DATA *pComp,
WCHAR const *pwszDir)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
DWORD dwAttr = GetFileAttributes(pwszDir);
if(MAXDWORD==dwAttr)
{
// file not found or other error
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpError(hr, error, "CreateDirectory");
}
hr = S_OK;
if (!CreateDirectory(pwszDir, NULL))
{
hr = myHLastError();
_JumpError(hr, error, "CreateDirectory");
}
}
if(!(dwAttr&FILE_ATTRIBUTE_DIRECTORY))
{
// file already exists but it's not a directory
hr = HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS);
_JumpError(hr, error, "CreateDirectory");
}
if (!pServer->fPreserveDB)
{
hr = myDeleteDBFilesInDir(pwszDir);
_JumpIfError(hr, error, "myDeleteDBFilesInDir");
}
hr = S_OK;
error:
if(S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
0,
hr,
L""); // only message is the system error message
}
return(hr);
}
//--------------------------------------------------------------------
// Create the web configuration files
HRESULT
CreateCertWebIncPages(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp,
IN BOOL bIsServer,
IN BOOL fUpgrade)
{
HRESULT hr;
CSASSERT(NULL != pComp);
// create the web configuration file
hr = CreateCertWebDatIncPage(pComp, bIsServer);
_JumpIfError(hr, error, "CreateCertWebDatIncPage");
error:
CSILOG(hr, IDS_LOG_WEB_INCLUDE, NULL, NULL, NULL);
return hr;
}
//--------------------------------------------------------------------
HRESULT
EnableVRootsAndShares(
IN BOOL fFileSharesOnly,
IN BOOL fUpgrade,
IN BOOL fServer,
IN PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
DWORD Flags = VFF_CREATEFILESHARES |
VFF_SETREGFLAGFIRST |
VFF_SETRUNONCEIFERROR |
VFF_CLEARREGFLAGIFOK;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
CSASSERT(!fServer || NULL != pServer);
CSASSERT(fServer || NULL != pComp->CA.pClient);
if (!fFileSharesOnly)
{
Flags |= VFF_CREATEVROOTS;
}
// if NT GUI mode (base setup) VRoot creation will fail during setup
// because IIS is not yet operational. Make a mark in the registry
// to try this on NEXT service startup or via runonce.
hr = myModifyVirtualRootsAndFileShares(
Flags,
fServer? pServer->CAType : pComp->CA.pClient->WebCAType,
FALSE, // synchronous -- blocking call
VFCSEC_TIMEOUT,
NULL,
NULL);
_JumpIfError(hr, error, "myModifyVirtualRootsAndFileShares");
if (!fUpgrade)
{
pServer->fCreatedShare = TRUE;
if (!fFileSharesOnly)
{
pComp->fCreatedVRoot = TRUE;
}
}
error:
return(hr);
}
HRESULT
DisableVRootsAndShares(
IN BOOL fVRoot,
IN BOOL fFileShares)
{
HRESULT hr;
DWORD Flags = 0;
if (fVRoot)
{
Flags |= VFF_DELETEVROOTS;
}
if (fFileShares)
{
Flags |= VFF_DELETEFILESHARES;
}
if (0 == Flags)
{
goto done;
}
hr = myModifyVirtualRootsAndFileShares(
Flags,
ENUM_UNKNOWN_CA,
FALSE, // synchronous -- blocking call
VFCSEC_TIMEOUT,
NULL,
NULL);
_JumpIfError(hr, error, "myModifyVirtualRootsAndFileShares");
done:
hr = S_OK;
error:
return(hr);
}
HRESULT
InstallClient(
HWND hwnd,
PER_COMPONENT_DATA *pComp)
{
BOOL fCoInit = FALSE;
HRESULT hr;
hr = CoInitialize(NULL);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "CoInitialize");
}
fCoInit = TRUE;
certocmBumpGasGauge(pComp, 10 DBGPARM(L"InstallClient"));
hr = CreateWebClientRegEntries(FALSE, pComp);
_JumpIfError(hr, error, "CreateWebClientRegEntries");
certocmBumpGasGauge(pComp, 30 DBGPARM(L"InstallClient"));
hr = RegisterAndUnRegisterDLLs(RD_CLIENT, pComp, hwnd);
_JumpIfError(hr, error, "RegisterAndUnRegisterDLLs");
certocmBumpGasGauge(pComp, 50 DBGPARM(L"InstallClient"));
DeleteProgramGroups(FALSE, pComp);
hr = CreateProgramGroups(TRUE, pComp, hwnd);
_JumpIfError(hr, error, "CreateProgramGroups");
certocmBumpGasGauge(pComp, 70 DBGPARM(L"InstallClient"));
hr = CreateCertWebIncPages(hwnd, pComp, FALSE /*IsServer*/, FALSE);
_JumpIfError(hr, error, "CreateCertWebIncPages");
hr = RenameMiscTargets(hwnd, pComp, FALSE);
_JumpIfError(hr, error, "RenameMiscTargets");
certocmBumpGasGauge(pComp, 80 DBGPARM(L"InstallClient"));
hr = EnableVRootsAndShares(FALSE, FALSE, FALSE, pComp);
_JumpIfError(hr, error, "EnableVRootsAndShares");
certocmBumpGasGauge(pComp, 100 DBGPARM(L"InstallClient"));
error:
if (fCoInit)
{
CoUninitialize();
}
CSILOG(hr, IDS_LOG_INSTALL_CLIENT, NULL, NULL, NULL);
return(hr);
}
HRESULT
RemoveWebClientRegEntries(VOID)
{
HRESULT hr;
hr = myDeleteCertRegValue(NULL, NULL, NULL, wszREGWEBCLIENTCAMACHINE);
_PrintIfError(hr, "myDeleteCertRegValue");
hr = myDeleteCertRegValue(NULL, NULL, NULL, wszREGWEBCLIENTCANAME);
_PrintIfError(hr, "myDeleteCertRegValue");
hr = myDeleteCertRegValue(NULL, NULL, NULL, wszREGWEBCLIENTCATYPE);
_PrintIfError(hr, "myDeleteCertRegValue");
hr = S_OK;
//error:
return hr;
}
HRESULT
InstallServer(
HWND hwnd,
PER_COMPONENT_DATA *pComp)
{
BOOL fCoInit = FALSE;
WCHAR *pwszDBFile = NULL;
DWORD dwSetupStatus;
HRESULT hr = pComp->hrContinue;
int ret;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
WCHAR *pwszConfig = NULL;
BOOL fSetDSSecurity;
_JumpIfError(hr, error, "can't continue");
hr = UpdateDomainAndUserName(hwnd, pComp);
_JumpIfError(hr, error, "UpdateDomainAndUserName");
hr = CoInitialize(NULL);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "CoInitialize");
}
fCoInit = TRUE;
hr = EnforceCertFileExtensions(pComp);
_JumpIfError(hr, error, "EnforceCertFileExtensions");
hr = PrepareEDBDirectory(hwnd, pComp, pServer->pwszDBDirectory);
_JumpIfError(hr, error, "PrepareEDBDirectory");
hr = PrepareEDBDirectory(hwnd, pComp, pServer->pwszLogDirectory);
_JumpIfError(hr, error, "PrepareEDBDirectory");
certocmBumpGasGauge(pComp, 10 DBGPARM(L"InstallServer"));
// alway uninstall before install
PreUninstallCore(hwnd, pComp, FALSE);
UninstallCore(hwnd, pComp, 10, 30, FALSE, FALSE, FALSE);
hr = CreateServerRegEntries(hwnd, FALSE, pComp);
_JumpIfError(hr, error, "CreateServerRegEntries");
if ((IS_SERVER_INSTALL & pComp->dwInstallStatus) &&
(IS_CLIENT_UPGRADE & pComp->dwInstallStatus))
{
// case of install server only and keep web client
// remove parent ca config info of the old web client
hr = RemoveWebClientRegEntries();
_PrintIfError(hr, "RemoveWebClientRegEntries");
}
certocmBumpGasGauge(pComp, 35 DBGPARM(L"InstallServer"));
hr = RegisterAndUnRegisterDLLs(RD_SERVER, pComp, hwnd);
_JumpIfError(hr, error, "RegisterAndUnRegisterDLLs");
certocmBumpGasGauge(pComp, 40 DBGPARM(L"InstallServer"));
hr = CreateCertificateService(pComp, hwnd);
_JumpIfError(hr, error, "CreateCertificateService");
certocmBumpGasGauge(pComp, 45 DBGPARM(L"InstallServer"));
hr = CreateCertificates(pComp, hwnd);
_JumpIfError(hr, error, "CreateCertificates");
certocmBumpGasGauge(pComp, 50 DBGPARM(L"InstallServer"));
hr = CreateProgramGroups(FALSE, pComp, hwnd);
_JumpIfError(hr, error, "CreateProgramGroups");
certocmBumpGasGauge(pComp, 60 DBGPARM(L"InstallServer"));
hr = CreateCertWebIncPages(hwnd, pComp, TRUE /*IsServer*/, FALSE);
_JumpIfError(hr, error, "CreateCertWebIncPages");
hr = RenameMiscTargets(hwnd, pComp, TRUE);
_JumpIfError(hr, error, "RenameMiscTargets");
certocmBumpGasGauge(pComp, 70 DBGPARM(L"InstallServer"));
hr = RegisterDcomServer(
CLSID_CCertRequestD,
wszREQUESTFRIENDLYNAME,
wszREQUESTVERINDPROGID,
wszREQUESTPROGID);
_JumpIfError(hr, error, "RegisterDcomServer");
hr = RegisterDcomServer(
CLSID_CCertAdminD,
wszADMINFRIENDLYNAME,
wszADMINVERINDPROGID,
wszADMINPROGID);
_JumpIfError(hr, error, "RegisterDcomServer");
certocmBumpGasGauge(pComp, 80 DBGPARM(L"InstallServer"));
hr = RegisterDcomApp(CLSID_CCertRequestD);
_JumpIfError(hr, error, "RegisterDcomApp");
certocmBumpGasGauge(pComp, 90 DBGPARM(L"InstallServer"));
if (pServer->fUseDS)
{
hr = AddCAMachineToCertPublishers();
_JumpIfError(hr, error, "AddCAMachineToCertPublishers");
hr = InitializeCertificateTemplates();
_JumpIfError(hr, error, "InitializeCertificateTemplates");
}
certocmBumpGasGauge(pComp, 95 DBGPARM(L"InstallServer"));
// Set the security locally.
// A SubCA sets security when it receives its certificate from
// its parent. ALL OTHER CA installs (Root & reuse of existing certs)
// need to have security set now.
// On a SubCA the DS object security will have been set
// by a previous call in initlib if we already got our cert, or it will
// be set later when we install our cert.
// However, root certs install doesn't run completefrompkcs7(), so
// ds security on ent roots is never set. We must set it here.
// TODO: set security properly at DS object creation time!
fSetDSSecurity = (IsRootCA(pServer->CAType) || pServer->pccExistingCert);
hr = csiInitializeCertSrvSecurity(
pServer->pwszSanitizedName,
pServer->fUseDS,
fSetDSSecurity? pServer->fUseDS : FALSE); // clean SUBCA: happens during cert install, ROOT & reuse cert: apply now
_JumpIfError(hr, error, "csiInitializeCertSrvSecurity");
hr = GetSetupStatus(pServer->pwszSanitizedName, &dwSetupStatus);
if (S_OK == hr)
{
if (IsSubordinateCA(pServer->CAType) &&
(SETUP_SUSPEND_FLAG & dwSetupStatus) &&
(SETUP_REQUEST_FLAG & dwSetupStatus))
{
// put out an info dlg
CertInfoMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_INCOMPLETE_REQUEST,
pServer->pwszRequestFile);
}
}
else
{
_PrintError(hr, "GetSetupStatus");
}
certocmBumpGasGauge(pComp, 100 DBGPARM(L"InstallServer"));
hr = S_OK;
error:
if (NULL != pwszConfig)
{
LocalFree(pwszConfig);
}
if (NULL != pwszDBFile)
{
LocalFree(pwszDBFile);
}
if (fCoInit)
{
CoUninitialize();
}
CSILOG(hr, IDS_LOG_INSTALL_SERVER, NULL, NULL, NULL);
return(hr);
}
HRESULT
CreateCertsrvDirectories(
IN PER_COMPONENT_DATA *pComp,
IN BOOL fUpgrade,
IN BOOL fServer)
{
HRESULT hr;
WCHAR wszCertEnroll[MAX_PATH];
wszCertEnroll[0] = L'\0';
BuildPath(
wszCertEnroll,
ARRAYSIZE(wszCertEnroll),
pComp->pwszSystem32,
wszCERTENROLLSHAREPATH);
if (0 == CreateDirectory(wszCertEnroll, NULL))
{
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS) != hr)
{
_JumpErrorStr(hr, error, "CreateDirectory", wszCertEnroll);
}
}
if (fServer && NULL != pComp->CA.pServer->pwszSharedFolder)
{
if (pComp->fUnattended && !fUpgrade)
{
// make sure shared folder is created
if (!CreateDirectory(pComp->CA.pServer->pwszSharedFolder, NULL))
{
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS) != hr)
{
_JumpErrorStr(hr, error, "CreateDirectory",
pComp->CA.pServer->pwszSharedFolder);
}
}
}
if (!fUpgrade)
{
// set security on shared folder to
// FULL: Admins, LocalSystem, DomainAdmins
// READ: Everyone
// NOTE: in upgrade path, the system doesn't enable file share yet
// so skip the call
hr = csiSetAdminOnlyFolderSecurity(
pComp->CA.pServer->pwszSharedFolder,
TRUE, // apply Everyone:READ
pComp->CA.pServer->fUseDS);
_JumpIfError(hr, error, "csiSetAdminOnlyFolderSecurity");
}
}
hr = S_OK;
error:
return hr;
}
// following remove unused file/directory/registry
#define wszOLDHELP L"..\\help\\"
#define wszOLDCERTADM L"\\certadm"
#define wszOLDCERTQUE L"\\certque"
VOID
DeleteOldFilesAndDirectories(
IN PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
WCHAR wszPath[MAX_PATH];
WCHAR *pwszCertsrv;
wszPath[0] = L'\0';
// old help dir path
wcscpy(wszPath, pComp->pwszSystem32);
wcscat(wszPath, wszOLDHELP);
wcscat(wszPath, wszCERTSRV);
// remove old help files
hr = myRemoveFilesAndDirectory(wszPath, TRUE);
_PrintIfErrorStr(hr, "myRemoveFilesAndDirectory", wszPath);
// point to system32\certsrv
wcscpy(wszPath, pComp->pwszSystem32);
wcscat(wszPath, wszCERTSRV);
pwszCertsrv = &wszPath[wcslen(wszPath)];
// old vroot dir path
wcscpy(pwszCertsrv, wszOLDCERTADM);
hr = myRemoveFilesAndDirectory(wszPath, TRUE);
_PrintIfErrorStr(hr, "myRemoveFilesAndDirectory", wszPath);
wcscpy(pwszCertsrv, wszOLDCERTQUE);
hr = myRemoveFilesAndDirectory(wszPath, TRUE);
_PrintIfErrorStr(hr, "myRemoveFilesAndDirectory", wszPath);
// delete some obsolete registry keys and values
// old doc sub-component
hr = myDeleteCertRegValueEx(wszREGKEYOCMSUBCOMPONENTS,
NULL,
NULL,
wszOLDDOCCOMPONENT,
TRUE); //absolute path,
_PrintIfErrorStr2(hr, "myDeleteCertRegValueEx", wszOLDDOCCOMPONENT, hr);
// old CA cert serial number
if (NULL != pComp->CA.pServer &&
NULL != pComp->CA.pServer->pwszSanitizedName)
{
hr = myDeleteCertRegValue(
pComp->CA.pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCASERIALNUMBER);
_PrintIfErrorStr2(hr, "myDeleteCertRegValue", wszREGCASERIALNUMBER, hr);
}
}
VOID
DeleteObsoleteResidue()
{
HRESULT hr;
hr = myDeleteCertRegValueEx(wszREGKEYKEYSNOTTORESTORE,
NULL,
NULL,
wszREGRESTORECERTIFICATEAUTHORITY,
TRUE); //absolute path,
_PrintIfErrorStr(hr, "myDeleteCertRegValueEx",
wszREGRESTORECERTIFICATEAUTHORITY);
hr = myDeleteCertRegValueEx(wszREGKEYFILESNOTTOBACKUP,
NULL,
NULL,
wszREGRESTORECERTIFICATEAUTHORITY,
TRUE); //absolute path,
_PrintIfErrorStr(hr, "myDeleteCertRegValueEx",
wszREGRESTORECERTIFICATEAUTHORITY);
}
HRESULT
InstallCore(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp,
IN BOOL fServer)
{
HRESULT hr = pComp->hrContinue;
_JumpIfError(hr, error, "can't continue");
hr = CreateCertsrvDirectories(pComp, FALSE, fServer);
_JumpIfError(hr, error, "CreateCertsrvDirectories");
// Trigger an autoenroll to download root certs (see bug# 341568)
if(IsEnterpriseCA(pComp->CA.pServer->CAType))
{
hr = TriggerAutoenrollment();
_JumpIfError(hr, error, "TriggerAutoenrollment");
}
if (fServer)
{
hr = InstallServer(hwnd, pComp);
_JumpIfError(hr, error, "InstallServer");
}
else
{
hr = InstallClient(hwnd, pComp);
_JumpIfError(hr, error, "InstallClient");
}
if (g_fW3SvcRunning)
{
hr = StartAndStopService(pComp->hInstance,
pComp->fUnattended,
hwnd,
wszW3SVCNAME,
FALSE,
FALSE,
0, //doesn't matter since no confirm
&g_fW3SvcRunning);
_PrintIfError(hr, "StartAndStopService");
}
DeleteOldFilesAndDirectories(pComp);
hr = S_OK;
error:
return(hr);
}
HRESULT
BuildMultiStringList(
IN WCHAR const * const *apwsz,
IN DWORD cpwsz,
OUT WCHAR **ppwszz)
{
HRESULT hr;
DWORD i;
DWORD cwc;
WCHAR *pwc;
WCHAR *apwszEmpty[] = { L"", };
if (0 == cpwsz)
{
cpwsz = ARRAYSIZE(apwszEmpty);
apwsz = apwszEmpty;
}
cwc = 1;
for (i = 0; i < cpwsz; i++)
{
cwc += wcslen(apwsz[i]) + 1;
}
*ppwszz = (WCHAR *) LocalAlloc(LMEM_FIXED, cwc * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, *ppwszz);
pwc = *ppwszz;
for (i = 0; i < cpwsz; i++)
{
wcscpy(pwc, apwsz[i]);
pwc += wcslen(pwc) + 1;
}
*pwc = L'\0';
CSASSERT(SAFE_SUBTRACT_POINTERS(pwc, *ppwszz) + 1 == cwc);
hr = S_OK;
error:
return(hr);
}
HRESULT
helperGetDisableExtensionList(
PER_COMPONENT_DATA *pComp,
OUT WCHAR **ppwszz)
{
HRESULT hr;
WCHAR *apwszExt[] = {
L"", // C compiler won't allow empty list
//TEXT(szOID_ENROLL_CERTTYPE_EXTENSION), // 1.3.6.1.4.1.311.20.2
};
hr = BuildMultiStringList(apwszExt, ARRAYSIZE(apwszExt), ppwszz);
_JumpIfError(hr, error, "BuildMultiStringList");
error:
return(hr);
}
HRESULT
helperGetRequestExtensionList(
PER_COMPONENT_DATA *pComp,
OUT WCHAR **ppwszz)
{
HRESULT hr;
WCHAR *apwszExt[] = {
TEXT(szOID_RSA_SMIMECapabilities), // 1.2.840.113549.1.9.15
TEXT(szOID_CROSS_CERT_DIST_POINTS), // 1.3.6.1.4.1.311.10.9.1
TEXT(szOID_ENROLL_CERTTYPE_EXTENSION), // 1.3.6.1.4.1.311.20.2
TEXT(szOID_CERTSRV_CA_VERSION), // 1.3.6.1.4.1.311.21.1
TEXT(szOID_CERTSRV_PREVIOUS_CERT_HASH), // 1.3.6.1.4.1.311.21.2
TEXT(szOID_CERTIFICATE_TEMPLATE), // 1.3.6.1.4.1.311.21.7
TEXT(szOID_APPLICATION_CERT_POLICIES), // 1.3.6.1.4.1.311.21.10
TEXT(szOID_APPLICATION_POLICY_MAPPINGS), // 1.3.6.1.4.1.311.21.11
TEXT(szOID_APPLICATION_POLICY_CONSTRAINTS), // 1.3.6.1.4.1.311.21.12
TEXT(szOID_KEY_USAGE), // 2.5.29.15
TEXT(szOID_SUBJECT_ALT_NAME2), // 2.5.29.17
TEXT(szOID_NAME_CONSTRAINTS), // 2.5.29.30
TEXT(szOID_CERT_POLICIES), // 2.5.29.32
TEXT(szOID_POLICY_MAPPINGS), // 2.5.29.33
TEXT(szOID_POLICY_CONSTRAINTS), // 2.5.29.36
TEXT(szOID_ENHANCED_KEY_USAGE), // 2.5.29.37
};
hr = BuildMultiStringList(apwszExt, ARRAYSIZE(apwszExt), ppwszz);
_JumpIfError(hr, error, "BuildMultiStringList");
error:
return(hr);
}
HRESULT
FindCACertByCommonNameAndSerialNumber(
IN HCERTSTORE hCAStore,
IN WCHAR const *pwszCommonName,
IN BYTE const *pbSerialNumber,
IN DWORD cbSerialNumber,
CERT_CONTEXT const **ppCACert)
{
HRESULT hr;
CERT_RDN_ATTR rdnAttr = { szOID_COMMON_NAME, CERT_RDN_ANY_TYPE,};
CERT_RDN rdn = { 1, &rdnAttr };
CRYPT_INTEGER_BLOB SerialNumber;
CERT_CONTEXT const *pCACert = NULL;
CSASSERT(NULL != hCAStore &&
NULL != pwszCommonName &&
NULL != pbSerialNumber &&
NULL != ppCACert);
*ppCACert = NULL;
rdnAttr.Value.pbData = (BYTE *) pwszCommonName;
rdnAttr.Value.cbData = 0;
pCACert = NULL;
SerialNumber.pbData = const_cast<BYTE *>(pbSerialNumber);
SerialNumber.cbData = cbSerialNumber;
while (TRUE)
{
pCACert = CertFindCertificateInStore(
hCAStore,
X509_ASN_ENCODING,
CERT_UNICODE_IS_RDN_ATTRS_FLAG |
CERT_CASE_INSENSITIVE_IS_RDN_ATTRS_FLAG,
CERT_FIND_SUBJECT_ATTR,
&rdn,
pCACert);
if (NULL == pCACert)
{
hr = myHLastError();
_JumpError(hr, error, "CertFindCertificateInStore");
}
if (myAreSerialNumberBlobsSame(
&SerialNumber,
&pCACert->pCertInfo->SerialNumber))
{
break; // found correct one
}
}
*ppCACert = pCACert;
pCACert = NULL;
hr = S_OK;
error:
if (NULL != pCACert)
{
CertFreeCertificateContext(pCACert);
}
return hr;
}
HRESULT
GetCARegSerialNumber(
IN WCHAR const *pwszSanitizedCAName,
OUT BYTE **ppbSerialNumber,
OUT DWORD *pcbSerialNumber)
{
HRESULT hr;
WCHAR *pwszSerialNumber = NULL;
BYTE *pbSN = NULL;
DWORD cbSN;
hr = myGetCertRegStrValue(
pwszSanitizedCAName,
NULL,
NULL,
wszREGCASERIALNUMBER,
&pwszSerialNumber);
_JumpIfErrorStr2(
hr,
error,
"myGetCertRegStrValue",
wszREGCASERIALNUMBER,
HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND));
hr = WszToMultiByteInteger(FALSE, pwszSerialNumber, &cbSN, &pbSN);
_JumpIfError(hr, error, "WszToMultiByteInteger");
*ppbSerialNumber = pbSN;
pbSN = NULL;
*pcbSerialNumber = cbSN;
error:
if (NULL != pwszSerialNumber)
{
LocalFree(pwszSerialNumber);
}
if (NULL != pbSN)
{
LocalFree(pbSN);
}
CSASSERT(S_OK == hr || FAILED(hr));
return(hr);
}
HRESULT
LoadCurrentCACert(
IN WCHAR const *pwszCommonName,
IN WCHAR const *pwszSanitizedName,
IN BOOL fSignTest,
OUT CERT_CONTEXT const **ppcc,
OUT DWORD *pdwNameId)
{
HRESULT hr;
DWORD Count;
HCERTSTORE hMyStore = NULL;
BYTE *pbSerialNumber = NULL;
DWORD cbSerialNumber;
CRYPT_KEY_PROV_INFO *pKey = NULL;
DWORD cbKey;
DWORD dwType;
CERT_CONTEXT const *pcc = NULL;
WCHAR *pwszProvName = NULL;
DWORD dwProvType;
ALG_ID idAlg;
BOOL fMachineKeyset;
*ppcc = NULL;
// get prov name
hr = myGetCertSrvCSP(
FALSE, // fEncryptionCSP
pwszSanitizedName,
&dwProvType,
&pwszProvName,
&idAlg,
&fMachineKeyset,
NULL); // pdwKeySize
_JumpIfError(hr, error, "myGetCertSrvCSP");
// open my store
hMyStore = CertOpenStore(
CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING,
NULL, // hProv
CERT_SYSTEM_STORE_LOCAL_MACHINE |
CERT_STORE_ENUM_ARCHIVED_FLAG,
wszMY_CERTSTORE);
if (NULL == hMyStore)
{
hr = myHLastError();
_JumpError(hr, error, "CertOpenStore");
}
hr = myGetCARegHashCount(pwszSanitizedName, CSRH_CASIGCERT, &Count);
_JumpIfError(hr, error, "myGetCARegHashCount");
if (0 == Count)
{
*pdwNameId = 0; // renewal wasn't implemented yet
// find current CA cert by serial number -- the old fashioned way
hr = GetCARegSerialNumber(
pwszSanitizedName,
&pbSerialNumber,
&cbSerialNumber);
_JumpIfError(hr, error, "GetCARegSerialNumber");
hr = FindCACertByCommonNameAndSerialNumber(
hMyStore,
pwszCommonName,
pbSerialNumber,
cbSerialNumber,
&pcc);
_JumpIfError(hr, error, "FindCACertByCommonNameAndSerialNumber");
}
else
{
hr = myFindCACertByHashIndex(
hMyStore,
pwszSanitizedName,
CSRH_CASIGCERT,
Count - 1,
pdwNameId,
&pcc);
_JumpIfError(hr, error, "myFindCACertByHashIndex");
}
// get the private key provider info
if (!myCertGetCertificateContextProperty(
pcc,
CERT_KEY_PROV_INFO_PROP_ID,
CERTLIB_USE_LOCALALLOC,
(VOID **) &pKey,
&cbKey))
{
hr = myHLastError();
_JumpError(hr, error, "CertGetCertificateContextProperty");
}
if (fSignTest)
{
// signing testing
hr = myValidateHashForSigning(
pKey->pwszContainerName,
pwszProvName,
dwProvType,
fMachineKeyset,
&pcc->pCertInfo->SubjectPublicKeyInfo,
idAlg);
_JumpIfError(hr, error, "myValidateHashForSigning");
}
*ppcc = pcc;
pcc = NULL;
hr = S_OK;
error:
if (NULL != hMyStore)
{
CertCloseStore(hMyStore, CERT_CLOSE_STORE_CHECK_FLAG);
}
if (NULL != pbSerialNumber)
{
LocalFree(pbSerialNumber);
}
if (NULL != pKey)
{
LocalFree(pKey);
}
if (NULL != pwszProvName)
{
LocalFree(pwszProvName);
}
if (NULL != pcc)
{
CertFreeCertificateContext(pcc);
}
CSASSERT(S_OK == hr || FAILED(hr));
return(hr);
}
HRESULT
ArchiveCACertificate(
IN WCHAR const *pwszSanitizedName)
{
HRESULT hr;
HCERTSTORE hMyStore = NULL;
DWORD Count;
DWORD i;
CERT_CONTEXT const *pCert = NULL;
DWORD dwNameId;
CRYPT_DATA_BLOB Archived;
// open my store
hMyStore = CertOpenStore(
CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING,
NULL, // hProv
CERT_SYSTEM_STORE_LOCAL_MACHINE,
wszMY_CERTSTORE);
if (NULL == hMyStore)
{
hr = myHLastError();
_JumpError(hr, error, "CertOpenStore");
}
hr = myGetCARegHashCount(pwszSanitizedName, CSRH_CASIGCERT, &Count);
_JumpIfError(hr, error, "myGetCARegHashCount");
for (i = 0; i < Count; i++)
{
hr = myFindCACertByHashIndex(
hMyStore,
pwszSanitizedName,
CSRH_CASIGCERT,
i,
&dwNameId,
&pCert);
_PrintIfError(hr, "myFindCACertByHashIndex");
if (S_OK == hr)
{
Archived.cbData = 0;
Archived.pbData = NULL;
// We force an archive on the old cert and close it.
CertSetCertificateContextProperty(
pCert,
CERT_ARCHIVED_PROP_ID,
0,
&Archived);
CertFreeCertificateContext(pCert);
pCert = NULL;
}
}
hr = S_OK;
error:
if (NULL != pCert)
{
CertFreeCertificateContext(pCert);
}
if (NULL != hMyStore)
{
CertCloseStore(hMyStore, CERT_CLOSE_STORE_CHECK_FLAG);
}
CSASSERT(S_OK == hr || FAILED(hr));
return(hr);
}
// determine CA info, from build to build upgrade
HRESULT
DetermineCAInfoAndType(
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
ENUM_CATYPES CATypeDummy;
WCHAR *pwszCommonName = NULL;
CERT_CONTEXT const *pCACert = NULL;
DWORD dwNameId;
// ca type
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCATYPE,
(DWORD *) &CATypeDummy);
if (S_OK == hr)
{
pServer->CAType = CATypeDummy;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszREGCATYPE);
}
// else keep default CAType flag
// get current ca common name
hr = myGetCertRegStrValue(
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCOMMONNAME,
&pwszCommonName);
_JumpIfErrorStr(hr, error, "myGetCertRegStrValue", wszREGCOMMONNAME);
hr = LoadCurrentCACert(
pwszCommonName,
pServer->pwszSanitizedName,
FALSE, // don't do signing test during upgrade
&pCACert,
&pServer->dwCertNameId);
_JumpIfError(hr, error, "LoadCurrentCACert");
// now ready to load DN info
hr = DetermineExistingCAIdInfo(pServer, pCACert);
_JumpIfError(hr, error, "DetermineExistingCAIdInfo");
if (NULL != pServer->pccUpgradeCert)
{
CertFreeCertificateContext(pServer->pccUpgradeCert);
}
pServer->pccUpgradeCert = pCACert;
pCACert = NULL;
error:
if (NULL != pwszCommonName)
{
LocalFree(pwszCommonName);
}
if (NULL != pCACert)
{
CertFreeCertificateContext(pCACert);
}
return(hr);
}
// the following will determine ca sanitized name and upgrade path
HRESULT
DetermineServerUpgradePath(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
WCHAR *pwszDummy = NULL;
WCHAR *pwszSanitizedCAName = NULL;
LPWSTR pwszB2PolicyGuid = NULL;
DWORD dwVersion;
if (CS_UPGRADE_UNKNOWN != pComp->UpgradeFlag)
{
// already know upgrade type
CSASSERT(pServer->pwszSanitizedName); // this is a side-effect of this fxn, better have been set already
return S_OK;
}
// get active ca name
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGACTIVE,
&pwszSanitizedCAName);
if (hr != S_OK)
{
BOOL fFinishCYS;
//for W2K after, it is possible to be in post mode
hr = CheckPostBaseInstallStatus(&fFinishCYS);
if (S_OK == hr && !fFinishCYS)
{
//this could be either w2k or whistler
//treat as whistler since upgrade path won't execute
pComp->UpgradeFlag = CS_UPGRADE_WHISTLER;
goto done;
}
// wszREGACTIVE used in Win2k product and after. If not found, this is way before our time
// make sure by grabbing wszREGSP4DEFAULTCONFIGURATION
LPWSTR pwszTmp = NULL;
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGSP4DEFAULTCONFIGURATION,
&pwszTmp);
if (pwszTmp)
LocalFree(pwszTmp);
// error! bail, we have no idea what we're seeing
_JumpIfError(hr, error, "myGetCertRegStrValue wszREGSP4DEFAULTCONFIGURATION");
// hr == S_OK: yep, looks like valid NT4 installation
pComp->UpgradeFlag = CS_UPGRADE_UNSUPPORTED;
CSILOG(S_OK, IDS_LOG_UPGRADE_UNSUPPORTED, NULL, NULL, NULL);
_JumpError(hr, error, "myGetCertRegStrValue");
}
// check wszREGVERSION to get current version
hr = myGetCertRegDWValue(
NULL,
NULL,
NULL,
wszREGVERSION,
&dwVersion);
if (S_OK != hr && HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND)!= hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszREGVERSION);
}
// now either OK or FILE_NOT_FOUND
if (S_OK == hr)
{
// pComp->UpgradeFlag = SOME_FUNCTION(dwVersion); // CS_UPGRADE_WHISTLER already set as default; in future, key off of this
pComp->UpgradeFlag = CS_UPGRADE_WHISTLER;
CSILOG(S_OK, IDS_LOG_UPGRADE_B2B, NULL, NULL, NULL);
}
else
{
// FILE_NOT_FOUND: now we know it's (NT5 Beta 2 <= X < Whistler)
// is this Win2k, or NT5 Beta? Test for active policy module to have "ICertManageModule" entry
// check nt5 beta 2 and get active policy name
hr = myGetCertRegStrValue(
pwszSanitizedCAName,
wszREGKEYPOLICYMODULES,
NULL,
wszREGACTIVE,
&pwszB2PolicyGuid);
if (S_OK == hr)
{
hr = myGetCertRegStrValue(
wszREGKEYPOLICYMODULES,
pwszB2PolicyGuid,
NULL,
wszREGB2ICERTMANAGEMODULE,
&pwszDummy);
if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr)
{
// this doesn't exist on Win2k
pComp->UpgradeFlag = CS_UPGRADE_WIN2000;
CSILOG(S_OK, IDS_LOG_UPGRADE_WIN2000, NULL, NULL, NULL);
}
else
{
// this is definitely beta 2
pComp->UpgradeFlag = CS_UPGRADE_UNSUPPORTED;
CSILOG(S_OK, IDS_LOG_UPGRADE_UNSUPPORTED, NULL, NULL, NULL);
}
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr)
{
// strange, maybe no active module. Assume OK, latest bits.
pComp->UpgradeFlag = CS_UPGRADE_WIN2000;
CSILOG(S_OK, IDS_LOG_UPGRADE_WIN2000, NULL, NULL, NULL);
}
else
{
// other failure, just bail
_JumpErrorStr(hr, error, "myGetCertRegStrValue",
wszREGKEYPOLICYMODULES);
}
} // wszREGVERSION: FILE_NOT_FOUND
// take sanitized name
if (NULL != pServer->pwszSanitizedName)
{
// this will free the default name
LocalFree(pServer->pwszSanitizedName);
}
pServer->pwszSanitizedName = pwszSanitizedCAName;
pwszSanitizedCAName = NULL;
done:
hr = S_OK;
error:
if (NULL != pwszDummy)
{
LocalFree(pwszDummy);
}
if (NULL != pwszSanitizedCAName)
{
LocalFree(pwszSanitizedCAName);
}
CSILOG(hr, IDS_LOG_UPGRADE_TYPE, NULL, NULL, (DWORD const *) &pComp->UpgradeFlag);
return hr;
}
// Description: load and determine necessary information for upgrade
// upgrade won't continue if any error
HRESULT
LoadAndDetermineServerUpgradeInfo(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
WCHAR *pwszRevocationType = NULL;
DWORD dwNetscapeType;
BOOL fDummy;
ALG_ID idAlgDummy;
CSP_INFO CSPInfoDummy;
ENUM_CATYPES CATypeDummy;
WCHAR *pwszCommonName = NULL;
// initialize
ZeroMemory(&CSPInfoDummy, sizeof(CSPInfoDummy));
// load information for all upgrade scenarios
// csp
hr = myGetCertRegStrValue(
pServer->pwszSanitizedName,
wszREGKEYCSP,
NULL,
wszREGPROVIDER,
&CSPInfoDummy.pwszProvName);
if (S_OK == hr && NULL != CSPInfoDummy.pwszProvName)
{
if (NULL != pServer->pCSPInfo->pwszProvName)
{
// free default csp
LocalFree(pServer->pCSPInfo->pwszProvName);
}
// use reg one as default for upgrade
pServer->pCSPInfo->pwszProvName = CSPInfoDummy.pwszProvName;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegStrValue", wszREGPROVIDER);
}
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
wszREGKEYCSP,
NULL,
wszREGPROVIDERTYPE,
&CSPInfoDummy.dwProvType);
if (S_OK == hr)
{
pServer->pCSPInfo->dwProvType = CSPInfoDummy.dwProvType;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszREGPROVIDERTYPE);
}
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
wszREGKEYCSP,
NULL,
wszMACHINEKEYSET,
(DWORD*)&CSPInfoDummy.fMachineKeyset);
if (S_OK == hr)
{
pServer->pCSPInfo->fMachineKeyset = CSPInfoDummy.fMachineKeyset;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszMACHINEKEYSET);
}
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
wszREGKEYCSP,
NULL,
wszHASHALGORITHM,
(DWORD*)&idAlgDummy);
if (S_OK == hr)
{
pServer->pHashInfo->idAlg = idAlgDummy;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpIfErrorStr(hr, error, "myGetCertRegDWValue", wszHASHALGORITHM);
}
if (NULL != pServer->pCSPInfoList)
{
// BUG, this will never happen because csp info list is not loaded yet
// one more checking, make sure csp is installed
if (NULL == findCSPInfoFromList(
pServer->pCSPInfoList,
pServer->pCSPInfo->pwszProvName,
pServer->pCSPInfo->dwProvType))
{
// if not, this is a broken ca
hr = E_INVALIDARG;
_JumpErrorStr(hr, error, "findCSPInfoFromList",
pServer->pCSPInfo->pwszProvName);
}
}
// UseDS flag
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAUSEDS,
(DWORD*)&fDummy);
if (S_OK == hr)
{
// use from reg
pServer->fUseDS = fDummy;
}
else if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszREGCAUSEDS);
}
// CACommonName
// this will be used for looking cert in store to determine ca DN info
hr = myGetCertRegStrValue(
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCOMMONNAME,
&pwszCommonName);
if (S_OK != hr && HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpError(hr, error, "myGetCertRegStrValue");
}
else if ((HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr) ||
(S_OK == hr && L'\0' == pwszCommonName[0]))
{
if (S_OK == hr && L'\0' == pwszCommonName[0])
{
// empty string, use snaitized name instead
LocalFree(pwszCommonName);
}
// in case empty or not found, use sanitized
pwszCommonName = (WCHAR*)LocalAlloc(LMEM_FIXED,
(wcslen(pServer->pwszSanitizedName) + 1) * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pwszCommonName);
wcscpy(pwszCommonName, pServer->pwszSanitizedName);
}
if (NULL != pServer->pwszCACommonName)
{
LocalFree(pServer->pwszCACommonName);
}
pServer->pwszCACommonName = pwszCommonName;
pwszCommonName = NULL;
// Collect CAType, DN info, dwCertNameId and upgrade ca cert
hr = DetermineCAInfoAndType(pComp);
_JumpIfError(hr, error, "DetermineCAInfoAndType");
// load following values for later
// check revocation type
hr = myGetCertRegDWValue(
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGREVOCATIONTYPE,
&pServer->dwRevocationFlags);
if(hr != S_OK)
{
pServer->dwRevocationFlags = pServer->fUseDS?
REVEXT_DEFAULT_DS : REVEXT_DEFAULT_NODS;
}
// following for web page creation
// load shared folder for ca cert file name creation
if (NULL != pServer->pwszSharedFolder)
{
// shouldn't happen but in case
LocalFree(pServer->pwszSharedFolder);
pServer->pwszSharedFolder = NULL;
}
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGDIRECTORY,
&pServer->pwszSharedFolder);
if (S_OK == hr && L'\0' == pServer->pwszSharedFolder[0])
{
// in case of empty, set to null
LocalFree(pServer->pwszSharedFolder);
pServer->pwszSharedFolder = NULL;
}
else if (S_OK != hr && HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) != hr)
{
_JumpError(hr, error, "myGetCertRegStrValue");
}
// build ca cert file name for web install and ca cert saving
if (NULL != pServer->pwszCACertFile)
{
// free old one
LocalFree(pServer->pwszCACertFile);
}
hr = csiBuildCACertFileName(
pComp->hInstance,
NULL, //hwnd, ok for upgrade
pComp->fUnattended,
pServer->pwszSharedFolder,
pServer->pwszSanitizedName,
L".crt",
0, // iCert
&pServer->pwszCACertFile);
_JumpIfError(hr, error, "BuildCACertFileName");
hr = S_OK;
error:
if (NULL != pwszRevocationType)
{
LocalFree(pwszRevocationType);
}
if (NULL != pwszCommonName)
{
LocalFree(pwszCommonName);
}
return hr;
}
HRESULT
GetConfigInSharedFolderWithCert(
WCHAR const *pwszSharedFolder,
OUT WCHAR **ppwszConfig)
{
HRESULT hr;
ICertConfig * pICertConfig = NULL;
BSTR bstrConfig = NULL;
BOOL fCoInit = FALSE;
WCHAR *pwszCAMachine;
WCHAR *pwszCAName;
WCHAR wszCertFileInSharedFolder[MAX_PATH];
LONG i;
LONG lCount;
LONG Index;
BOOL fLocal;
CSASSERT(NULL != ppwszConfig);
*ppwszConfig = NULL;
hr = CoInitialize(NULL);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "CoInitialize");
}
fCoInit = TRUE;
hr = CoCreateInstance(
CLSID_CCertConfig,
NULL,
CLSCTX_INPROC_SERVER,
IID_ICertConfig,
(VOID**) &pICertConfig);
_JumpIfError(hr, error, "CoCreateInstance");
// get local
hr = pICertConfig->Reset(0, &lCount);
_JumpIfError(hr, error, "ICertConfig->Reset");
for (i = 0; i < lCount; ++i)
{
hr = pICertConfig->Next(&Index);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "ICertConfig->Next");
}
if (-1 == Index)
{
hr = HRESULT_FROM_WIN32(ERROR_NO_MORE_ITEMS);
_JumpError(hr, error, "No CA cert file");
}
hr = pICertConfig->GetField(wszCONFIG_CONFIG, &bstrConfig);
_JumpIfError(hr, error, "ICertConfig->GetField(Config)");
pwszCAMachine = (WCHAR*)bstrConfig;
pwszCAName = wcschr(pwszCAMachine, L'\\');
if (NULL == pwszCAName)
{
hr = E_INVALIDARG;
_JumpError(hr, error, "Invalid config string");
}
pwszCAName[0] = '\0'; // make pwszCAMachine
++pwszCAName;
// form NT4 ca cert file path in shared folder
wcscpy(wszCertFileInSharedFolder, pwszSharedFolder);
wcscat(wszCertFileInSharedFolder, L"\\");
wcscat(wszCertFileInSharedFolder, pwszCAMachine);
wcscat(wszCertFileInSharedFolder, L"_");
wcscat(wszCertFileInSharedFolder, pwszCAName);
wcscat(wszCertFileInSharedFolder, L".crt");
DBGPRINT((
DBG_SS_CERTOCM,
"wszCertFileInSharedFolder: %ws\n",
wszCertFileInSharedFolder));
if (myDoesFileExist(wszCertFileInSharedFolder))
{
//done
break;
}
SysFreeString(bstrConfig);
bstrConfig = NULL;
}
if (i >= lCount)
{
hr = HRESULT_FROM_WIN32(ERROR_NO_MORE_ITEMS);
_JumpError(hr, error, "No CA cert file");
}
*ppwszConfig = (WCHAR*)LocalAlloc(LMEM_FIXED,
SysStringByteLen(bstrConfig) + sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, *ppwszConfig);
wcscpy(*ppwszConfig, bstrConfig);
hr = S_OK;
error:
if (NULL != pICertConfig)
{
pICertConfig->Release();
}
if (NULL != bstrConfig)
{
SysFreeString(bstrConfig);
}
if (fCoInit)
{
CoUninitialize();
}
return hr;
}
HRESULT
DetermineClientUpgradePath(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CAWEBCLIENTSETUPINFO *pClient = pComp->CA.pClient;
WCHAR *pwszCAName;
DWORD dwVersion;
if (CS_UPGRADE_UNKNOWN != pComp->UpgradeFlag)
{
// already know upgrade type
CSASSERT(pClient->pwszWebCAMachine); // this is a side-effect of this fxn, better have been set already
CSASSERT(pClient->pwszWebCAName); // this is a side-effect of this fxn, better have been set already
return S_OK;
}
//get ca info from registry here
if (NULL != pClient->pwszWebCAMachine)
{
// shouldn't happen, just in case
LocalFree(pClient->pwszWebCAMachine);
pClient->pwszWebCAMachine = NULL;
}
if (NULL != pClient->pwszWebCAName)
{
// shouldn't happen, just in case
LocalFree(pClient->pwszWebCAName);
pClient->pwszWebCAName = NULL;
}
// get ca machine
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGWEBCLIENTCAMACHINE,
&pClient->pwszWebCAMachine);
if (S_OK != hr || L'\0' == pClient->pwszWebCAMachine[0])
{
BOOL fFinishCYS;
//for W2K after, it is possible to be in post mode
hr = CheckPostBaseInstallStatus(&fFinishCYS);
if (S_OK == hr && !fFinishCYS)
{
//this could be either w2k or whistler
//treat as whistler since upgrade path won't execute
pComp->UpgradeFlag = CS_UPGRADE_WHISTLER;
goto done;
}
// incorrect reg state,
// either not found entry or empty string: NT4
pComp->UpgradeFlag = CS_UPGRADE_UNSUPPORTED;
hr = S_OK;
CSILOG(S_OK, IDS_LOG_UPGRADE_UNSUPPORTED, NULL, NULL, NULL);
_JumpErrorStr(hr, error, "myGetCertRegStrValue", wszREGWEBCLIENTCAMACHINE);
}
// get ca
hr = myGetCertRegStrValue(
NULL,
NULL,
NULL,
wszREGWEBCLIENTCANAME,
&pClient->pwszWebCAName);
if (S_OK != hr || L'\0' == pClient->pwszWebCAName[0])
{
// incorrect reg state,
// either not found entry or empty string: NT4
if (pClient->pwszWebCAMachine)
LocalFree(pClient->pwszWebCAMachine);
pComp->UpgradeFlag = CS_UPGRADE_UNSUPPORTED;
hr = S_OK;
CSILOG(S_OK, IDS_LOG_UPGRADE_UNSUPPORTED, NULL, NULL, NULL);
_JumpErrorStr(hr, error, "myGetCertRegStrValue", wszREGWEBCLIENTCANAME);
}
// Now either W2k or Whistler
// check wszREGVERSION to get current version on Whistler++
hr = myGetCertRegDWValue(
NULL,
NULL,
NULL,
wszREGVERSION,
&dwVersion);
if (S_OK != hr && HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND)!= hr)
{
_JumpErrorStr(hr, error, "myGetCertRegDWValue", wszREGVERSION);
}
// now either OK or FILE_NOT_FOUND
if (S_OK == hr)
{
// pComp->UpgradeFlag = SOME_FUNCTION(dwVersion); // CS_UPGRADE_WHISTLER already set as default; in future, key off of this
pComp->UpgradeFlag = CS_UPGRADE_WHISTLER;
CSILOG(S_OK, IDS_LOG_UPGRADE_B2B, NULL, NULL, NULL);
}
else
{
pComp->UpgradeFlag = CS_UPGRADE_WIN2000;
CSILOG(S_OK, IDS_LOG_UPGRADE_WIN2000, NULL, NULL, NULL);
}
done:
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_UPGRADE_TYPE, NULL, NULL, (DWORD const *) &pComp->UpgradeFlag);
return hr;
}
HRESULT
LoadAndDetermineClientUpgradeInfo(
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CAWEBCLIENTSETUPINFO *pClient = pComp->CA.pClient;
WCHAR *pwszCAName;
DWORD dwVersion;
//get ca info from registry here
if (NULL != pClient->pwszSanitizedWebCAName)
{
// shouldn't happen, just in case
LocalFree(pClient->pwszSanitizedWebCAName);
pClient->pwszSanitizedWebCAName = NULL;
}
hr = DetermineClientUpgradePath(pComp);
_JumpIfError(hr, error, "DetermineClientUpgradePath");
// get ca
hr = myGetCertRegDWValue(
NULL,
NULL,
NULL,
wszREGWEBCLIENTCATYPE,
(DWORD *) &pClient->WebCAType);
_PrintIfErrorStr(hr, "myGetCertRegDWValue", wszREGWEBCLIENTCATYPE);
hr = mySanitizeName(pClient->pwszWebCAName,
&pClient->pwszSanitizedWebCAName);
_JumpIfError(hr, error, "mySanitizeName");
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_UPGRADE_TYPE, NULL, NULL, (DWORD const *) &pComp->UpgradeFlag);
return hr;
}
// apply ACL to key container for all upgrade scenarios
HRESULT
UpgradeKeyContainerSecurity(
HWND hwnd,
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
HCRYPTPROV hProv = NULL;
WCHAR *pwszProvName = NULL;
DWORD dwProvType;
ALG_ID idAlg;
BOOL fMachineKeyset;
// get prov name
hr = myGetCertSrvCSP(
FALSE, // fEncryptionCSP
pComp->CA.pServer->pwszSanitizedName,
&dwProvType,
&pwszProvName,
&idAlg,
&fMachineKeyset,
NULL); // pdwKeySize
_JumpIfError(hr, error, "myGetCertSrvCSP");
if (!myCertSrvCryptAcquireContext(&hProv,
pComp->CA.pServer->pwszSanitizedName,
pwszProvName,
dwProvType,
CRYPT_SILENT, // get key, upgrade, no UI
fMachineKeyset))
{
hr = myHLastError();
_JumpError(hr, error, "myCertSrvCryptAcquireContext");
}
// set acl on it
hr = csiSetKeyContainerSecurity(hProv);
_JumpIfError(hr, error, "csiSetKeyContainerSecurity");
error:
if (NULL != hProv)
{
CryptReleaseContext(hProv, 0);
}
if (NULL != pwszProvName)
{
LocalFree(pwszProvName);
}
CSILOG(hr, IDS_LOG_UPGRADE_KEY_SECURITY, NULL, NULL, NULL);
return hr;
}
HRESULT InstallNewTemplates(HWND hwnd)
{
HRESULT hr = S_OK;
SHELLEXECUTEINFO shi;
ZeroMemory(&shi, sizeof(shi));
shi.cbSize = sizeof(shi);
shi.hwnd = hwnd;
shi.lpVerb = SZ_VERB_OPEN;
shi.lpFile = SZ_REGSVR32;
shi.lpParameters = SZ_REGSVR32_CERTCLI;
shi.fMask = SEE_MASK_FLAG_NO_UI |
SEE_MASK_NOCLOSEPROCESS;
if(!ShellExecuteEx(&shi))
{
hr = myHLastError();
_JumpError(hr, error, "ShellExecuteEx");
}
if(WAIT_FAILED == WaitForSingleObject(shi.hProcess, INFINITE))
{
hr = myHLastError();
_JumpError(hr, error, "WaitForSingleObject");
}
error:
if(shi.hProcess)
{
CloseHandle(shi.hProcess);
}
return hr;
}
HRESULT
UpgradeServer(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
hr = RegisterAndUnRegisterDLLs(RD_SERVER|RD_SKIPUNREGMMC, pComp, hwnd);
_JumpIfError(hr, error, "RegisterAndUnRegisterDLLs");
hr = LoadAndDetermineServerUpgradeInfo(pComp);
_JumpIfError(hr, error, "LoadAndDetermineServerUpgradeInfo");
// create enroll dir
hr = CreateCertsrvDirectories(pComp, TRUE, TRUE);
_JumpIfError(hr, error, "CreateCertsrvDirectories");
hr = EnableVRootsAndShares(FALSE, TRUE, TRUE, pComp);
_PrintIfError(hr, "EnableVRootsAndShares(share only)");
hr = UpgradeServerRegEntries(hwnd, pComp);
_JumpIfError(hr, error, "UpgradeServerRegEntries");
hr = CreateCertWebIncPages(hwnd, pComp, TRUE /*IsServer*/, TRUE);
_JumpIfError(hr, error, "CreateCertWebIncPages");
hr = RenameMiscTargets(hwnd, pComp, TRUE);
_JumpIfError(hr, error, "RenameMiscTargets");
hr = UpgradeKeyContainerSecurity(hwnd, pComp);
// ignore new acl failure
_PrintIfError(hr, "UpgradeKeyContainerSecurity");
// always register dcom
hr = RegisterDcomServer(
CLSID_CCertRequestD,
wszREQUESTFRIENDLYNAME,
wszREQUESTVERINDPROGID,
wszREQUESTPROGID);
_JumpIfError(hr, error, "RegisterDcomServer");
hr = RegisterDcomServer(
CLSID_CCertAdminD,
wszADMINFRIENDLYNAME,
wszADMINVERINDPROGID,
wszADMINPROGID);
_JumpIfError(hr, error, "RegisterDcomServer");
hr = RegisterDcomApp(CLSID_CCertRequestD);
_JumpIfError(hr, error, "RegisterDcomApp");
// fix for 155772 Certsrv: After upgrading a 2195 Enterprise Root CA
// to 2254.01VBL03 the CA will no longer issue Certs
hr = InstallNewTemplates(hwnd);
_JumpIfError(hr, error, "InstallNewTemplates");
// always fix certsvc in upgrade
hr = FixCertsvcService(pComp);
_PrintIfError(hr, "FixCertsvcService");
// delete any old program groups
DeleteProgramGroups(TRUE, pComp);
hr = CreateProgramGroups(FALSE, pComp, hwnd);
_PrintIfError(hr, "CreateProgramGroups");
// delete old stuff
DeleteOldFilesAndDirectories(pComp);
DBGPRINT((DBG_SS_CERTOCM, "UpgradeServer: setting SETUP_SERVER_UPGRADED_FLAG\n"));
hr = SetSetupStatus(NULL, SETUP_SERVER_UPGRADED_FLAG, TRUE);
_JumpIfError(hr, error, "SetSetupStatus");
hr = csiUpgradeCertSrvSecurity(
pServer->pwszSanitizedName,
IsEnterpriseCA(pServer->CAType)?true:false,
pServer->fUseDS?true:false,
pComp->UpgradeFlag);
if (hr == HRESULT_FROM_WIN32(ERROR_CAN_NOT_COMPLETE))
{
_PrintError(hr, "csiUpgradeCertSrvSecurity marked for later fixup; error ignored");
hr = S_OK;
}
_JumpIfError(hr, error, "csiUpgradeCertSrvSecurity");
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_UPGRADE_SERVER, NULL, NULL, NULL);
return hr;
}
HRESULT
UpgradeClient(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CAWEBCLIENTSETUPINFO *pClient = pComp->CA.pClient;
if ((IS_CLIENT_UPGRADE & pComp->dwInstallStatus) &&
(IS_SERVER_UPGRADE & pComp->dwInstallStatus))
{
// upgrade server will also hit here so skip it
goto done;
}
// unset client
hr = SetSetupStatus(NULL, SETUP_CLIENT_FLAG, FALSE);
_JumpIfError(hr, error, "SetSetupStatus");
hr = RegisterAndUnRegisterDLLs(RD_CLIENT|RD_SKIPUNREGMMC, pComp, hwnd);
_JumpIfError(hr, error, "RegisterAndUnRegisterDLLs");
hr = LoadAndDetermineClientUpgradeInfo(pComp);
_JumpIfError(hr, error, "LoadAndDetermineClientUpgradeInfo");
hr = UpgradeWebClientRegEntries(pComp);
_JumpIfError(hr, error, "UpgradeWebClientRegEntries");
hr = CreateCertWebIncPages(hwnd, pComp, FALSE /*IsServer*/, TRUE);
_JumpIfError(hr, error, "CreateCertWebIncPages");
hr = RenameMiscTargets(hwnd, pComp, FALSE);
_JumpIfError(hr, error, "RenameMiscTargets");
// delete any old program groups
DeleteProgramGroups(FALSE, pComp);
hr = CreateProgramGroups(TRUE, pComp, hwnd);
_PrintIfError(hr, "CreateProgramGroups");
hr = SetSetupStatus(NULL, SETUP_CLIENT_FLAG, TRUE);
_JumpIfError(hr, error, "SetSetupStatus");
DeleteOldFilesAndDirectories(pComp);
done:
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_UPGRADE_CLIENT, NULL, NULL, NULL);
return hr;
}
HRESULT
GetServerNames(
IN HWND hwnd,
IN HINSTANCE hInstance,
IN BOOL fUnattended,
OUT WCHAR **ppwszServerName,
OUT WCHAR **ppwszServerNameOld)
{
HRESULT hr;
// Retrieve computer name strings.
hr = myGetComputerNames(ppwszServerName, ppwszServerNameOld);
if (S_OK != hr)
{
CertErrorMessageBox(
hInstance,
fUnattended,
hwnd,
IDS_ERR_GETCOMPUTERNAME,
hr,
NULL);
_JumpError(hr, error, "myGetComputerNames");
}
error:
return(hr);
}
HRESULT
UpdateDomainAndUserName(
IN HWND hwnd,
IN OUT PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
// free old server and installer info because we might get them in nt base
if (NULL != pComp->pwszServerName)
{
LocalFree(pComp->pwszServerName);
pComp->pwszServerName = NULL;
}
if (NULL != pComp->pwszServerNameOld)
{
LocalFree(pComp->pwszServerNameOld);
pComp->pwszServerNameOld = NULL;
}
hr = GetServerNames(
hwnd,
pComp->hInstance,
pComp->fUnattended,
&pComp->pwszServerName,
&pComp->pwszServerNameOld);
_JumpIfError(hr, error, "GetServerNames");
DBGPRINT((DBG_SS_CERTOCM, "UpdateDomainAndUserName:%ws,%ws\n", pComp->pwszServerName, pComp->pwszServerNameOld));
DumpBackTrace("UpdateDomainAndUserName");
error:
return(hr);
}
HRESULT
StopCertService(
IN HWND hwnd,
IN SC_HANDLE hSC,
IN WCHAR const *pwszServiceName)
{
HRESULT hr;
SERVICE_STATUS status;
DWORD dwAttempt;
if (!ControlService(hSC, SERVICE_CONTROL_STOP, &status))
{
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_SERVICE_NOT_ACTIVE) != hr)
{
_JumpErrorStr(hr, error, "ControlService(Stop)", pwszServiceName);
}
}
// get out after it is really stopped
for (dwAttempt = 0; dwAttempt < CERT_MAX_ATTEMPT; dwAttempt++)
{
DBGCODE(status.dwCurrentState = -1);
if (!QueryServiceStatus(hSC, &status))
{
// query failed, ignore error
hr = S_OK;
_JumpErrorStr(
myHLastError(), // Display ignored error
error,
"QueryServiceStatus",
pwszServiceName);
}
if (status.dwCurrentState != SERVICE_STOP_PENDING &&
status.dwCurrentState != SERVICE_RUNNING)
{
// it was stopped already
break;
}
DBGPRINT((
DBG_SS_CERTOCM,
"Stopping %ws service: current state=%d\n",
pwszServiceName,
status.dwCurrentState));
Sleep(CERT_HALF_SECOND);
}
if (dwAttempt >= CERT_MAX_ATTEMPT)
{
DBGPRINT((
DBG_SS_CERTOCM,
"Timeout stopping %ws service: current state=%d\n",
pwszServiceName,
status.dwCurrentState));
}
else
{
DBGPRINT((
DBG_SS_CERTOCM,
"Stopped %ws service\n",
pwszServiceName));
}
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_SERVICE_STOPPED, pwszServiceName, NULL, NULL);
return(hr);
}
HRESULT
GetServiceControl(
WCHAR const *pwszServiceName,
OUT SC_HANDLE *phService)
{
HRESULT hr;
SC_HANDLE hSCManager = NULL;
*phService = NULL;
hSCManager = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
if (NULL == hSCManager)
{
hr = myHLastError();
_JumpError(hr, error, "OpenSCManager");
}
*phService = OpenService(hSCManager, pwszServiceName, SERVICE_ALL_ACCESS);
if (NULL == *phService)
{
hr = myHLastError();
_JumpErrorStr2(
hr,
error,
"OpenService",
pwszServiceName,
HRESULT_FROM_WIN32(ERROR_SERVICE_DOES_NOT_EXIST));
}
hr = S_OK;
error:
if (NULL != hSCManager)
{
CloseServiceHandle(hSCManager);
}
return hr;
}
HRESULT
IsServiceRunning(
IN SC_HANDLE const hSCService,
OUT BOOL *pfRun)
{
HRESULT hr;
SERVICE_STATUS status;
*pfRun = FALSE;
if (!QueryServiceStatus(hSCService, &status))
{
hr = myHLastError();
_JumpError(hr, error, "QueryServiceStatus");
}
if (SERVICE_STOPPED != status.dwCurrentState &&
SERVICE_STOP_PENDING != status.dwCurrentState)
{
*pfRun = TRUE;
}
hr = S_OK;
error:
return hr;
}
HRESULT
StartAndStopService(
IN HINSTANCE hInstance,
IN BOOL fUnattended,
IN HWND const hwnd,
IN WCHAR const *pwszServiceName,
IN BOOL const fStopService,
IN BOOL const fConfirm,
IN int iMsg,
OUT BOOL *pfServiceWasRunning)
{
HRESULT hr;
SC_HANDLE hService = NULL;
SERVICE_STATUS status;
*pfServiceWasRunning = FALSE;
hr = GetServiceControl(pwszServiceName, &hService);
if (S_OK != hr)
{
_PrintError2(
hr,
"GetServiceControl",
HRESULT_FROM_WIN32(ERROR_SERVICE_DOES_NOT_EXIST));
if (HRESULT_FROM_WIN32(ERROR_SERVICE_DOES_NOT_EXIST) == hr)
{
hr = S_OK;
}
goto error;
}
// to get status if the service is running
hr = IsServiceRunning(hService, pfServiceWasRunning);
_JumpIfError(hr, error, "IsServiceRunning");
if (fStopService)
{
if (*pfServiceWasRunning)
{
// stop the service
if (fConfirm)
{
// confirmation dialog
int ret = CertMessageBox(
hInstance,
fUnattended,
hwnd,
iMsg,
0,
MB_YESNO |
MB_ICONWARNING |
CMB_NOERRFROMSYS,
NULL);
if (IDYES != ret)
{
hr = E_ABORT;
_JumpError(hr, error, "Cancel it");
}
}
hr = StopCertService(hwnd, hService, pwszServiceName);
_JumpIfErrorStr(hr, error, "StopCertService", pwszServiceName);
}
}
else
{
// START the service
if (!*pfServiceWasRunning)
{
if (!StartService(hService, 0, NULL))
{
hr = myHLastError();
_JumpErrorStr(hr, error, "StartService", pwszServiceName);
}
}
}
hr = S_OK;
error:
if (NULL != hService)
{
CloseServiceHandle(hService);
}
CSILOG(
hr,
fStopService? IDS_LOG_SERVICE_STOPPED : IDS_LOG_SERVICE_STARTED,
pwszServiceName,
NULL,
NULL);
return hr;
}
// fix existing certsvc service to add/use new service description
HRESULT
FixCertsvcService(PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
SC_HANDLE hService = NULL;
QUERY_SERVICE_CONFIG *pServiceConfig = NULL;
WCHAR *pwszServiceDesc = NULL;
WCHAR const *pwszDisplayName;
SERVICE_DESCRIPTION sd;
DWORD dwSize;
hr = GetServiceControl(wszSERVICE_NAME, &hService);
_JumpIfError(hr, error, "GetServiceControl");
// get size
if (!QueryServiceConfig(hService,
NULL,
0,
&dwSize))
{
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER) != hr)
{
_JumpError(hr, fix_desc, "QueryServiceConfig");
}
}
else
{
// impossible???
hr = E_INVALIDARG;
_JumpError(hr, fix_desc, "QueryServiceConfig");
}
CSASSERT(0 < dwSize);
// allocate config buffer
pServiceConfig = (QUERY_SERVICE_CONFIG*)LocalAlloc(
LMEM_FIXED | LMEM_ZEROINIT,
dwSize);
_JumpIfOutOfMemory(hr, error, pServiceConfig);
// get config
if (!QueryServiceConfig(hService,
pServiceConfig,
dwSize,
&dwSize))
{
hr = myHLastError();
_JumpError(hr, fix_desc, "QueryServiceConfig");
}
// use new display name
pwszDisplayName = myLoadResourceString(IDS_CA_SERVICEDISPLAYNAME);
if (NULL == pwszDisplayName)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
if (!ChangeServiceConfig(hService,
pServiceConfig->dwServiceType, //dwServiceType
SERVICE_NO_CHANGE, //dwStartType
SERVICE_NO_CHANGE, //dwErrorControl
NULL, //lpBinaryPathName
NULL, //lpLoadOrderGroup
NULL, //lpdwTagId
NULL, //lpDependences
NULL, //lpServiceStartName
NULL, //lpPassword
pwszDisplayName))
{
hr = myHLastError();
_JumpIfError(hr, fix_desc, "ChangeServiceConfig");
}
fix_desc:
// add description
hr = myLoadRCString(pComp->hInstance, IDS_CA_SERVICEDESCRIPTION, &pwszServiceDesc);
_JumpIfError(hr, error, "myLoadRCString");
sd.lpDescription = pwszServiceDesc;
if (!ChangeServiceConfig2(hService,
SERVICE_CONFIG_DESCRIPTION,
(VOID*)&sd))
{
hr = myHLastError();
_JumpError(hr, error, "ChangeServiceConfig2");
}
hr = S_OK;
error:
if (NULL != hService)
{
CloseServiceHandle(hService);
}
if (NULL != pwszServiceDesc)
{
LocalFree(pwszServiceDesc);
}
if (NULL != pServiceConfig)
{
LocalFree(pServiceConfig);
}
return hr;
}
HRESULT
PreUninstallCore(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp,
IN BOOL fPreserveClient)
{
BOOL fDummy;
HRESULT hr;
DWORD dwFlags = RD_UNREGISTER;
hr = StartAndStopService(pComp->hInstance,
pComp->fUnattended,
hwnd,
wszSERVICE_NAME,
TRUE, // stop the service
FALSE, // no confirm
0, //doesn't matter since no confirm
&fDummy);
_PrintIfError(hr, "StartAndStopService");
hr = RegisterAndUnRegisterDLLs(dwFlags, pComp, hwnd);
_PrintIfError(hr, "RegisterAndUnRegisterDLLs");
hr = S_OK;
return(hr);
}
VOID
DeleteServiceAndGroups(
IN HINSTANCE hInstance,
IN BOOL fUnattended,
IN HWND hwnd)
{
SC_HANDLE hSCManager = NULL;
SC_HANDLE hSC = NULL;
HRESULT hr;
hSCManager = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
if (NULL == hSCManager)
{
hr = myHLastError();
_JumpError(hr, error, "OpenSCManager");
}
hSC = OpenService(hSCManager, wszSERVICE_NAME, SERVICE_ALL_ACCESS);
if (NULL != hSC)
{
if (!DeleteService(hSC))
{
hr = myHLastError();
CertErrorMessageBox(
hInstance,
fUnattended,
hwnd,
IDS_ERR_DELETESERVICE,
hr,
wszSERVICE_NAME);
_PrintError(hr, "DeleteService");
}
CloseServiceHandle(hSC);
}
error:
if (NULL != hSCManager)
{
CloseServiceHandle(hSCManager);
}
}
HRESULT SetCertSrvInstallVersion()
{
HRESULT hr;
hr = mySetCertRegDWValueEx(
FALSE,
NULL,
NULL,
NULL,
wszREGVERSION,
(CSVER_MAJOR << 16) | CSVER_MINOR);
_PrintIfErrorStr(hr, "mySetCertRegDWValueEx", wszREGVERSION);
return hr;
}
HRESULT
CreateWebClientRegEntries(
BOOL fUpgrade,
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CAWEBCLIENTSETUPINFO *pClient = pComp->CA.pClient;
hr = mySetCertRegStrValueEx(fUpgrade, NULL, NULL, NULL,
wszREGWEBCLIENTCAMACHINE, pClient->pwszWebCAMachine);
_JumpIfError(hr, error, "mySetCertRegStrValueEx");
hr = mySetCertRegStrValueEx(fUpgrade, NULL, NULL, NULL,
wszREGWEBCLIENTCANAME, pClient->pwszWebCAName);
_JumpIfError(hr, error, "mySetCertRegStrValueEx");
hr = mySetCertRegDWValueEx(fUpgrade, NULL, NULL, NULL,
wszREGWEBCLIENTCATYPE, pClient->WebCAType);
_JumpIfError(hr, error, "mySetCertRegDWValueEx");
hr = SetCertSrvInstallVersion();
_JumpIfError(hr, error, "SetCertSrvInstallVersion");
if (NULL != pClient->pwszSharedFolder)
{
hr = mySetCertRegStrValueEx(fUpgrade, NULL, NULL, NULL,
wszREGDIRECTORY, pClient->pwszSharedFolder);
_JumpIfError(hr, error, "mySetCertRegStrValue");
}
hr = S_OK;
error:
CSILOG(hr, IDS_LOG_CREATE_CLIENT_REG, NULL, NULL, NULL);
return hr;
}
HRESULT
UpgradeWebClientRegEntries(
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
hr = CreateWebClientRegEntries(TRUE, pComp);
_JumpIfError(hr, error, "CreateWebClientRegEntries");
// hr = S_OK;
error:
return hr;
}
HRESULT
UninstallCore(
IN HWND hwnd,
IN PER_COMPONENT_DATA *pComp,
IN DWORD PerCentCompleteBase,
IN DWORD PerCentCompleteMax,
IN BOOL fPreserveClient,
IN BOOL fRemoveVD,
IN BOOL fPreserveToDoList)
{
BOOL fCoInit = FALSE;
HRESULT hr;
WCHAR *pwszSharedFolder = NULL;
WCHAR *pwszSanitizedCAName = NULL;
ENUM_CATYPES caType = ENUM_UNKNOWN_CA;
BOOL fUseDS = FALSE;
WCHAR *pwszDBDirectory = NULL;
WCHAR *pwszLogDirectory = NULL;
WCHAR *pwszSysDirectory = NULL;
WCHAR *pwszTmpDirectory = NULL;
DWORD DBSessionCount = 0;
DWORD PerCentCompleteDelta;
PerCentCompleteDelta = (PerCentCompleteMax - PerCentCompleteBase) / 10;
#define _UNINSTALLPERCENT(tenths) \
(PerCentCompleteBase + (tenths) * PerCentCompleteDelta)
// get current active CA info
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGACTIVE, &pwszSanitizedCAName);
_PrintIfError(hr, "UninstallCore(no active CA)");
if (S_OK == hr)
{
hr = ArchiveCACertificate(pwszSanitizedCAName);
_PrintIfError2(hr, "ArchiveCACertificate", HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND));
hr = myGetCertRegDWValue(pwszSanitizedCAName, NULL, NULL, wszREGCATYPE, (DWORD*)&caType);
_PrintIfError(hr, "no reg ca type");
hr = myGetCertRegDWValue(pwszSanitizedCAName, NULL, NULL, wszREGCAUSEDS, (DWORD*)&fUseDS);
_PrintIfError(hr, "no reg use ds");
hr = DeleteCertificates(pwszSanitizedCAName, IsRootCA(caType));
if(S_OK != hr)
{
CertWarningMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_IDINFO_DELETECERTIFICATES,
0,
NULL);
}
}
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDIRECTORY, &pwszSharedFolder);
_PrintIfError(hr, "no shared folder");
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDBDIRECTORY, &pwszDBDirectory);
_PrintIfError(hr, "no db directory");
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDBLOGDIRECTORY, &pwszLogDirectory);
_PrintIfError(hr, "no log directory");
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDBSYSDIRECTORY, &pwszSysDirectory);
_PrintIfError(hr, "no sys directory");
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDBTEMPDIRECTORY, &pwszTmpDirectory);
_PrintIfError(hr, "no tmp directory");
hr = myGetCertRegDWValue(NULL, NULL, NULL, wszREGDBSESSIONCOUNT, &DBSessionCount);
_PrintIfError(hr, "no session count");
hr = CoInitialize(NULL);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "CoInitialize");
}
fCoInit = TRUE;
DeleteServiceAndGroups(pComp->hInstance, pComp->fUnattended, hwnd);
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(1) DBGPARM(L"UninstallCore"));
if (!fPreserveToDoList)
{
// if we're uninstalling, always clear post-base ToDo list
RegDeleteKey(HKEY_LOCAL_MACHINE, wszREGKEYCERTSRVTODOLIST);
}
if (fPreserveClient)
{
hr = RegisterAndUnRegisterDLLs(RD_CLIENT, pComp, hwnd);
_JumpIfError(hr, error, "RegisterAndUnRegisterDLLs");
hr = CreateCertWebIncPages(hwnd, pComp, FALSE /*IsServer*/, FALSE);
_JumpIfError(hr, error, "CreateCertWebIncPages");
}
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(2) DBGPARM(L"UninstallCore"));
DeleteProgramGroups(TRUE, pComp);
if (fPreserveClient)
{
hr = CreateProgramGroups(TRUE, pComp, hwnd);
_JumpIfError(hr, error, "CreateProgramGroups");
}
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(3) DBGPARM(L"UninstallCore"));
UnregisterDcomServer(
CLSID_CCertRequestD,
wszREQUESTVERINDPROGID,
wszREQUESTPROGID);
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(4) DBGPARM(L"UninstallCore"));
UnregisterDcomServer(
CLSID_CCertAdminD,
wszADMINVERINDPROGID,
wszADMINPROGID);
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(5) DBGPARM(L"UninstallCore"));
UnregisterDcomApp();
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(6) DBGPARM(L"UninstallCore"));
if (fRemoveVD && !fPreserveClient)
{
DisableVRootsAndShares(TRUE, TRUE);
myDeleteFilePattern(pComp->pwszSystem32, wszCERTSRV, TRUE);
}
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(7) DBGPARM(L"UninstallCore"));
if (NULL != pwszSharedFolder)
{
// this must be restore before CreateConfigFiles()
hr = mySetCertRegStrValue(NULL, NULL, NULL,
wszREGDIRECTORY, pwszSharedFolder);
_PrintIfError(hr, "mySetCertRegStrValue");
//remove entry
hr = CreateConfigFiles(pwszSharedFolder, pComp, TRUE, hwnd);
_PrintIfError2(hr, "CreateConfigFiles(Remove old entry)", hr);
}
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(8) DBGPARM(L"UninstallCore"));
// restore db path
if (NULL != pwszDBDirectory)
{
hr = mySetCertRegStrValue(NULL, NULL, NULL,
wszREGDBDIRECTORY, pwszDBDirectory);
_PrintIfError(hr, "mySetCertRegStrValue");
}
if (NULL != pwszLogDirectory)
{
hr = mySetCertRegStrValue(NULL, NULL, NULL,
wszREGDBLOGDIRECTORY, pwszLogDirectory);
_PrintIfError(hr, "mySetCertRegStrValue");
}
if (NULL != pwszSysDirectory)
{
hr = mySetCertRegStrValue(NULL, NULL, NULL,
wszREGDBSYSDIRECTORY, pwszSysDirectory);
_PrintIfError(hr, "mySetCertRegStrValue");
}
if (NULL != pwszTmpDirectory)
{
hr = mySetCertRegStrValue(NULL, NULL, NULL,
wszREGDBTEMPDIRECTORY, pwszTmpDirectory);
_PrintIfError(hr, "mySetCertRegStrValue");
}
if (0 != DBSessionCount)
{
hr = mySetCertRegDWValue(NULL, NULL, NULL,
wszREGDBSESSIONCOUNT, DBSessionCount);
_PrintIfError(hr, "mySetCertRegDWValueEx");
}
if (fPreserveClient)
{
// this means uninstall server component and keep web client
hr = CreateWebClientRegEntries(FALSE, pComp);
_JumpIfError(hr, error, "CreateWebClientRegEntries");
}
DeleteObsoleteResidue();
DeleteOldFilesAndDirectories(pComp);
certocmBumpGasGauge(pComp, _UNINSTALLPERCENT(9) DBGPARM(L"UninstallCore"));
if (fUseDS)
{
hr = RemoveCAInDS(pwszSanitizedCAName);
_PrintIfError2(hr, "RemoveCAInDS", hr);
}
certocmBumpGasGauge(pComp, PerCentCompleteMax DBGPARM(L"UninstallCore"));
hr = S_OK;
error:
if (NULL != pwszSanitizedCAName)
{
LocalFree(pwszSanitizedCAName);
}
if (NULL != pwszSharedFolder)
{
LocalFree(pwszSharedFolder);
}
if (NULL != pwszDBDirectory)
{
LocalFree(pwszDBDirectory);
}
if (NULL != pwszLogDirectory)
{
LocalFree(pwszLogDirectory);
}
if (NULL != pwszSysDirectory)
{
LocalFree(pwszSysDirectory);
}
if (NULL != pwszTmpDirectory)
{
LocalFree(pwszTmpDirectory);
}
if (fCoInit)
{
CoUninitialize();
}
return(hr);
}
HRESULT
AddCAToRPCNullSessions()
{
HRESULT hr;
HKEY hRegKey = NULL;
char *pszOriginal = NULL;
char *psz;
DWORD cb;
DWORD cbTmp;
DWORD cbSum;
DWORD dwType;
hr = RegOpenKeyExA(
HKEY_LOCAL_MACHINE,
szNULL_SESSION_REG_LOCATION,
0, // dwOptions
KEY_READ | KEY_WRITE,
&hRegKey);
_JumpIfError(hr, error, "RegOpenKeyExA");
// Need to get the size of the value first
hr = RegQueryValueExA(hRegKey, szNULL_SESSION_VALUE, 0, &dwType, NULL, &cb);
_JumpIfError(hr, error, "RegQueryValueExA");
if (REG_MULTI_SZ != dwType)
{
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
_JumpError(hr, error, "RegQueryValueExA: Type");
}
cb += sizeof(rgcCERT_NULL_SESSION) - 1;
pszOriginal = (char *) LocalAlloc(LMEM_FIXED | LMEM_ZEROINIT, cb);
if (NULL == pszOriginal)
{
hr = E_OUTOFMEMORY;
_JumpError(hr, error, "LocalAlloc");
}
// get the multi string of RPC null session pipes
hr = RegQueryValueExA(
hRegKey,
szNULL_SESSION_VALUE,
0,
&dwType,
(BYTE *) pszOriginal,
&cb);
_JumpIfError(hr, error, "RegQueryValueExA");
psz = pszOriginal;
// look for CERT in the list
cbSum = 0;
while (TRUE)
{
if (0 == strcmp(rgcCERT_NULL_SESSION, psz))
{
break;
}
cbTmp = strlen(psz) + 1;
psz += cbTmp;
cbSum += cbTmp;
if (cb < cbSum + 1)
{
break;
}
if ('\0' == psz[0])
{
// add the CA pipe to the multi string
CopyMemory(psz, rgcCERT_NULL_SESSION, sizeof(rgcCERT_NULL_SESSION));
// set the new multi string in the reg value
hr = RegSetValueExA(
hRegKey,
szNULL_SESSION_VALUE,
0,
REG_MULTI_SZ,
(BYTE *) pszOriginal,
cbSum + sizeof(rgcCERT_NULL_SESSION));
_JumpIfError(hr, error, "RegSetValueExA");
break;
}
}
hr = S_OK;
error:
if (NULL != pszOriginal)
{
LocalFree(pszOriginal);
}
if (NULL != hRegKey)
{
RegCloseKey(hRegKey);
}
return(hr);
}
HRESULT
AddCARegKeyToRegConnectExemptions()
{
// add ourselves to list of people that take ACLs seriously
// and should be allowed to reveal our key to outsiders.
HRESULT hr;
LPWSTR pszExempt = NULL;
HKEY hkeyWinReg = NULL, hkeyAllowedPaths = NULL;
LPWSTR pszTmp;
DWORD dwDisposition, dwType;
DWORD cb=0, cbRegKeyCertSrvPath = (wcslen(wszREGKEYCERTSVCPATH)+1) *sizeof(WCHAR);
// carefully query this -- if it doesn't exist, we don't have to apply workaround
hr = RegOpenKeyEx(
HKEY_LOCAL_MACHINE,
L"SYSTEM\\CurrentControlSet\\Control\\SecurePipeServers\\Winreg",
0,
KEY_ALL_ACCESS,
&hkeyWinReg);
_JumpIfError(hr, Ret, "RegOpenKeyEx");
// creation of this optional key is always ok if above key exists
hr = RegCreateKeyEx(
hkeyWinReg,
L"AllowedPaths",
NULL,
NULL,
0,
KEY_ALL_ACCESS,
NULL,
&hkeyAllowedPaths,
&dwDisposition);
_JumpIfError(hr, Ret, "RegCreateKeyEx exempt regkey");
hr = RegQueryValueEx(
hkeyAllowedPaths,
L"Machine",
NULL, // reserved
&dwType, // type
NULL, // pb
&cb);
_PrintIfError(hr, "RegQueryValueEx exempt regkey 1");
if ((hr == S_OK) && (dwType != REG_MULTI_SZ))
{
hr = HRESULT_FROM_WIN32(ERROR_DATATYPE_MISMATCH);
_JumpError(hr, Ret, "RegQueryValueEx invalid type");
}
// always include at least a terminator
if (cb < sizeof(WCHAR))
cb = sizeof(WCHAR);
pszExempt = (LPWSTR)LocalAlloc(LMEM_FIXED, cb + cbRegKeyCertSrvPath );
_JumpIfOutOfMemory(hr, Ret, pszExempt);
// start with double null for safety
pszExempt[0] = L'\0';
pszExempt[1] = L'\0';
hr = RegQueryValueEx(
hkeyAllowedPaths,
L"Machine",
NULL, // reserved
NULL, // type
(PBYTE)pszExempt, // pb
&cb);
_PrintIfError(hr, "RegQueryValueEx exempt regkey 2");
pszTmp = pszExempt;
while(pszTmp[0] != L'\0') // skip all existing strings
{
// if entry already exists, bail
if (0 == lstrcmpi(wszREGKEYCERTSVCPATH, pszTmp))
{
hr = S_OK;
goto Ret;
}
pszTmp += wcslen(pszTmp)+1;
}
wcscpy(&pszTmp[0], wszREGKEYCERTSVCPATH);
pszTmp[wcslen(wszREGKEYCERTSVCPATH)+1] = L'\0'; // double NULL
hr = RegSetValueEx(
hkeyAllowedPaths,
L"Machine",
NULL,
REG_MULTI_SZ,
(PBYTE)pszExempt,
cb + cbRegKeyCertSrvPath);
_JumpIfError(hr, Ret, "RegSetValueEx exempt regkey");
Ret:
if (hkeyAllowedPaths)
RegCloseKey(hkeyAllowedPaths);
if (hkeyWinReg)
RegCloseKey(hkeyWinReg);
if (pszExempt)
LocalFree(pszExempt);
return hr;
}
HRESULT
helperGetFilesNotToRestore(
PER_COMPONENT_DATA *pComp,
OUT WCHAR **ppwszz)
{
HRESULT hr;
WCHAR *pwsz;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
DWORD cwc;
WCHAR const wszDBDIRPATTERN[] = L"\\*.edb";
WCHAR const wszDBLOGDIRPATTERN[] = L"\\*";
*ppwszz = NULL;
cwc = wcslen(pServer->pwszDBDirectory) +
WSZARRAYSIZE(wszDBDIRPATTERN) +
1 +
wcslen(pServer->pwszLogDirectory) +
WSZARRAYSIZE(wszDBLOGDIRPATTERN) +
1 +
1;
pwsz = (WCHAR *) LocalAlloc(LMEM_FIXED, cwc * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pwsz);
*ppwszz = pwsz;
wcscpy(pwsz, pServer->pwszDBDirectory);
wcscat(pwsz, wszDBDIRPATTERN);
pwsz += wcslen(pwsz) + 1;
wcscpy(pwsz, pServer->pwszLogDirectory);
wcscat(pwsz, wszDBLOGDIRPATTERN);
pwsz += wcslen(pwsz) + 1;
*pwsz = L'\0';
CSASSERT(cwc == (DWORD) (pwsz - *ppwszz + 1));
hr = S_OK;
error:
return(hr);
}
HRESULT
CreateServerRegEntries(
HWND hwnd,
BOOL fUpgrade,
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
HKEY hKeyBase = NULL;
WCHAR *pwszCLSIDCertGetConfig = NULL;
WCHAR *pwszCLSIDCertRequest = NULL;
WCHAR *pwszCRLPeriod = NULL;
WCHAR *pwszCRLDeltaPeriod = NULL;
WCHAR *pwszzCRLPublicationValue = NULL;
WCHAR *pwszzCACertPublicationValue = NULL;
WCHAR *pwszzRequestExtensionList = NULL;
WCHAR *pwszzDisableExtensionList = NULL;
WCHAR *pwszzFilesNotToRestore = NULL;
WCHAR *pwszProvNameReg = NULL;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
DWORD dwUpgradeFlags = fUpgrade ? CSREG_UPGRADE : 0x0;
DWORD dwAppendFlags = 0x0; // default
DWORD dwCRLPeriodCount, dwCRLDeltaPeriodCount;
LDAP *pld = NULL;
BSTR strDomainDN = NULL;
BSTR strConfigDN = NULL;
// no error checking?
hr = AddCAToRPCNullSessions();
_PrintIfError(hr, "AddCAToRPCNullSessions");
hr = AddCARegKeyToRegConnectExemptions();
_PrintIfError(hr, "AddCARegKeyToRegConnectExemptions");
// create the CA key, so we can set security on it.
hr = myCreateCertRegKey(pServer->pwszSanitizedName, NULL, NULL);
_JumpIfError(hr, error, "myCreateCertRegKey");
// configuration level
// active ca
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGACTIVE,
pServer->pwszSanitizedName);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGACTIVE);
if (NULL != pServer->pwszSharedFolder)
{
// shared folder
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDIRECTORY,
pServer->pwszSharedFolder);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDIRECTORY);
}
// db dir
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDBDIRECTORY,
pServer->pwszDBDirectory);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDBDIRECTORY);
// log dir
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDBLOGDIRECTORY,
pServer->pwszLogDirectory);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDBLOGDIRECTORY);
// db tmp dir
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDBTEMPDIRECTORY,
pServer->pwszLogDirectory);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDBTEMPDIRECTORY);
// db sys dir
hr = mySetCertRegStrValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDBSYSDIRECTORY,
pServer->pwszLogDirectory);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDBSYSDIRECTORY);
hr = mySetCertRegDWValueEx(
fUpgrade,
NULL,
NULL,
NULL,
wszREGDBSESSIONCOUNT,
DBSESSIONCOUNTDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGDBSESSIONCOUNT);
hr = SetCertSrvInstallVersion();
_JumpIfError(hr, error, "SetCertSrvInstallVersion");
if (!fUpgrade)
{
// preserve db
hr = SetSetupStatus(NULL, SETUP_CREATEDB_FLAG, !pServer->fPreserveDB);
_JumpIfError(hr, error, "SetSetupStatus");
}
// ca level
if (!fUpgrade && pServer->fUseDS)
{
hr = myLdapOpen(&pld, &strDomainDN, &strConfigDN);
_JumpIfError(hr, error, "myLdapOpen");
// Config DN
hr = mySetCertRegStrValueEx(
FALSE,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGDSCONFIGDN,
strConfigDN);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDSCONFIGDN);
// Domain DN
hr = mySetCertRegStrValueEx(
FALSE,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGDSDOMAINDN,
strDomainDN);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGDSDOMAINDN);
}
// ca type
CSASSERT(IsEnterpriseCA(pServer->CAType) || IsStandaloneCA(pServer->CAType));
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCATYPE,
pServer->CAType);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCATYPE);
// use DS flag
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAUSEDS,
pServer->fUseDS);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCAUSEDS);
// teletex flag
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGFORCETELETEX,
ENUM_TELETEX_AUTO | ENUM_TELETEX_UTF8);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGFORCETELETEX);
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
NULL,
NULL,
wszSECUREDATTRIBUTES,
wszzDEFAULTSIGNEDATTRIBUTES);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszSECUREDATTRIBUTES,);
// common name
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCOMMONNAME,
pServer->pwszCACommonName);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCOMMONNAME);
// enable reg
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGENABLED,
TRUE);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGENABLED);
// policy flag
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGPOLICYFLAGS,
0);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGPOLICYFLAGS);
// enroll compatible flag, always turn it off
// BUG, consider use mySetCertRegDWValueEx with fUpgrade
// after W2K to support CertEnrollCompatible upgrade
hr = mySetCertRegDWValue(
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCERTENROLLCOMPATIBLE,
FALSE);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValue", wszREGCERTENROLLCOMPATIBLE);
// Cert Server CRL Edit Flags
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLEDITFLAGS,
EDITF_ENABLEAKIKEYID);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLEDITFLAGS);
// Cert Server CRL Flags
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLFLAGS,
CRLF_DELETE_EXPIRED_CRLS);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLFLAGS);
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGENFORCEX500NAMELENGTHS,
TRUE);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGENFORCEX500NAMELENGTHS);
// subject template
hr = mySetCertRegMultiStrValueEx(
0, // dwUpgradeFlags: always overwrite!
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGSUBJECTTEMPLATE,
wszzREGSUBJECTTEMPLATEVALUE);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGSUBJECTTEMPLATE);
// (hard code) clock skew minutes
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCLOCKSKEWMINUTES,
CCLOCKSKEWMINUTESDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCLOCKSKEWMINUTES);
// (hard code) log level
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGLOGLEVEL,
CERTLOG_WARNING);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGLOGLEVEL);
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGHIGHSERIAL,
0);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGLOGLEVEL);
// register server name
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCASERVERNAME,
pComp->pwszServerName);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCASERVERNAME);
// default validity period string and count for issued certs
// use years for string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGVALIDITYPERIODSTRING,
wszVALIDITYPERIODSTRINGDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGVALIDITYPERIODSTRING);
// validity period count
// use 1 year for standalone and 2 years for enterprise
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGVALIDITYPERIODCOUNT,
IsEnterpriseCA(pServer->CAType)?
dwVALIDITYPERIODCOUNTDEFAULT_ENTERPRISE :
dwVALIDITYPERIODCOUNTDEFAULT_STANDALONE);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGVALIDITYPERIODCOUNT);
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAXCHGCERTHASH,
NULL);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGCAXCHGCERTHASH);
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGKRACERTHASH,
NULL);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGKRACERTHASH);
hr = mySetCertRegDWValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGKRACERTCOUNT,
0);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGKRACERTCOUNT);
hr = mySetCertRegDWValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGKRAFLAGS,
0);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGKRAFLAGS);
// CRL Publication URLs:
hr = csiGetCRLPublicationURLTemplates(
pServer->fUseDS,
pComp->pwszSystem32,
&pwszzCRLPublicationValue);
_JumpIfError(hr, error, "csiGetCRLPublicationURLTemplates");
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags | CSREG_MERGE,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLPUBLICATIONURLS,
pwszzCRLPublicationValue);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGCRLPUBLICATIONURLS);
// if this API returns non-null strings, it's got good data
hr = csiGetCRLPublicationParams(
TRUE,
&pwszCRLPeriod,
&dwCRLPeriodCount);
_PrintIfError(hr, "csiGetCRLPublicationParams");
// crl period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLPERIODSTRING,
(pwszCRLPeriod == NULL) ? wszCRLPERIODSTRINGDEFAULT : pwszCRLPeriod);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLPERIODSTRING);
// crl period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLPERIODCOUNT,
(pwszCRLPeriod == NULL) ? dwCRLPERIODCOUNTDEFAULT : dwCRLPeriodCount);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLPERIODCOUNT);
// crl overlap period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLOVERLAPPERIODSTRING,
wszCRLOVERLAPPERIODSTRINGDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLOVERLAPPERIODSTRING);
// crl overlap period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLOVERLAPPERIODCOUNT,
dwCRLOVERLAPPERIODCOUNTDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLOVERLAPPERIODCOUNT);
// if this API returns non-null strings, it's got good data
hr = csiGetCRLPublicationParams(
FALSE, // delta
&pwszCRLDeltaPeriod,
&dwCRLDeltaPeriodCount);
_PrintIfError(hr, "csiGetCRLPublicationParams");
// delta crl period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLDELTAPERIODSTRING,
(pwszCRLDeltaPeriod == NULL) ? wszCRLDELTAPERIODSTRINGDEFAULT : pwszCRLDeltaPeriod);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLDELTAPERIODSTRING);
// delta crl period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLDELTAPERIODCOUNT,
(pwszCRLDeltaPeriod == NULL) ? dwCRLPERIODCOUNTDEFAULT : dwCRLDeltaPeriodCount);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLDELTAPERIODCOUNT);
// delta crl overlap period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLDELTAOVERLAPPERIODSTRING,
wszCRLDELTAOVERLAPPERIODSTRINGDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLDELTAOVERLAPPERIODSTRING);
// delta crl overlap period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCRLDELTAOVERLAPPERIODCOUNT,
dwCRLDELTAOVERLAPPERIODCOUNTDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLDELTAOVERLAPPERIODCOUNT);
// CA xchg cert validity period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAXCHGVALIDITYPERIODSTRING,
wszCAXCHGVALIDITYPERIODSTRINGDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLDELTAOVERLAPPERIODSTRING);
// CA xchg cert validity period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAXCHGVALIDITYPERIODCOUNT,
dwCAXCHGVALIDITYPERIODCOUNTDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLDELTAOVERLAPPERIODCOUNT);
// CA xchg cert overlap period string
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAXCHGOVERLAPPERIODSTRING,
wszCAXCHGOVERLAPPERIODSTRINGDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCRLDELTAOVERLAPPERIODSTRING);
// CA xchg cert overlap period count
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCAXCHGOVERLAPPERIODCOUNT,
dwCAXCHGOVERLAPPERIODCOUNTDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGCRLDELTAOVERLAPPERIODCOUNT);
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGMAXINCOMINGMESSAGESIZE,
MAXINCOMINGMESSAGESIZEDEFAULT);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx", wszREGMAXINCOMINGMESSAGESIZE);
if (NULL != pServer->pwszSharedFolder)
{
// register CA file name for certhier and renewal
hr = mySetCARegFileNameTemplate(
wszREGCACERTFILENAME,
pComp->pwszServerName,
pServer->pwszSanitizedName,
pServer->pwszCACertFile);
_JumpIfError(hr, error, "SetRegCertFileName");
}
// policy
// create default policy entry explicitly to get correct acl if upgrade
hr = myCreateCertRegKeyEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY);
_JumpIfErrorStr(hr, error, "myCreateCertRegKeyEx", wszCLASS_CERTPOLICY);
// if customized policy, create a new entry with correct acl
if (fUpgrade &&
NULL != pServer->pwszCustomPolicy &&
0 != wcscmp(wszCLASS_CERTPOLICY, pServer->pwszCustomPolicy) )
{
hr = myCreateCertRegKeyEx(
TRUE, // upgrade
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
pServer->pwszCustomPolicy);
_JumpIfError(hr, error, "myCreateCertRegKey");
}
// set default policy
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
NULL,
wszREGACTIVE,
(fUpgrade && (NULL != pServer->pwszCustomPolicy)) ?
pServer->pwszCustomPolicy :
wszCLASS_CERTPOLICY);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGACTIVE);
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGREVOCATIONTYPE,
pServer->dwRevocationFlags | pServer->dwUpgradeRevFlags);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGREVOCATIONTYPE);
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGCAPATHLENGTH,
CAPATHLENGTH_INFINITE);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGCAPATHLENGTH);
// revocation url
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGREVOCATIONURL,
g_wszASPRevocationURLTemplate);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGREVOCATIONURL);
// Exit module publish flags
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYEXITMODULES,
wszCLASS_CERTEXIT,
wszREGCERTPUBLISHFLAGS,
pServer->fUseDS ?
EXITPUB_DEFAULT_ENTERPRISE :
EXITPUB_DEFAULT_STANDALONE);
_JumpIfErrorStr(
hr,
error,
"mySetCertRegStrValueEx",
wszREGCERTPUBLISHFLAGS);
// Enable Request Extensions:
hr = helperGetRequestExtensionList(pComp, &pwszzRequestExtensionList);
_JumpIfError(hr, error, "helperGetRequestExtensionList");
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags | CSREG_MERGE,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGENABLEREQUESTEXTENSIONLIST,
pwszzRequestExtensionList);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGENABLEREQUESTEXTENSIONLIST);
hr = helperGetDisableExtensionList(pComp, &pwszzDisableExtensionList);
_JumpIfError(hr, error, "helperGetDisableExtensionList");
// Disables Template Extensions:
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags | CSREG_MERGE,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGDISABLEEXTENSIONLIST,
pwszzDisableExtensionList);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGDISABLEEXTENSIONLIST);
// Subject Alt Name Extension
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGSUBJECTALTNAME,
wszREGSUBJECTALTNAMEVALUE);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGSUBJECTALTNAME);
// Subject Alt Name 2 Extension
hr = mySetCertRegStrValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGSUBJECTALTNAME2,
wszREGSUBJECTALTNAME2VALUE);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGSUBJECTALTNAME2);
// Request Disposition
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGREQUESTDISPOSITION,
IsEnterpriseCA(pServer->CAType)?
REQDISP_DEFAULT_ENTERPRISE :
REQDISP_DEFAULT_STANDALONE);
_JumpIfErrorStr(hr, error, "mySetCertRegDWValueEx",
wszREGREQUESTDISPOSITION);
// Edit Flags
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGEDITFLAGS,
IsEnterpriseCA(pServer->CAType)?
(EDITF_DEFAULT_ENTERPRISE | pServer->dwUpgradeEditFlags) :
(EDITF_DEFAULT_STANDALONE | pServer->dwUpgradeEditFlags));
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx", wszREGEDITFLAGS);
// Issuer Cert URL Flags
hr = mySetCertRegDWValueEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGISSUERCERTURLFLAGS,
pServer->fUseDS?
ISSCERT_DEFAULT_DS : ISSCERT_DEFAULT_NODS);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValueEx",
wszREGISSUERCERTURLFLAGS);
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
wszREGDEFAULTSMIME,
wszzREGVALUEDEFAULTSMIME);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGDEFAULTSMIME);
hr = csiGetCACertPublicationURLTemplates(
pServer->fUseDS,
pComp->pwszSystem32,
&pwszzCACertPublicationValue);
_JumpIfError(hr, error, "csiGetCACertPublicationURLTemplates");
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags | CSREG_MERGE,
pServer->pwszSanitizedName,
NULL,
NULL,
wszREGCACERTPUBLICATIONURLS,
pwszzCACertPublicationValue);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGCRLPUBLICATIONURLS);
// exit
// create default exit entry to get correct acl if upgrade
hr = myCreateCertRegKeyEx(
fUpgrade,
pServer->pwszSanitizedName,
wszREGKEYEXITMODULES,
wszCLASS_CERTEXIT);
_JumpIfErrorStr(hr, error, "myCreateCertRegKeyEx", wszCLASS_CERTPOLICY);
// if customized exit, create a new entry with correct acl
if (fUpgrade &&
NULL != pServer->pwszzCustomExit &&
0 != wcscmp(wszCLASS_CERTEXIT, pServer->pwszzCustomExit) )
{
// create a new entry for custom exit
hr = myCreateCertRegKeyEx(
TRUE, // upgrade
pServer->pwszSanitizedName,
wszREGKEYEXITMODULES,
pServer->pwszzCustomExit);
_JumpIfError(hr, error, "myCreateCertRegKey");
}
// set default exit
hr = mySetCertRegMultiStrValueEx(
dwUpgradeFlags,
pServer->pwszSanitizedName,
wszREGKEYEXITMODULES,
NULL,
wszREGACTIVE,
(fUpgrade && (NULL != pServer->pwszzCustomExit)) ?
pServer->pwszzCustomExit :
wszCLASS_CERTEXIT L"\0");
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValueEx", wszREGACTIVE);
// set some absolute keys and values
hr = mySetAbsRegMultiStrValue(
wszREGKEYKEYSNOTTORESTORE,
wszREGRESTORECERTIFICATEAUTHORITY,
wszzREGVALUERESTORECERTIFICATEAUTHORITY);
_JumpIfError(hr, error, "mySetAbsRegMultiStrValue");
hr = helperGetFilesNotToRestore(pComp, &pwszzFilesNotToRestore);
_JumpIfError(hr, error, "helperGetFilesNotToRestore");
hr = mySetAbsRegMultiStrValue(
wszREGKEYFILESNOTTOBACKUP,
wszREGRESTORECERTIFICATEAUTHORITY,
pwszzFilesNotToRestore);
_JumpIfError(hr, error, "mySetAbsRegMultiStrValue");
// ICertGetConfig
hr = StringFromCLSID(CLSID_CCertGetConfig, &pwszCLSIDCertGetConfig);
_JumpIfError(hr, error, "StringFromCLSID(CCertGetConfig)");
hr = mySetAbsRegStrValue(
wszREGKEYKEYRING,
wszREGCERTGETCONFIG,
pwszCLSIDCertGetConfig);
_JumpIfError(hr, error, "mySetAbsRegStrValue");
// ICertCertRequest
hr = StringFromCLSID(CLSID_CCertRequest, &pwszCLSIDCertRequest);
_JumpIfError(hr, error, "StringFromCLSID(CCertRequest)");
hr = mySetAbsRegStrValue(
wszREGKEYKEYRING,
wszREGCERTREQUEST,
pwszCLSIDCertRequest);
_JumpIfError(hr, error, "mySetAbsRegStrValue");
if (NULL != pServer->pCSPInfo &&
NULL != pServer->pHashInfo)
{
WCHAR const *pwszProvName = pServer->pCSPInfo->pwszProvName;
DWORD dwProvType;
ALG_ID idAlg;
BOOL fMachineKeyset;
DWORD dwKeySize;
if (0 == lstrcmpi(pwszProvName, MS_DEF_PROV_W))
{
pwszProvName = MS_STRONG_PROV_W;
}
hr = SetCertSrvCSP(
FALSE, // fEncryptionCSP
pServer->pwszSanitizedName,
pServer->pCSPInfo->dwProvType,
pwszProvName,
pServer->pHashInfo->idAlg,
pServer->pCSPInfo->fMachineKeyset,
0); // dwKeySize
_JumpIfError(hr, error, "SetCertSrvCSP");
hr = myGetCertSrvCSP(
TRUE, // fEncryptionCSP
pServer->pwszSanitizedName,
&dwProvType,
&pwszProvNameReg,
&idAlg,
&fMachineKeyset,
&dwKeySize); // pdwKeySize
if (S_OK != hr)
{
_PrintError(hr, "myGetCertSrvCSP");
dwProvType = pServer->pCSPInfo->dwProvType;
idAlg = CALG_3DES;
fMachineKeyset = pServer->pCSPInfo->fMachineKeyset;
dwKeySize = 1024;
}
else if (NULL != pwszProvNameReg && L'\0' != *pwszProvNameReg)
{
pwszProvName = pwszProvNameReg;
if (0 == lstrcmpi(pwszProvName, MS_DEF_PROV_W))
{
pwszProvName = MS_STRONG_PROV_W;
}
}
hr = SetCertSrvCSP(
TRUE, // fEncryptionCSP
pServer->pwszSanitizedName,
dwProvType,
pwszProvName,
idAlg,
fMachineKeyset,
dwKeySize); // dwKeySize
_JumpIfError(hr, error, "SetCertSrvCSP");
}
hr = S_OK;
error:
if (NULL != pwszProvNameReg)
{
LocalFree(pwszProvNameReg);
}
myLdapClose(pld, strDomainDN, strConfigDN);
if (NULL != pwszCLSIDCertGetConfig)
{
CoTaskMemFree(pwszCLSIDCertGetConfig);
}
if (NULL != pwszCRLPeriod)
{
LocalFree(pwszCRLPeriod);
}
if (NULL != pwszCRLDeltaPeriod)
{
LocalFree(pwszCRLDeltaPeriod);
}
if (NULL != pwszCLSIDCertRequest)
{
CoTaskMemFree(pwszCLSIDCertRequest);
}
if (NULL != pwszzCRLPublicationValue)
{
LocalFree(pwszzCRLPublicationValue);
}
if (NULL != pwszzCACertPublicationValue)
{
LocalFree(pwszzCACertPublicationValue);
}
if (NULL != pwszzRequestExtensionList)
{
LocalFree(pwszzRequestExtensionList);
}
if (NULL != pwszzDisableExtensionList)
{
LocalFree(pwszzDisableExtensionList);
}
if (NULL != pwszzFilesNotToRestore)
{
LocalFree(pwszzFilesNotToRestore);
}
if (NULL != hKeyBase)
{
RegCloseKey(hKeyBase);
}
CSILOG(hr, IDS_LOG_CREATE_SERVER_REG, NULL, NULL, NULL);
return(hr);
}
HRESULT
UpgradeRevocationURLReplaceParam(
IN BOOL fPolicy,
IN BOOL fMultiString,
IN WCHAR const *pwszSanitizedName,
IN WCHAR const *pwszValueName)
{
HRESULT hr;
WCHAR *pwszzValue = NULL;
WCHAR *pwsz;
BOOL fModified = FALSE;
CSASSERT(
WSZARRAYSIZE(wszFCSAPARM_CERTFILENAMESUFFIX) ==
WSZARRAYSIZE(wszFCSAPARM_CRLFILENAMESUFFIX));
// getMultiStr will read REG_SZs as well and double-terminate
hr = myGetCertRegMultiStrValue(
pwszSanitizedName,
fPolicy ? wszREGKEYPOLICYMODULES : wszREGKEYEXITMODULES,
fPolicy ? wszCLASS_CERTPOLICY : wszCLASS_CERTEXIT,
pwszValueName,
&pwszzValue);
_JumpIfErrorStr2(hr, error, "myGetCertRegMultiStrValue", pwszValueName, hr);
for (pwsz = pwszzValue; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
{
WCHAR *pwszT = pwsz;
// Replace wszFCSAPARM_CERTFILENAMESUFFIX with
// wszFCSAPARM_CRLFILENAMESUFFIX. Beta 3's registry values incorrectly
// used a Cert Suffix instead of CRL Suffix.
while (TRUE)
{
DWORD i;
pwszT = wcschr(pwszT, wszFCSAPARM_CERTFILENAMESUFFIX[0]);
if (NULL == pwszT)
{
break;
}
for (i = 1; ; i++)
{
if (i == WSZARRAYSIZE(wszFCSAPARM_CERTFILENAMESUFFIX))
{
CopyMemory(
pwszT,
wszFCSAPARM_CRLFILENAMESUFFIX,
i * sizeof(WCHAR));
pwszT += i;
fModified = TRUE;
break;
}
if (pwszT[i] != wszFCSAPARM_CERTFILENAMESUFFIX[i])
{
pwszT++;
break;
}
}
}
}
if (fModified)
{
if (fMultiString)
{
// set as REG_MULTI_SZ
hr = mySetCertRegMultiStrValue(
pwszSanitizedName,
fPolicy ? wszREGKEYPOLICYMODULES : wszREGKEYEXITMODULES,
fPolicy ? wszCLASS_CERTPOLICY : wszCLASS_CERTEXIT,
pwszValueName,
pwszzValue);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValue", pwszValueName);
}
else
{
// set as REG_SZ
hr = mySetCertRegStrValue(
pwszSanitizedName,
fPolicy ? wszREGKEYPOLICYMODULES : wszREGKEYEXITMODULES,
fPolicy ? wszCLASS_CERTPOLICY : wszCLASS_CERTEXIT,
pwszValueName,
pwszzValue);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValue", pwszValueName);
}
}
error:
if (NULL != pwszzValue)
{
LocalFree(pwszzValue);
}
CSASSERT(S_OK == hr || FAILED(hr));
return(hr);
}
HRESULT
UpgradeRevocationURLRemoveParam(
IN WCHAR const *pwszSanitizedName,
IN WCHAR const *pwszValueName)
{
HRESULT hr;
WCHAR *pwszValue = NULL;
WCHAR *pwsz;
BOOL fModified = FALSE;
WCHAR *pwszT;
hr = myGetCertRegStrValue(
pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
pwszValueName,
&pwszValue);
_JumpIfErrorStr2(hr, error, "myGetCertRegStrValue", pwszValueName, hr);
pwszT = pwszValue;
// Remove wszFCSAPARM_CERTFILENAMESUFFIX from the Netscape Revocaton URL
// It should never have been written out in Beta 3's registry value.
while (TRUE)
{
DWORD i;
pwszT = wcschr(pwszT, wszFCSAPARM_CERTFILENAMESUFFIX[0]);
if (NULL == pwszT)
{
break;
}
for (i = 1; ; i++)
{
if (i == WSZARRAYSIZE(wszFCSAPARM_CERTFILENAMESUFFIX))
{
MoveMemory(
pwszT,
&pwszT[i],
(wcslen(&pwszT[i]) + 1) * sizeof(WCHAR));
pwszT += i;
fModified = TRUE;
break;
}
if (pwszT[i] != wszFCSAPARM_CERTFILENAMESUFFIX[i])
{
pwszT++;
break;
}
}
}
if (fModified)
{
hr = mySetCertRegStrValue(
pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
pwszValueName,
pwszValue);
_JumpIfErrorStr(hr, error, "mySetCertRegStrValue", pwszValueName);
}
error:
if (NULL != pwszValue)
{
LocalFree(pwszValue);
}
CSASSERT(S_OK == hr || FAILED(hr));
return(hr);
}
WCHAR const *apwszB3ExitEntriesToFix[] =
{
wszREGLDAPREVOCATIONDNTEMPLATE_OLD,
NULL
};
HRESULT
UpgradeCRLPath(
WCHAR const *pwszSanitizedName)
{
HRESULT hr;
WCHAR *pwszzCRLPath = NULL;
WCHAR *pwszzFixedCRLPath = NULL;
WCHAR *pwsz;
BOOL fRenewReady = TRUE;
DWORD dwSize = 0;
// get current crl path
hr = myGetCertRegMultiStrValue(
pwszSanitizedName,
NULL,
NULL,
wszREGCRLPATH_OLD,
&pwszzCRLPath);
_JumpIfErrorStr(hr, error, "myGetCertRegStrValue", wszREGCRLPATH_OLD);
// to see if it is in renew ready format
for (pwsz = pwszzCRLPath; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
{
dwSize += wcslen(pwsz) + 1;
if (NULL == wcsstr(pwsz, wszFCSAPARM_CRLFILENAMESUFFIX))
{
// found one without suffix
fRenewReady = FALSE;
// add suffix len
dwSize += WSZARRAYSIZE(wszFCSAPARM_CRLFILENAMESUFFIX);
}
}
if (!fRenewReady)
{
++dwSize; // multi string
// at least one of crl path missed suffix
pwszzFixedCRLPath = (WCHAR*)LocalAlloc(LMEM_FIXED,
dwSize * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pwszzFixedCRLPath);
WCHAR *pwszPt = pwszzFixedCRLPath;
for (pwsz = pwszzCRLPath; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
{
// copy over whole path 1st
wcscpy(pwszPt, pwsz);
if (NULL == wcsstr(pwszPt, wszFCSAPARM_CRLFILENAMESUFFIX))
{
// miss suffix, find file portion
WCHAR *pwszFile = wcsrchr(pwszPt, L'\\');
if (NULL == pwszFile)
{
// may be relative path, point to begin
pwszFile = pwszPt;
}
// find crl extension portion
WCHAR *pwszCRLExt = wcsrchr(pwszFile, L'.');
if (NULL != pwszCRLExt)
{
// insert suffix
wcscpy(pwszCRLExt, wszFCSAPARM_CRLFILENAMESUFFIX);
// add extension portion from original buffer
wcscat(pwszCRLExt,
pwsz + SAFE_SUBTRACT_POINTERS(pwszCRLExt, pwszPt));
}
else
{
// no crl file extension, append suffix at end
wcscat(pwszPt, wszFCSAPARM_CRLFILENAMESUFFIX);
}
}
// update pointer
pwszPt += wcslen(pwszPt) + 1;
}
// mutil string
*pwszPt = L'\0';
CSASSERT(dwSize == SAFE_SUBTRACT_POINTERS(pwszPt, pwszzFixedCRLPath) + 1);
// reset crl path with the fixed crl path
hr = mySetCertRegMultiStrValue(
pwszSanitizedName,
NULL,
NULL,
wszREGCRLPATH_OLD,
pwszzFixedCRLPath);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValue", wszREGCRLPATH_OLD);
}
hr = S_OK;
error:
if (NULL != pwszzCRLPath)
{
LocalFree(pwszzCRLPath);
}
if (NULL != pwszzFixedCRLPath)
{
LocalFree(pwszzFixedCRLPath);
}
return hr;
}
typedef struct _URLPREFIXSTRUCT
{
WCHAR const *pwszURLPrefix;
DWORD dwURLFlags;
} URLPREFIXSTRUCT;
//array of cdp url type and its default usage which is prefix of url
URLPREFIXSTRUCT aCDPURLPrefixList[] =
{
{L"file:", CSURL_ADDTOCERTCDP | CSURL_ADDTOFRESHESTCRL | CSURL_ADDTOCRLCDP},
{L"http:", CSURL_ADDTOCERTCDP | CSURL_ADDTOFRESHESTCRL | CSURL_ADDTOCRLCDP},
{L"ldap:", CSURL_SERVERPUBLISH | CSURL_ADDTOCERTCDP | CSURL_ADDTOFRESHESTCRL | CSURL_ADDTOCRLCDP},
{NULL, 0}
};
//array of aia url type and its default usage which is prefix of url
URLPREFIXSTRUCT aAIAURLPrefixList[] =
{
{L"file:", CSURL_ADDTOCERTCDP},
{L"http:", CSURL_ADDTOCERTCDP},
{L"ldap:", CSURL_ADDTOCERTCDP | CSURL_SERVERPUBLISH},
{NULL, 0}
};
#define wszURLPREFIXFORMAT L"%d:"
//pass an old url, determine what is prefix in a format of "XX:"
HRESULT
DetermineURLPrefixFlags(
IN BOOL fDisabled,
IN BOOL fCDP,
IN WCHAR const *pwszURL,
IN WCHAR *pwszPrefixFlags)
{
HRESULT hr;
URLPREFIXSTRUCT *pURLPrefix;
DWORD dwPathFlags;
WCHAR *pwszT;
WCHAR *pwszLower = NULL;
//default to disable
wsprintf(pwszPrefixFlags, wszURLPREFIXFORMAT, 0);
if (fDisabled)
{
//easy
goto done;
}
if (myIsFullPath(pwszURL, &dwPathFlags))
{
//local path, easy
wsprintf(pwszPrefixFlags, wszURLPREFIXFORMAT, CSURL_SERVERPUBLISH);
goto done;
}
//make lower case url string
pwszLower = (WCHAR*)LocalAlloc(LMEM_FIXED,
(wcslen(pwszURL) + 1) * sizeof(WCHAR));
if (NULL == pwszLower)
{
hr = E_OUTOFMEMORY;
_JumpError(hr, error, "LocalAlloc");
}
wcscpy(pwszLower, pwszURL);
CharLower(pwszLower);
//loop through to find out url type
for (pURLPrefix = fCDP ? aCDPURLPrefixList : aAIAURLPrefixList;
NULL != pURLPrefix->pwszURLPrefix; ++pURLPrefix)
{
pwszT = wcsstr(pwszLower, pURLPrefix->pwszURLPrefix);
if (0 == wcsncmp(pwszLower, pURLPrefix->pwszURLPrefix,
wcslen(pURLPrefix->pwszURLPrefix)))
{
//prefix appears at the begining
wsprintf(pwszPrefixFlags, wszURLPREFIXFORMAT, pURLPrefix->dwURLFlags);
goto done;
}
}
//nothing matches, keep 0 flag
done:
hr = S_OK;
error:
if (NULL != pwszLower)
{
LocalFree(pwszLower);
}
return hr;
}
//move old cdp or aia url from policy to a new location under ca
HRESULT
UpgradeMoveURLsLocation(
IN BOOL fCDP,
IN WCHAR const *pwszSanitizedName,
IN WCHAR const *pwszValueName)
{
HRESULT hr;
WCHAR *pwszzValue = NULL;
WCHAR *pwszzURLs = NULL;
BOOL fDisabled;
DWORD cURLs = 0; //count of url from multi_sz
DWORD dwLen = 0;
DWORD dwSize = 0; //total size of chars in multi_sz url exluding '-'
WCHAR *pwsz;
WCHAR *pwszT;
WCHAR *pwszNoMinus;
WCHAR wszPrefixFlags[9]; //should be enough
// get urls in the old location
hr = myGetCertRegMultiStrValue(
pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
pwszValueName,
&pwszzValue);
_JumpIfErrorStr2(hr, error, "myGetCertRegMultiStrValue", pwszValueName, hr);
// fix "-" prefix for disable and count size
for (pwsz = pwszzValue; L'\0' != *pwsz; pwsz += dwLen + 1)
{
//current url length
dwLen = wcslen(pwsz);
//update size
dwSize += dwLen;
++cURLs;
pwszNoMinus = pwsz;
while (L'-' == *pwszNoMinus)
{
//exclude prefix '-'s
--dwSize;
++pwszNoMinus;
}
}
//allocate buffer in "XX:URL" format
pwszzURLs = (WCHAR*)LocalAlloc(LMEM_FIXED,
(dwSize + cURLs * sizeof(wszPrefixFlags) + 1) * sizeof(WCHAR));
if (NULL == pwszzURLs)
{
hr = E_OUTOFMEMORY;
_JumpError(hr, error, "LocalAlloc");
}
pwszT = pwszzURLs;
//form string in new url format
for (pwsz = pwszzValue; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
{
fDisabled = FALSE;
pwszNoMinus = pwsz;
while (L'-' == *pwszNoMinus)
{
//exclude prefix '-'s
++pwszNoMinus;
fDisabled = TRUE;
}
hr = DetermineURLPrefixFlags(fDisabled, fCDP, pwszNoMinus, wszPrefixFlags);
_JumpIfErrorStr(hr, error, "DetermineURLPrefixFlags", pwszNoMinus);
//format "xx:url"
wcscpy(pwszT, wszPrefixFlags);
wcscat(pwszT, pwszNoMinus);
//ready for next url
pwszT += wcslen(pwszT) + 1;
}
//zz
*pwszT = L'\0';
pwszT = fCDP ? wszREGCRLPUBLICATIONURLS : wszREGCACERTPUBLICATIONURLS,
//move or merge to ca
hr = mySetCertRegMultiStrValueEx(
CSREG_UPGRADE | CSREG_MERGE,
pwszSanitizedName,
NULL,
NULL,
pwszT,
pwszzURLs);
_JumpIfErrorStr(hr, error, "mySetCertRegMultiStrValue", pwszT);
//remove url under policy
hr = myDeleteCertRegValue(
pwszSanitizedName,
wszREGKEYPOLICYMODULES,
wszCLASS_CERTPOLICY,
pwszValueName);
_PrintIfErrorStr(hr, "myGetCertRegMultiStrValue", pwszValueName);
hr = S_OK;
error:
if (NULL != pwszzValue)
{
LocalFree(pwszzValue);
}
if (NULL != pwszzURLs)
{
LocalFree(pwszzURLs);
}
return(hr);
}
WCHAR const *apwszPolicyCDPEntriesToFix[] =
{
wszREGLDAPREVOCATIONCRLURL_OLD, //"LDAPRevocationCRLURL"
wszREGREVOCATIONCRLURL_OLD, //"RevocationCRLURL"
wszREGFTPREVOCATIONCRLURL_OLD, //"FTPRevocationCRLURL"
wszREGFILEREVOCATIONCRLURL_OLD, //"FileRevocationCRLURL"
NULL
};
HRESULT
myPrintIfError(
IN HRESULT hrNew,
IN HRESULT hrOld,
IN CHAR const *psz,
IN OPTIONAL WCHAR const *pwsz)
{
if (S_OK != hrNew)
{
if (NULL != pwsz)
{
_PrintErrorStr(hrNew, psz, pwsz);
}
else
{
_PrintError(hrNew, psz);
}
if (S_OK == hrOld)
{
//save only oldest err
hrOld = hrNew;
}
}
return hrOld;
}
HRESULT
UpgradePolicyCDPURLs(
IN WCHAR const *pwszSanitizedName)
{
HRESULT hr = S_OK;
HRESULT hr2;
WCHAR const **ppwsz;
for (ppwsz = apwszPolicyCDPEntriesToFix; NULL != *ppwsz; ppwsz++)
{
// all entries are multi-valued
hr2 = UpgradeRevocationURLReplaceParam(
TRUE, TRUE, pwszSanitizedName, *ppwsz);
hr = myPrintIfError(hr2, hr, "UpgradeRevocationURLReplaceParam", *ppwsz);
hr2 = UpgradeMoveURLsLocation(TRUE, pwszSanitizedName, *ppwsz);
hr = myPrintIfError(hr2, hr, "UpgradeMoveURLsLocation", *ppwsz);
}
hr2 = UpgradeRevocationURLRemoveParam(pwszSanitizedName, wszREGREVOCATIONURL);
hr = myPrintIfError(hr2, hr, "UpgradeRevocationURLRemoveParam", wszREGREVOCATIONURL);
hr2 = UpgradeCRLPath(pwszSanitizedName);
hr = myPrintIfError(hr2, hr, "UpgradeCRLPath", NULL);
return(hr);
}
WCHAR const *apwszPolicyAIAEntriesToFix[] =
{
wszREGLDAPISSUERCERTURL_OLD, //"LDAPIssuerCertURL"
wszREGISSUERCERTURL_OLD, //"IssuerCertURL"
wszREGFTPISSUERCERTURL_OLD, //"FTPIssuerCertURL"
wszREGFILEISSUERCERTURL_OLD, //"FileIssuerCertURL"
NULL
};
HRESULT
UpgradePolicyAIAURLs(
IN WCHAR const *pwszSanitizedName)
{
HRESULT hr = S_OK;
HRESULT hr2;
WCHAR const **ppwsz;
for (ppwsz = apwszPolicyAIAEntriesToFix; NULL != *ppwsz; ppwsz++)
{
// all entries are multi-valued
hr2 = UpgradeMoveURLsLocation(FALSE, pwszSanitizedName, *ppwsz);
hr = myPrintIfError(hr2, hr, "UpgradeMoveURLsLocation", *ppwsz);
}
return(hr);
}
HRESULT
UpgradeExitRevocationURLs(
IN WCHAR const *pwszSanitizedName)
{
WCHAR const **ppwsz;
for (ppwsz = apwszB3ExitEntriesToFix; NULL != *ppwsz; ppwsz++)
{
// all entries are single-valued
UpgradeRevocationURLReplaceParam(FALSE, FALSE, pwszSanitizedName, *ppwsz);
}
return(S_OK);
}
// following code to determine if current policy/exit modules are custom
// if find any custom module and assign it to
// pServer->pwszCustomPolicy/Exit
// otherwise pServer->pwszCustomPolicy/Exit = NULL means default as active
#define wszCERTSRV10POLICYPROGID L"CertificateAuthority.Policy"
#define wszCERTSRV10EXITPROGID L"CertificateAuthority.Exit"
#define wszCLSID L"ClsID\\"
#define wszINPROCSERVER32 L"\\InprocServer32"
HRESULT
DetermineServerCustomModule(
PER_COMPONENT_DATA *pComp,
IN BOOL fPolicy)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
// init
if (fPolicy)
{
if (NULL != pServer->pwszCustomPolicy)
{
LocalFree(pServer->pwszCustomPolicy);
pServer->pwszCustomPolicy = NULL;
}
}
else
{
if (NULL != pServer->pwszzCustomExit)
{
LocalFree(pServer->pwszzCustomExit);
pServer->pwszzCustomExit = NULL;
}
}
// build to build
// to pass what is the current active policy
if (fPolicy)
{
// policy module
hr = myGetCertRegStrValue(
pServer->pwszSanitizedName,
wszREGKEYPOLICYMODULES,
NULL,
wszREGACTIVE,
&pServer->pwszCustomPolicy);
_JumpIfError(hr, done, "myGetCertRegStrValue");
}
else
{
// exit module
hr = myGetCertRegMultiStrValue(
pServer->pwszSanitizedName,
wszREGKEYEXITMODULES,
NULL,
wszREGACTIVE,
&pServer->pwszzCustomExit);
_JumpIfError(hr, done, "myGetCertRegStrValue");
}
done:
hr = S_OK;
//error:
return hr;
}
HRESULT
UpgradeServerRegEntries(
HWND hwnd,
PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
WCHAR *pwszCRLPeriodString = NULL;
BOOL fUseNewCRLPublish = FALSE;
DWORD dwCRLPeriodCount;
DWORD Count;
BOOL fUpgradeW2K = CS_UPGRADE_WIN2000 == pComp->UpgradeFlag;
CSASSERT(
NULL != pServer->pwszSanitizedName &&
NULL != pServer->pccUpgradeCert);
// Description:
// - if upgrade and get this point, all necessary data structure
// should be loaded and created in LoadAndDetermineServerUpgradeInfo()
// - in this module, check all different upgrade cases,
// upgrade (move) reg entries
// - remove old unused reg entries if upgrade
// Note: each of above steps applys from config level down to ca then
// to policy, etc.
// - lastly call CreateServerRegEntries with upgrade flag
// CONFIGURATION LEVEL
// CA LEVEL
hr = myGetCARegHashCount(
pServer->pwszSanitizedName,
CSRH_CASIGCERT,
&Count);
_JumpIfError(hr, error, "myGetCARegHashCount");
if (0 == Count)
{
hr = mySetCARegHash(
pServer->pwszSanitizedName,
CSRH_CASIGCERT,
0, // iCert
pServer->pccUpgradeCert);
_JumpIfError(hr, error, "mySetCARegHash");
}
// POLICY LEVEL
{
//could fix two things, 1) W2K from B3 needs fixing token plus 2) or
// 2) W2K needs fix CDP location
hr = UpgradePolicyCDPURLs(pServer->pwszSanitizedName);
_PrintIfError(hr, "UpgradePolicyCDPURLs");
hr = UpgradePolicyAIAURLs(pServer->pwszSanitizedName);
_PrintIfError(hr, "UpgradePolicyAIAURLs");
hr = UpgradeExitRevocationURLs(pServer->pwszSanitizedName);
_PrintIfError(hr, "UpgradeExitRevocationURLs");
//UNDONE, we need move url for cdp and aia under policy to ca level
}
// EXIT LEVEL
// DELETE OLD AND UNUSED ENTRIES
hr = CreateServerRegEntries(hwnd, TRUE, pComp);
_JumpIfError(hr, error, "CreateServerRegEntries");
// hr = S_OK;
error:
if (NULL != pwszCRLPeriodString)
{
LocalFree(pwszCRLPeriodString);
}
CSILOG(hr, IDS_LOG_UPGRADE_SERVER_REG, NULL, NULL, NULL);
return(hr);
}
HRESULT
RegisterAndUnRegisterDLLs(
IN DWORD Flags,
IN PER_COMPONENT_DATA *pComp,
IN HWND hwnd)
{
HRESULT hr;
HMODULE hMod = NULL;
typedef HRESULT (STDAPICALLTYPE FNDLLREGISTERSERVER)(VOID);
FNDLLREGISTERSERVER *pfnRegister;
CHAR const *pszFuncName;
REGISTERDLL const *prd;
WCHAR wszString[MAX_PATH];
UINT errmode = 0;
BOOL fCoInit = FALSE;
hr = CoInitialize(NULL);
if (S_OK != hr && S_FALSE != hr)
{
_JumpError(hr, error, "CoInitialize");
}
fCoInit = TRUE;
errmode = SetErrorMode(SEM_FAILCRITICALERRORS);
pszFuncName = 0 == (RD_UNREGISTER & Flags)?
aszRegisterServer[0] : aszRegisterServer[1];
for (prd = g_aRegisterDll; NULL != prd->pwszDllName; prd++)
{
if ((Flags & RD_UNREGISTER) &&
((Flags & RD_SKIPUNREGPOLICY) && (prd->Flags & RD_SKIPUNREGPOLICY) ||
(Flags & RD_SKIPUNREGEXIT) && (prd->Flags & RD_SKIPUNREGEXIT) ||
(Flags & RD_SKIPUNREGMMC) && (prd->Flags & RD_SKIPUNREGMMC)))
{
// case of upgrade path & this dll doesn't want to unreg
continue;
}
if (Flags & prd->Flags)
{
if (NULL != g_pwszArgvPath)
{
wcscpy(wszString, g_pwszArgvPath);
if (L'\0' != wszString[0] &&
L'\\' != wszString[wcslen(wszString) - 1])
{
wcscat(wszString, L"\\");
}
}
else
{
wcscpy(wszString, pComp->pwszSystem32);
}
wcscat(wszString, prd->pwszDllName);
hMod = LoadLibrary(wszString);
if (NULL == hMod)
{
hr = myHLastError();
if (0 == (RD_UNREGISTER & Flags) &&
(!(RD_WHISTLER & prd->Flags) || IsWhistler()))
{
SaveCustomMessage(pComp, wszString);
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_DLLFUNCTION_CALL,
hr,
wszString);
_JumpErrorStr(hr, error, "DllRegisterServer", wszString);
CSILOG(hr, IDS_LOG_DLLS_REGISTERED, wszString, NULL, NULL);
_JumpErrorStr(hr, error, "LoadLibrary", wszString);
}
hr = S_OK;
continue;
}
pfnRegister = (FNDLLREGISTERSERVER *) GetProcAddress(
hMod,
pszFuncName);
if (NULL == pfnRegister)
{
hr = myHLastError();
_JumpErrorStr(hr, error, "GetProcAddress", wszString);
}
__try
{
hr = (*pfnRegister)();
}
__except(hr = myHEXCEPTIONCODE(), EXCEPTION_EXECUTE_HANDLER)
{
}
FreeLibrary(hMod);
hMod = NULL;
if (S_OK != hr)
{
CSILOG(
hr,
(RD_UNREGISTER & Flags)?
IDS_LOG_DLLS_UNREGISTERED : IDS_LOG_DLLS_REGISTERED,
wszString,
NULL,
NULL);
if (0 == (RD_UNREGISTER & Flags))
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_DLLFUNCTION_CALL,
hr,
wszString);
_JumpErrorStr(hr, error, "DllRegisterServer", wszString);
}
else
{
_PrintErrorStr(hr, "DllUnregisterServer", wszString);
}
}
}
}
hr = S_OK;
error:
if (NULL != hMod)
{
FreeLibrary(hMod);
}
SetErrorMode(errmode);
if (fCoInit)
{
CoUninitialize();
}
if (S_OK == hr)
{
CSILOG(
hr,
(RD_UNREGISTER & Flags)?
IDS_LOG_DLLS_UNREGISTERED : IDS_LOG_DLLS_REGISTERED,
NULL,
NULL,
NULL);
}
return(hr);
}
HRESULT
CreateProgramGroups(
BOOL fClient,
PER_COMPONENT_DATA *pComp,
HWND hwnd)
{
HRESULT hr;
PROGRAMENTRY const *ppe;
WCHAR const *pwszLinkName = NULL;
DWORD Flags = fClient? PE_CLIENT : PE_SERVER;
DBGPRINT((
DBG_SS_CERTOCMI,
"CreateProgramGroups: %ws\n",
fClient? L"Client" : L"Server"));
for (ppe = g_aProgramEntry; ppe < &g_aProgramEntry[CPROGRAMENTRY]; ppe++)
{
if ((Flags & ppe->Flags) && 0 == (PE_DELETEONLY & ppe->Flags))
{
WCHAR const *pwszGroupName;
WCHAR const *pwszDescription;
WCHAR awc[MAX_PATH];
WCHAR const *pwszArgs;
wcscpy(awc, pComp->pwszSystem32);
wcscat(awc, ppe->pwszExeName);
pwszArgs = fClient? ppe->pwszClientArgs : ppe->pwszServerArgs;
if (NULL != pwszArgs)
{
wcscat(awc, L" ");
wcscat(awc, pwszArgs);
}
pwszLinkName = myLoadResourceString(ppe->uiLinkName);
if (NULL == pwszLinkName)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
pwszGroupName = NULL;
if (0 != ppe->uiGroupName)
{
pwszGroupName = myLoadResourceString(ppe->uiGroupName);
if (NULL == pwszGroupName)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
}
pwszDescription = NULL;
if (0 != ppe->uiDescription)
{
pwszDescription = myLoadResourceString(ppe->uiDescription);
if (NULL == pwszDescription)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
}
if (!CreateLinkFile(
ppe->csidl, // CSIDL_*
pwszGroupName, // IN LPCSTR lpSubDirectory
pwszLinkName, // IN LPCSTR lpFileName
awc, // IN LPCSTR lpCommandLine
NULL, // IN LPCSTR lpIconPath
0, // IN INT iIconIndex
NULL, // IN LPCSTR lpWorkingDirectory
0, // IN WORD wHotKey
SW_SHOWNORMAL, // IN INT iShowCmd
pwszDescription)) // IN LPCSTR lpDescription
{
hr = myHLastError();
_PrintErrorStr(hr, "CreateLinkFile", awc);
_JumpErrorStr(hr, error, "CreateLinkFile", pwszLinkName);
}
}
}
hr = S_OK;
error:
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_CREATELINK,
hr,
pwszLinkName);
pComp->fShownErr = TRUE;
}
CSILOG(hr, IDS_LOG_PROGRAM_GROUPS, NULL, NULL, NULL);
return(hr);
}
HRESULT
MakeRevocationPage(
HWND hwnd,
PER_COMPONENT_DATA *pComp,
IN WCHAR const *pwszFile)
{
DWORD hr;
WCHAR *pwszASP = NULL;
WCHAR *pwszConfig = NULL;
HANDLE hFile = NULL;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
#define wszASP1 \
L"<%\r\n" \
L"Response.ContentType = \"application/x-netscape-revocation\"\r\n" \
L"serialnumber = Request.QueryString\r\n" \
L"set Admin = Server.CreateObject(\"CertificateAuthority.Admin\")\r\n" \
L"\r\n" \
L"stat = Admin.IsValidCertificate(\""
#define wszASP2 \
L"\", serialnumber)\r\n" \
L"\r\n" \
L"if stat = 3 then Response.Write(\"0\") else Response.Write(\"1\") end if\r\n" \
L"%>\r\n"
hr = myFormConfigString(pComp->pwszServerName,
pServer->pwszSanitizedName,
&pwszConfig);
_JumpIfError(hr, error, "myFormConfigString");
pwszASP = (WCHAR *) LocalAlloc(
LMEM_FIXED,
(WSZARRAYSIZE(wszASP1) +
wcslen(pwszConfig) +
WSZARRAYSIZE(wszASP2) + 1) * sizeof(WCHAR));
_JumpIfOutOfMemory(hr, error, pwszASP);
wcscpy(pwszASP, wszASP1);
wcscat(pwszASP, pwszConfig);
wcscat(pwszASP, wszASP2);
hFile = CreateFile(
pwszFile, // lpFileName
GENERIC_WRITE, // dwDesiredAccess
0, // dwShareMode
NULL, // lpSecurityAttributes
CREATE_ALWAYS, // dwCreationDisposition
0, // dwFlagsAndAttributes
0); // hTemplateFile
if (INVALID_HANDLE_VALUE == hFile)
{
hr = HRESULT_FROM_WIN32(ERROR_OPEN_FAILED);
_JumpError(hr, error, "CreateFile");
}
hr = myStringToAnsiFile(hFile, pwszASP, -1);
error:
if (hFile)
CloseHandle(hFile);
if (NULL != pwszASP)
{
LocalFree(pwszASP);
}
if (NULL != pwszConfig)
{
LocalFree(pwszConfig);
}
return(hr);
}
VOID
setupDeleteFile(
IN WCHAR const *pwszDir,
IN WCHAR const *pwszFile)
{
HRESULT hr;
WCHAR *pwszFilePath = NULL;
hr = myBuildPathAndExt(pwszDir, pwszFile, NULL, &pwszFilePath);
_JumpIfError(hr, error, "myBuildPathAndExt");
if (!DeleteFile(pwszFilePath))
{
hr = myHLastError();
_PrintErrorStr2(hr, "DeleteFile", pwszFilePath, hr);
}
error:
if (NULL != pwszFilePath)
{
LocalFree(pwszFilePath);
}
}
//+------------------------------------------------------------------------
// Function: RenameMiscTargets(. . . .)
//
// Synopsis: Handles various renaming jobs from the names that things
// are given at installation time to the names that they need
// in their new homes to run properly.
//
// Arguments: None
//
// Returns: DWORD error code.
//
// History: 3/21/97 JerryK Created
//-------------------------------------------------------------------------
HRESULT
RenameMiscTargets(
HWND hwnd,
PER_COMPONENT_DATA *pComp,
BOOL fServer)
{
HRESULT hr = S_OK;
WCHAR wszAspPath[MAX_PATH]; wszAspPath[0] = L'\0';
WCHAR wszCertSrv[MAX_PATH]; wszCertSrv[0] = L'\0';
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
if (fServer)
{
// Create nsrev_<CA Name>.asp
BuildPath(
wszCertSrv,
ARRAYSIZE(wszCertSrv),
pComp->pwszSystem32,
wszCERTENROLLSHAREPATH);
BuildPath(
wszAspPath,
ARRAYSIZE(wszAspPath),
wszCertSrv,
L"nsrev_");
wcscat(wszAspPath, pServer->pwszSanitizedName);
wcscat(wszAspPath, TEXT(".asp"));
CSASSERT(wcslen(wszAspPath) < ARRAYSIZE(wszAspPath));
hr = MakeRevocationPage(hwnd, pComp, wszAspPath);
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_CREATEFILE,
hr,
wszAspPath);
_JumpError(hr, error, "MakeRevocationPage");
}
}
error:
return(hr);
}
HRESULT
CreateCertificateService(
PER_COMPONENT_DATA *pComp,
HWND hwnd)
{
HRESULT hr;
WCHAR const *pwszDisplayName;
SERVICE_DESCRIPTION sd;
SC_HANDLE hSCManager = NULL;
SC_HANDLE hSC = NULL;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
if (NULL != pServer->pwszSharedFolder)
{
// add entry
hr = CreateConfigFiles(pServer->pwszSharedFolder, pComp, FALSE, hwnd);
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_CREATECERTSRVFILE,
hr,
pServer->pwszSharedFolder);
_JumpError(hr, error, "CreateConfigFiles");
}
}
if (NULL != g_pwszArgvPath)
{
wcscpy(g_wszServicePath, g_pwszArgvPath);
if (L'\0' != g_wszServicePath[0] &&
L'\\' != g_wszServicePath[wcslen(g_wszServicePath) - 1])
{
wcscat(g_wszServicePath, L"\\");
}
}
else
{
wcscpy(g_wszServicePath, pComp->pwszSystem32);
}
wcscat(g_wszServicePath, wszCERTSRVEXENAME);
hSCManager = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
if (NULL == hSCManager)
{
hr = myHLastError();
_JumpError(hr, error, "OpenSCManager");
}
if (NULL == g_pwszNoService)
{
pwszDisplayName = myLoadResourceString(IDS_CA_SERVICEDISPLAYNAME);
if (NULL == pwszDisplayName)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
sd.lpDescription = const_cast<WCHAR *>(myLoadResourceString(
IDS_CA_SERVICEDESCRIPTION));
if (NULL == sd.lpDescription)
{
hr = myHLastError();
_JumpError(hr, error, "myLoadResourceString");
}
hSC = CreateService(
hSCManager, // hSCManager
wszSERVICE_NAME, // lpServiceName
pwszDisplayName, // lpDisplayName
SERVICE_ALL_ACCESS, // dwDesiredAccess
SERVICE_WIN32_OWN_PROCESS| // dwServiceType
(pServer->fInteractiveService?
SERVICE_INTERACTIVE_PROCESS:0),
SERVICE_AUTO_START, // dwStartType
SERVICE_ERROR_NORMAL, // dwErrorControl
g_wszServicePath, // lpBinaryPathName
NULL, // lpLoadOrderGroup
NULL, // lplpdwTagId
NULL, // lpDependencies
NULL, // lpServiceStartName
NULL); // lpPassword
if (NULL == hSC)
{
hr = myHLastError();
if (HRESULT_FROM_WIN32(ERROR_DUPLICATE_SERVICE_NAME) != hr &&
HRESULT_FROM_WIN32(ERROR_SERVICE_EXISTS) != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_CREATESERVICE,
hr,
wszSERVICE_NAME);
_JumpError(hr, error, "CreateService");
}
}
if (!ChangeServiceConfig2(
hSC, // hService
SERVICE_CONFIG_DESCRIPTION, // dwInfoLevel
(VOID *) &sd)) // lpInfo
{
// This error is not critical.
hr = myHLastError();
_PrintError(hr, "ChangeServiceConfig2");
}
}
// add event log message DLL (ok, it's really an EXE) as a message source
hr = myAddLogSourceToRegistry(g_wszServicePath, wszSERVICE_NAME);
if (S_OK != hr)
{
CertErrorMessageBox(
pComp->hInstance,
pComp->fUnattended,
hwnd,
IDS_ERR_ADDSOURCETOREGISTRY,
hr,
NULL);
_JumpError(hr, error, "AddLogSourceToRegistry");
}
error:
if (NULL != hSC)
{
CloseServiceHandle(hSC);
}
if (NULL != hSCManager)
{
CloseServiceHandle(hSCManager);
}
CSILOG(hr, IDS_LOG_CREATE_SERVICE, NULL, NULL, NULL);
return(hr);
}
HRESULT
DeleteProgramGroups(
IN BOOL fAll,
IN PER_COMPONENT_DATA *pComp)
{
HRESULT hr;
PROGRAMENTRY const *ppe;
WCHAR const *pwszLinkName;
WCHAR const *pwszGroupName;
for (ppe = g_aProgramEntry; ppe < &g_aProgramEntry[CPROGRAMENTRY]; ppe++)
{
if (fAll || (PE_DELETEONLY & ppe->Flags))
{
pwszLinkName = myLoadResourceString(ppe->uiLinkName);
if (NULL == pwszLinkName)
{
hr = myHLastError();
_PrintError(hr, "myLoadResourceString");
continue;
}
pwszGroupName = NULL;
if (0 != ppe->uiGroupName)
{
pwszGroupName = myLoadResourceString(ppe->uiGroupName);
if (NULL == pwszGroupName)
{
hr = myHLastError();
_PrintError(hr, "myLoadResourceString");
continue;
}
}
if (!DeleteLinkFile(
ppe->csidl, // CSIDL_*
pwszGroupName, // IN LPCSTR lpSubDirectory
pwszLinkName, // IN LPCSTR lpFileName
FALSE)) // IN BOOL fDeleteSubDirectory
{
hr = myHLastError();
_PrintError3(
hr,
"DeleteLinkFile",
HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND),
HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND));
}
}
}
pwszGroupName = myLoadResourceString(IDS_STARTMENU_CERTSERVER);
if (NULL == pwszGroupName)
{
hr = myHLastError();
_PrintError(hr, "myLoadResourceString");
}
else if (!DeleteGroup(pwszGroupName, TRUE))
{
hr = myHLastError();
_PrintError(hr, "DeleteGroup");
}
hr = S_OK;
//error:
return(hr);
}
HRESULT
CancelCertsrvInstallation(
HWND hwnd,
PER_COMPONENT_DATA *pComp)
{
static BOOL s_fCancelled = FALSE;
HRESULT hr;
CASERVERSETUPINFO *pServer = pComp->CA.pServer;
if (s_fCancelled)
{
goto done;
}
if (IS_SERVER_INSTALL & pComp->dwInstallStatus)
{
// uninstall will remove reg entries and others
PreUninstallCore(hwnd, pComp, FALSE);
// Note, GUI mode, we allow re-try post setup in case of cancel or failure
// but unattended mode, only allow once
UninstallCore(hwnd, pComp, 0, 0, FALSE, FALSE, !pComp->fUnattended);
if (pComp->fUnattended)
{
hr = SetSetupStatus(
NULL,
SETUP_CLIENT_FLAG | SETUP_SERVER_FLAG,
FALSE);
}
}
if (NULL != pServer)
{
if (NULL == pServer->pccExistingCert)
{
if (pServer->fSavedCAInDS)
{
// remove ca entry from ds
hr = RemoveCAInDS(pServer->pwszSanitizedName);
if (S_OK == hr)
{
pServer->fSavedCAInDS = FALSE;
}
else
{
_PrintError(hr, "RemoveCAInDS");
}
}
}
// delete the new key container, if necessary.
ClearKeyContainerName(pServer);
DisableVRootsAndShares(pComp->fCreatedVRoot, pServer->fCreatedShare);
}
DBGPRINT((DBG_SS_CERTOCM, "Certsrv setup is cancelled.\n"));
s_fCancelled = TRUE; // only once
done:
hr = S_OK;
//error:
CSILOG(hr, IDS_LOG_CANCEL_INSTALL, NULL, NULL, NULL);
return hr;
}
// Returns true if the specified period is valid. For year it
// should be in the VP_MIN,VP_MAX range. For days/weeks/months,
// we define a separate upper limit to be consistent with the
// attended setup which restricts the edit box to 4 digits.
bool IsValidPeriod(const CASERVERSETUPINFO *pServer)
{
return VP_MIN <= pServer->dwValidityPeriodCount &&
!(ENUM_PERIOD_YEARS == pServer->enumValidityPeriod &&
VP_MAX < pServer->dwValidityPeriodCount) &&
!(ENUM_PERIOD_YEARS != pServer->enumValidityPeriod &&
VP_MAX_DAYS_WEEKS_MONTHS < pServer->dwValidityPeriodCount);
}
HRESULT DeleteCertificates(const WCHAR* pwszSanitizedCAName, BOOL fRoot)
{
HRESULT hr = S_OK, hr2 = S_OK;
DWORD cCACerts, cCACert, dwNameId;
HCERTSTORE hStore = NULL;
CERT_CONTEXT const *pCACert = NULL, *pDupCert;
hr = myGetCARegHashCount(pwszSanitizedCAName, CSRH_CASIGCERT, &cCACerts);
_JumpIfError(hr, error, "myGetCARegHashCount CSRH_CASIGCERT");
hStore = CertOpenStore(
CERT_STORE_PROV_SYSTEM_W,
X509_ASN_ENCODING,
NULL, // hProv
CERT_SYSTEM_STORE_LOCAL_MACHINE,
fRoot?wszROOT_CERTSTORE:wszCA_CERTSTORE);
for(cCACert=0; cCACert<cCACerts; cCACert++)
{
hr2 = myFindCACertByHashIndex(
hStore,
pwszSanitizedCAName,
CSRH_CASIGCERT,
cCACert,
&dwNameId,
&pCACert);
if(S_OK!=hr2)
{
hr = hr2;
_PrintIfError(hr2, "myFindCACertByHashIndex");
continue;
}
if(!CertDeleteCertificateFromStore(pCACert))
{
_PrintError(hr, "CertDeleteCertificateFromStore");
CertFreeCertificateContext(pCACert);
}
pCACert = NULL;
}
error:
CSASSERT(!pCACert);
if (NULL != hStore)
{
CertCloseStore(hStore, CERT_CLOSE_STORE_FORCE_FLAG);
}
return hr;
}
HRESULT TriggerAutoenrollment()
{
HRESULT hr = S_OK;
// must be cleaned up
CAutoHANDLE hEvent;
hEvent=OpenEvent(
EVENT_MODIFY_STATE,
false,
L"Global\\" MACHINE_AUTOENROLLMENT_TRIGGER_EVENT);
if (!hEvent)
{
hr = myHLastError();
_JumpError(hr, error, "OpenEvent");
}
if (!SetEvent(hEvent))
{
hr = myHLastError();
_JumpError(hr, error, "OpenEvent");
}
error:
return hr;
}