mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1205 lines
34 KiB
1205 lines
34 KiB
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows NT Security
|
|
// Copyright (C) Microsoft Corporation, 1997 - 1999
|
|
//
|
|
// File: initpki.cpp
|
|
//
|
|
// Contents: migrates Bob Store to SPC store and adds root certificates
|
|
//
|
|
// History: 03-Jun-97 kirtd Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include "global.hxx"
|
|
#include "cryptreg.h"
|
|
#include "..\mscat32\mscatprv.h"
|
|
|
|
HMODULE hModule = NULL;
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
#define INITPKI_HRESULT_FROM_WIN32(a) ((a >= 0x80000000) ? a : HRESULT_FROM_WIN32(a))
|
|
|
|
|
|
#define SHA1_HASH_LENGTH 20
|
|
|
|
|
|
#define wsz_ROOT_STORE L"Root"
|
|
#define wsz_TRUST_STORE L"Trust"
|
|
#define wsz_CA_STORE L"CA"
|
|
#define wsz_TRUST_PUB_STORE L"TrustedPublisher"
|
|
#define wsz_DISALLOWED_STORE L"Disallowed"
|
|
static LPCWSTR rgpwszPredefinedEnterpriseStore[] = {
|
|
wsz_ROOT_STORE,
|
|
wsz_TRUST_STORE,
|
|
wsz_CA_STORE,
|
|
wsz_TRUST_PUB_STORE,
|
|
wsz_DISALLOWED_STORE
|
|
};
|
|
#define NUM_PREDEFINED_ENTERPRISE_STORE \
|
|
(sizeof(rgpwszPredefinedEnterpriseStore) / \
|
|
sizeof(rgpwszPredefinedEnterpriseStore[0]))
|
|
|
|
void RegisterEnterpriseStores()
|
|
{
|
|
DWORD i;
|
|
|
|
for (i = 0; i < NUM_PREDEFINED_ENTERPRISE_STORE; i++) {
|
|
CertRegisterSystemStore(
|
|
rgpwszPredefinedEnterpriseStore[i],
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE_ENTERPRISE,
|
|
NULL, // pSystemStoreInfo
|
|
NULL // pvReserved
|
|
);
|
|
}
|
|
}
|
|
|
|
void RemoveCert(HCERTSTORE hStore, BYTE *pThumbPrint)
|
|
{
|
|
PCERT_CONTEXT pCertContext;
|
|
CRYPT_HASH_BLOB CryptHashBlob;
|
|
|
|
CryptHashBlob.cbData = SHA1_HASH_LENGTH;
|
|
CryptHashBlob.pbData = pThumbPrint;
|
|
|
|
pCertContext = (PCERT_CONTEXT)CertFindCertificateInStore( hStore,
|
|
X509_ASN_ENCODING,
|
|
0,
|
|
CERT_FIND_SHA1_HASH,
|
|
&CryptHashBlob,
|
|
NULL);
|
|
if (pCertContext)
|
|
{
|
|
CertDeleteCertificateFromStore(pCertContext);
|
|
}
|
|
}
|
|
|
|
//
|
|
// if byte 0 is null, make sure to change while loop below!
|
|
//
|
|
BYTE CertRemoveList[][SHA1_HASH_LENGTH] =
|
|
{
|
|
{ 0x4B, 0x33, 0x8D, 0xCD, 0x50, 0x18, 0x10, 0xB9, 0x36, 0xA0,
|
|
0x63, 0x61, 0x4C, 0x3C, 0xDD, 0x3F, 0xC2, 0xC4, 0x88, 0x55 }, // GTE Glue - '96
|
|
|
|
{ 0x56, 0xB0, 0x65, 0xA7, 0x4B, 0xDC, 0xE3, 0x7C, 0x96, 0xD3,
|
|
0xBA, 0x69, 0x81, 0x08, 0x02, 0xD5, 0x87, 0x03, 0xC0, 0xBD }, // Verisign Comm Glue - '96
|
|
|
|
{ 0x13, 0x39, 0x72, 0xAA, 0x97, 0xD3, 0x65, 0xFB, 0x6A, 0x1D,
|
|
0x47, 0xA5, 0xC7, 0x7A, 0x5C, 0x03, 0x94, 0xBD, 0xB9, 0xED }, // Verisign Indv Glue - '96
|
|
|
|
{ 0x69, 0xD0, 0x4F, 0xFB, 0x62, 0xE1, 0xC9, 0xAE, 0x30, 0x76,
|
|
0x99, 0x2A, 0xE7, 0x46, 0xFD, 0x69, 0x08, 0x3A, 0xBD, 0xE9 }, // MS Root cert - '96
|
|
|
|
{ 0xA7, 0xD7, 0xD5, 0xFD, 0xBB, 0x26, 0xB4, 0x10, 0xC1, 0xD6,
|
|
0x7A, 0xFB, 0xF5, 0xC9, 0x05, 0x39, 0x42, 0xDE, 0xE0, 0xEF }, // MS SGC Root Authority - '99
|
|
|
|
// { 0xCC, 0x7E, 0xD0, 0x77, 0xF0, 0xF2, 0x92, 0x59, 0x5A, 0x81,
|
|
// 0x66, 0xB0, 0x17, 0x09, 0xE2, 0x0C, 0x08, 0x84, 0xA5, 0xF8 }, // verisign "Class1" - '97
|
|
//
|
|
// { 0xD4, 0x73, 0x5D, 0x8A, 0x9A, 0xE5, 0xBC, 0x4B, 0x0A, 0x0D,
|
|
// 0xC2, 0x70, 0xD6, 0xA6, 0x25, 0x38, 0xA5, 0x87, 0xD3, 0x2F }, // verisign "timestamp" - '97
|
|
//
|
|
// { 0x68, 0x8B, 0x6E, 0xB8, 0x07, 0xE8, 0xED, 0xA5, 0xC7, 0xB1,
|
|
// 0x7C, 0x43, 0x93, 0xD0, 0x79, 0x5F, 0x0F, 0xAE, 0x15, 0x5F }, // verisign "commercial" - '97
|
|
//
|
|
// { 0xB1, 0x9D, 0xD0, 0x96, 0xDC, 0xD4, 0xE3, 0xE0, 0xFD, 0x67,
|
|
// 0x68, 0x85, 0x50, 0x5A, 0x67, 0x2C, 0x43, 0x8D, 0x4E, 0x9C }, // verisign "individual" - '97
|
|
|
|
|
|
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } // term
|
|
|
|
};
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: PurgeExpiringCertsFromStores
|
|
//
|
|
// Synopsis: blech!
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT PurgeExpiringCertsFromStores ()
|
|
{
|
|
DWORD cRemove;
|
|
DWORD cStores;
|
|
HCERTSTORE hStore = NULL;
|
|
HKEY hKey = NULL;
|
|
char *pszStores[] = { "SPC", "ROOT", "CU_ROOT", NULL };
|
|
|
|
//
|
|
// HACKHACK! no crypt32 UI about the root store.
|
|
//
|
|
if (RegCreateHKCUKeyExU(HKEY_CURRENT_USER, ROOT_STORE_REGPATH,
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &hKey, NULL) != ERROR_SUCCESS)
|
|
{
|
|
hKey = NULL;
|
|
}
|
|
|
|
cStores = 0;
|
|
|
|
while (pszStores[cStores])
|
|
{
|
|
if (strcmp(pszStores[cStores], "CU_ROOT") == 0)
|
|
{
|
|
if (hKey)
|
|
hStore = CertOpenStore(CERT_STORE_PROV_REG, 0, NULL, 0, (LPVOID)hKey);
|
|
else
|
|
hStore = NULL;
|
|
}
|
|
else
|
|
{
|
|
hStore = CertOpenStore(CERT_STORE_PROV_SYSTEM_A, 0, NULL,
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE |
|
|
CERT_STORE_NO_CRYPT_RELEASE_FLAG,
|
|
pszStores[cStores]);
|
|
}
|
|
|
|
if (hStore)
|
|
{
|
|
cRemove = 0;
|
|
while (CertRemoveList[cRemove][0] != 0x00)
|
|
{
|
|
if (hStore)
|
|
{
|
|
RemoveCert(hStore, &CertRemoveList[cRemove][0]);
|
|
}
|
|
|
|
cRemove++;
|
|
}
|
|
|
|
CertCloseStore(hStore, 0);
|
|
}
|
|
|
|
cStores++;
|
|
}
|
|
|
|
if (hKey)
|
|
RegCloseKey(hKey);
|
|
return( S_OK );
|
|
}
|
|
|
|
PCCERT_CONTEXT FindCertificateInOtherStore(
|
|
IN HCERTSTORE hOtherStore,
|
|
IN PCCERT_CONTEXT pCert
|
|
)
|
|
{
|
|
BYTE rgbHash[SHA1_HASH_LENGTH];
|
|
CRYPT_DATA_BLOB HashBlob;
|
|
|
|
HashBlob.pbData = rgbHash;
|
|
HashBlob.cbData = SHA1_HASH_LENGTH;
|
|
if (!CertGetCertificateContextProperty(
|
|
pCert,
|
|
CERT_SHA1_HASH_PROP_ID,
|
|
rgbHash,
|
|
&HashBlob.cbData
|
|
) || SHA1_HASH_LENGTH != HashBlob.cbData)
|
|
return NULL;
|
|
|
|
return CertFindCertificateInStore(
|
|
hOtherStore,
|
|
0, // dwCertEncodingType
|
|
0, // dwFindFlags
|
|
CERT_FIND_SHA1_HASH,
|
|
(const void *) &HashBlob,
|
|
NULL //pPrevCertContext
|
|
);
|
|
}
|
|
|
|
BOOL IsCertificateInOtherStore(
|
|
IN HCERTSTORE hOtherStore,
|
|
IN PCCERT_CONTEXT pCert
|
|
)
|
|
{
|
|
PCCERT_CONTEXT pOtherCert;
|
|
|
|
if (pOtherCert = FindCertificateInOtherStore(hOtherStore, pCert)) {
|
|
CertFreeCertificateContext(pOtherCert);
|
|
return TRUE;
|
|
} else
|
|
return FALSE;
|
|
}
|
|
|
|
void DeleteCertificateFromOtherStore(
|
|
IN HCERTSTORE hOtherStore,
|
|
IN PCCERT_CONTEXT pCert
|
|
)
|
|
{
|
|
PCCERT_CONTEXT pOtherCert;
|
|
|
|
if (pOtherCert = FindCertificateInOtherStore(hOtherStore, pCert))
|
|
CertDeleteCertificateFromStore(pOtherCert);
|
|
}
|
|
|
|
|
|
//+-------------------------------------------------------------------------
|
|
// Read a SignedData message consisting of certificates and
|
|
// CRLs from memory and copy to the specified cert store.
|
|
//
|
|
// Except for the SPC being loaded from memory, identical to SpcReadSpcFile.
|
|
//
|
|
// For hLMStore != NULL: if the certificate or CRL already exists in the
|
|
// LocalMachine store don't add it. Also if it exists in hCertstore,
|
|
// delete it.
|
|
//--------------------------------------------------------------------------
|
|
HRESULT
|
|
SpcReadSpcFromMemory(
|
|
IN BYTE *pbData,
|
|
IN DWORD cbData,
|
|
IN HCERTSTORE hCertStore,
|
|
IN DWORD dwMsgAndCertEncodingType,
|
|
IN DWORD dwFlags,
|
|
IN OPTIONAL HCERTSTORE hLMStore
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HCERTSTORE hSpcStore = NULL;
|
|
CRYPT_DATA_BLOB sSpcBlob;
|
|
HCRYPTPROV hCryptProv = NULL;
|
|
PCCERT_CONTEXT pCert = NULL;
|
|
PCCRL_CONTEXT pCrl = NULL;
|
|
|
|
if (!(hCertStore))
|
|
{
|
|
goto InvalidArg;
|
|
}
|
|
|
|
// Set the blob data.
|
|
sSpcBlob.pbData = pbData;
|
|
sSpcBlob.cbData = cbData;
|
|
|
|
|
|
// Open up the spc store
|
|
hSpcStore = CertOpenStore(CERT_STORE_PROV_SERIALIZED, //CERT_STORE_PROV_PKCS7,
|
|
X509_ASN_ENCODING | PKCS_7_ASN_ENCODING,
|
|
hCryptProv,
|
|
CERT_STORE_NO_CRYPT_RELEASE_FLAG,
|
|
&sSpcBlob);
|
|
if (!hSpcStore)
|
|
{
|
|
goto CertStoreError;
|
|
}
|
|
|
|
// Copy in the certificates from the caller.
|
|
while (pCert = CertEnumCertificatesInStore(hSpcStore, pCert))
|
|
{
|
|
if (hLMStore && IsCertificateInOtherStore(hLMStore, pCert))
|
|
// Certificate exists in LocalMachine. Delete it from
|
|
// CurrentUser if it already exists there.
|
|
DeleteCertificateFromOtherStore(hCertStore, pCert);
|
|
else
|
|
CertAddCertificateContextToStore(hCertStore,
|
|
pCert,
|
|
CERT_STORE_ADD_REPLACE_EXISTING,
|
|
NULL);
|
|
}
|
|
|
|
while (pCrl = CertEnumCRLsInStore(hSpcStore, pCrl))
|
|
{
|
|
CertAddCRLContextToStore(hCertStore,
|
|
pCrl,
|
|
CERT_STORE_ADD_NEWER,
|
|
NULL);
|
|
|
|
if (hLMStore) {
|
|
// Check if newer or same CRL exists in the hLMStore
|
|
PCCRL_CONTEXT pLMCrl;
|
|
|
|
pLMCrl = CertFindCRLInStore(
|
|
hLMStore,
|
|
pCrl->dwCertEncodingType,
|
|
0, // dwFindFlags
|
|
CRL_FIND_EXISTING,
|
|
(const void *) pCrl,
|
|
NULL // pPrevCrlContext
|
|
);
|
|
|
|
if (NULL != pLMCrl) {
|
|
PCCRL_CONTEXT pCUCrl;
|
|
|
|
pCUCrl = CertFindCRLInStore(
|
|
hCertStore,
|
|
pCrl->dwCertEncodingType,
|
|
0, // dwFindFlags
|
|
CRL_FIND_EXISTING,
|
|
(const void *) pCrl,
|
|
NULL // pPrevCrlContext
|
|
);
|
|
|
|
if (NULL != pCUCrl) {
|
|
if (0 <= CompareFileTime(
|
|
&pLMCrl->pCrlInfo->ThisUpdate,
|
|
&pCUCrl->pCrlInfo->ThisUpdate
|
|
))
|
|
CertDeleteCRLFromStore(pCUCrl);
|
|
else
|
|
CertFreeCRLContext(pCUCrl);
|
|
}
|
|
|
|
CertFreeCRLContext(pLMCrl);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
CommonReturn:
|
|
if (hSpcStore)
|
|
{
|
|
CertCloseStore(hSpcStore, 0);
|
|
}
|
|
return(hr);
|
|
|
|
ErrorReturn:
|
|
SetLastError((DWORD)hr);
|
|
goto CommonReturn;
|
|
|
|
SET_HRESULT_EX(DBG_SS, InvalidArg, E_INVALIDARG);
|
|
SET_HRESULT_EX(DBG_SS, CertStoreError, GetLastError());
|
|
}
|
|
|
|
// For nonNULL pszLMStoreName, doesn't add certificates if already in
|
|
// pszLMStoreName store.
|
|
HRESULT AddCertificates2(
|
|
IN LPCSTR pszStoreName,
|
|
IN OPTIONAL LPCSTR pszLMStoreName,
|
|
IN DWORD dwOpenStoreFlags,
|
|
IN LPCSTR pszResourceName,
|
|
IN LPCSTR pszResourceType
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HCERTSTORE hCertStore = NULL;
|
|
HCERTSTORE hLMStore = NULL;
|
|
LPBYTE pb = NULL;
|
|
DWORD cb;
|
|
HRSRC hrsrc;
|
|
|
|
hCertStore = CertOpenStore(
|
|
CERT_STORE_PROV_SYSTEM_REGISTRY_A,
|
|
0, // dwEncodingType
|
|
NULL, // hCryptProv
|
|
dwOpenStoreFlags |
|
|
CERT_SYSTEM_STORE_UNPROTECTED_FLAG,
|
|
(const void *) pszStoreName
|
|
);
|
|
|
|
if (!(hCertStore))
|
|
{
|
|
return(GetLastError());
|
|
}
|
|
|
|
if (NULL != pszLMStoreName)
|
|
{
|
|
hLMStore = CertOpenStore(
|
|
CERT_STORE_PROV_SYSTEM_REGISTRY_A,
|
|
0, // dwEncodingType
|
|
NULL, // hCryptProv
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE |
|
|
CERT_STORE_READONLY_FLAG |
|
|
CERT_SYSTEM_STORE_UNPROTECTED_FLAG,
|
|
(const void *) pszLMStoreName
|
|
);
|
|
}
|
|
|
|
|
|
hrsrc = FindResourceA(hModule, pszResourceName, pszResourceType);
|
|
if ( hrsrc != NULL )
|
|
{
|
|
HGLOBAL hglobRes;
|
|
|
|
hglobRes = LoadResource(hModule, hrsrc);
|
|
if ( hglobRes != NULL )
|
|
{
|
|
ULONG cbRes;
|
|
BYTE* pbRes;
|
|
|
|
cbRes = SizeofResource(hModule, hrsrc);
|
|
pbRes = (BYTE *)LockResource(hglobRes);
|
|
|
|
hr = SpcReadSpcFromMemory( pbRes,
|
|
cbRes,
|
|
hCertStore,
|
|
PKCS_7_ASN_ENCODING | X509_ASN_ENCODING,
|
|
0,
|
|
hLMStore);
|
|
|
|
UnlockResource(hglobRes);
|
|
FreeResource(hglobRes);
|
|
|
|
}
|
|
else
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
|
|
if ( hCertStore != NULL )
|
|
{
|
|
CertCloseStore(hCertStore, CERT_CLOSE_STORE_FORCE_FLAG);
|
|
}
|
|
|
|
if ( hLMStore != NULL )
|
|
{
|
|
CertCloseStore(hLMStore, CERT_CLOSE_STORE_FORCE_FLAG);
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
|
|
// For non-LocalMachine store, doesn't add certificates if already in
|
|
// corresponding LocalMachine store.
|
|
HRESULT AddCertificates(
|
|
IN LPCSTR pszStoreName,
|
|
IN DWORD dwOpenStoreFlags,
|
|
IN LPCSTR pszResourceName,
|
|
IN LPCSTR pszResourceType
|
|
)
|
|
{
|
|
LPCSTR pszLMStoreName;
|
|
|
|
if (CERT_SYSTEM_STORE_LOCAL_MACHINE !=
|
|
(dwOpenStoreFlags & CERT_SYSTEM_STORE_LOCATION_MASK))
|
|
pszLMStoreName = pszStoreName;
|
|
else
|
|
pszLMStoreName = NULL;
|
|
|
|
return AddCertificates2(
|
|
pszStoreName,
|
|
pszLMStoreName,
|
|
dwOpenStoreFlags,
|
|
pszResourceName,
|
|
pszResourceType
|
|
);
|
|
}
|
|
|
|
HRESULT AddCurrentUserCACertificates()
|
|
{
|
|
return AddCertificates(
|
|
"CA",
|
|
CERT_SYSTEM_STORE_CURRENT_USER,
|
|
MAKEINTRESOURCE(IDR_CAS),
|
|
"CAS"
|
|
);
|
|
}
|
|
HRESULT AddLocalMachineCACertificates()
|
|
{
|
|
return AddCertificates(
|
|
"CA",
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_CAS),
|
|
"CAS"
|
|
);
|
|
}
|
|
|
|
HRESULT AddCurrentUserDisallowedCertificates()
|
|
{
|
|
return AddCertificates(
|
|
"Disallowed",
|
|
CERT_SYSTEM_STORE_CURRENT_USER,
|
|
MAKEINTRESOURCE(IDR_DISALLOW),
|
|
"DISALLOW"
|
|
);
|
|
}
|
|
HRESULT AddLocalMachineDisallowedCertificates()
|
|
{
|
|
return AddCertificates(
|
|
"Disallowed",
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_DISALLOW),
|
|
"DISALLOW"
|
|
);
|
|
}
|
|
|
|
HRESULT AddCurrentUserRootCertificates()
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hr2;
|
|
|
|
hr = AddCertificates(
|
|
"Root",
|
|
CERT_SYSTEM_STORE_CURRENT_USER,
|
|
MAKEINTRESOURCE(IDR_ROOTS),
|
|
"ROOTS"
|
|
);
|
|
|
|
hr2 = AddCertificates2(
|
|
"Root",
|
|
"AuthRoot", // check if already in LM AuthRoot store
|
|
CERT_SYSTEM_STORE_CURRENT_USER,
|
|
MAKEINTRESOURCE(IDR_AUTHROOTS),
|
|
"AUTHROOTS"
|
|
);
|
|
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT AddLocalMachineRootCertificates()
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hr2;
|
|
HRESULT hr3;
|
|
HRESULT hr4;
|
|
|
|
hr = AddCertificates(
|
|
"AuthRoot",
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_AUTHROOTS),
|
|
"AUTHROOTS"
|
|
);
|
|
|
|
// Remove all the AuthRoots from the "Root" store
|
|
hr2 = AddCertificates2(
|
|
"Root",
|
|
"AuthRoot", // check if already in LM AuthRoot store
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_AUTHROOTS),
|
|
"AUTHROOTS"
|
|
);
|
|
|
|
hr3 = AddCertificates(
|
|
"Root",
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_ROOTS),
|
|
"ROOTS"
|
|
);
|
|
|
|
// Remove all the Roots from the "AuthRoot" store
|
|
hr4 = AddCertificates2(
|
|
"AuthRoot",
|
|
"Root", // check if already in LM Root store
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE,
|
|
MAKEINTRESOURCE(IDR_ROOTS),
|
|
"ROOTS"
|
|
);
|
|
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr3;
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr4;
|
|
|
|
return hr;
|
|
}
|
|
|
|
void CreateKey(
|
|
IN HKEY hKey,
|
|
IN LPCWSTR pwszSubKey
|
|
)
|
|
{
|
|
LONG err;
|
|
DWORD dwDisposition;
|
|
HKEY hSubKey;
|
|
|
|
if (ERROR_SUCCESS != (err = RegCreateKeyExU(
|
|
hKey,
|
|
pwszSubKey,
|
|
0, // dwReserved
|
|
NULL, // lpClass
|
|
REG_OPTION_NON_VOLATILE,
|
|
MAXIMUM_ALLOWED,
|
|
NULL, // lpSecurityAttributes
|
|
&hSubKey,
|
|
&dwDisposition))) {
|
|
#if DBG
|
|
DbgPrintf(DBG_SS_CRYPT32,
|
|
"RegCreateKeyEx(%S) returned error: %d 0x%x\n",
|
|
pwszSubKey, err, err);
|
|
#endif
|
|
} else {
|
|
RegCloseKey(hSubKey);
|
|
}
|
|
}
|
|
|
|
// Loop through the certificates in the "My" store and get their
|
|
// KeyIdentifier property. If the certificate also has a KEY_PROV_INFO,
|
|
// then, this will cause its KeyIdentifier to be created.
|
|
void UpdateMyKeyIdentifiers(
|
|
IN DWORD dwOpenStoreFlags
|
|
)
|
|
{
|
|
HCERTSTORE hStore;
|
|
if (hStore = CertOpenStore(
|
|
CERT_STORE_PROV_SYSTEM_A,
|
|
0, // dwEncodingType
|
|
NULL, // hCryptProv
|
|
dwOpenStoreFlags | CERT_STORE_ENUM_ARCHIVED_FLAG,
|
|
(const void *) "My"
|
|
)) {
|
|
PCCERT_CONTEXT pCert = NULL;
|
|
while (pCert = CertEnumCertificatesInStore(hStore, pCert)) {
|
|
DWORD cbData = 0;
|
|
|
|
// Dummy get to force the KeyIdentifer property to be created
|
|
// if it doesn't already exist.
|
|
CertGetCertificateContextProperty(
|
|
pCert,
|
|
CERT_KEY_IDENTIFIER_PROP_ID,
|
|
NULL, // pvData
|
|
&cbData
|
|
);
|
|
}
|
|
|
|
CertCloseStore(hStore, 0);
|
|
}
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Set Software Publisher State Key Value
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
BOOL SetSoftPubKey(DWORD dwMask, BOOL fOn)
|
|
{
|
|
DWORD dwState=0;
|
|
DWORD dwDisposition=0;
|
|
DWORD dwType=0;
|
|
DWORD cbData=0;
|
|
LPWSTR wszState=REGNAME_WINTRUST_POLICY_FLAGS;
|
|
BOOL fResult=FALSE;
|
|
|
|
HKEY hKey=NULL;
|
|
|
|
|
|
// Set the State in the registry
|
|
if (ERROR_SUCCESS != RegCreateKeyExU(
|
|
HKEY_CURRENT_USER,
|
|
REGPATH_WINTRUST_POLICY_FLAGS,
|
|
0, // dwReserved
|
|
NULL, // lpszClass
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_ALL_ACCESS,
|
|
NULL, // lpSecurityAttributes
|
|
&hKey,
|
|
&dwDisposition))
|
|
goto RegErr;
|
|
|
|
|
|
dwState = 0;
|
|
cbData = sizeof(dwState);
|
|
|
|
if(ERROR_SUCCESS != RegQueryValueExU
|
|
( hKey,
|
|
wszState,
|
|
NULL, // lpReserved
|
|
&dwType,
|
|
(BYTE *) &dwState,
|
|
&cbData
|
|
))
|
|
goto RegErr;
|
|
|
|
if ((dwType != REG_DWORD) && (dwType != REG_BINARY))
|
|
goto UnexpectedErr;
|
|
|
|
switch(dwMask)
|
|
{
|
|
case WTPF_IGNOREREVOCATIONONTS:
|
|
case WTPF_IGNOREREVOKATION:
|
|
case WTPF_IGNOREEXPIRATION:
|
|
// Revocation and expiration are a double negative so the bit set
|
|
// means revocation and expriation checking is off.
|
|
fOn = !fOn;
|
|
break;
|
|
default:
|
|
break;
|
|
};
|
|
|
|
if (fOn)
|
|
dwState |= dwMask;
|
|
else
|
|
dwState &= ~dwMask;
|
|
|
|
if(ERROR_SUCCESS != RegSetValueExU(
|
|
hKey,
|
|
wszState,
|
|
0, // dwReserved
|
|
REG_DWORD,
|
|
(BYTE *) &dwState,
|
|
sizeof(dwState)
|
|
))
|
|
goto SetValueErr;
|
|
|
|
|
|
fResult=TRUE;
|
|
|
|
CommonReturn:
|
|
if(hKey)
|
|
RegCloseKey(hKey);
|
|
|
|
return fResult;
|
|
|
|
ErrorReturn:
|
|
|
|
goto CommonReturn;
|
|
|
|
TRACE_ERROR(RegErr);
|
|
SET_ERROR(UnexpectedErr, E_UNEXPECTED);
|
|
TRACE_ERROR(SetValueErr);
|
|
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetNextRegToken
|
|
//
|
|
// Synopsis:
|
|
// Find the next token with space as the deliminator
|
|
//----------------------------------------------------------------------------
|
|
LPWSTR GetNextRegToken(LPWSTR pwsz, LPWSTR pwszPreToken, BOOL *pfEnd)
|
|
{
|
|
LPWSTR pwszStart=NULL;
|
|
LPWSTR pwszSearch=NULL;
|
|
|
|
if(NULL == pwsz)
|
|
return NULL;
|
|
|
|
if(TRUE == (*pfEnd))
|
|
return NULL;
|
|
|
|
pwszStart=pwsz;
|
|
|
|
if(pwszPreToken)
|
|
pwszStart=pwszPreToken + wcslen(pwszPreToken) + 1;
|
|
|
|
//skip the spaces
|
|
while((*pwszStart)==L' ')
|
|
pwszStart++;
|
|
|
|
//check for NULL
|
|
if(*pwszStart==L'\0')
|
|
return NULL;
|
|
|
|
pwszSearch=pwszStart;
|
|
|
|
while(((*pwszSearch) != L'\0') && ((*pwszSearch) !=L' ') )
|
|
pwszSearch++;
|
|
|
|
if(*pwszSearch == L'\0')
|
|
{
|
|
*pfEnd=TRUE;
|
|
return pwszStart;
|
|
}
|
|
|
|
*pwszSearch=L'\0';
|
|
|
|
*pfEnd=FALSE;
|
|
|
|
return pwszStart;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: InitRegistryValue
|
|
//
|
|
// Synopsis: This function replace SetReg.exe. The expected
|
|
// command line would be: 1 TRUE 3 FALSE 9 TRUE 4 FALSE ....
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT InitRegistryValue(LPWSTR pwszCommand)
|
|
{
|
|
HRESULT hr= E_FAIL;
|
|
DWORD SoftPubFlags[] =
|
|
{
|
|
WTPF_TRUSTTEST | WTPF_TESTCANBEVALID,
|
|
WTPF_IGNOREEXPIRATION,
|
|
WTPF_IGNOREREVOKATION,
|
|
WTPF_OFFLINEOK_IND,
|
|
WTPF_OFFLINEOK_COM,
|
|
WTPF_OFFLINEOKNBU_IND,
|
|
WTPF_OFFLINEOKNBU_COM,
|
|
WTPF_VERIFY_V1_OFF,
|
|
WTPF_IGNOREREVOCATIONONTS,
|
|
WTPF_ALLOWONLYPERTRUST
|
|
};
|
|
|
|
|
|
LPWSTR pwszNextToken=NULL;
|
|
int iIndex=-1;
|
|
BOOL fOn=FALSE;
|
|
int cFlags=sizeof(SoftPubFlags)/sizeof(SoftPubFlags[0]);
|
|
DWORD cParam=0;
|
|
LPWSTR pwszCopy=NULL;
|
|
BOOL fPassThrough=FALSE;
|
|
|
|
//make a copy of the command line since we will change it
|
|
pwszCopy=(LPWSTR)LocalAlloc(LPTR, (1+wcslen(pwszCommand)) * sizeof(WCHAR));
|
|
|
|
if(NULL== pwszCopy)
|
|
goto MemoryErr;
|
|
|
|
wcscpy(pwszCopy, pwszCommand);
|
|
|
|
while(pwszNextToken=GetNextRegToken(pwszCopy, pwszNextToken, &fPassThrough))
|
|
{
|
|
|
|
if(-1 == iIndex)
|
|
{
|
|
iIndex=_wtoi(pwszNextToken);
|
|
|
|
if((iIndex <= 0) || (iIndex > cFlags))
|
|
goto InvalidArgErr;
|
|
|
|
cParam++;
|
|
}
|
|
else
|
|
{
|
|
|
|
if(0 == _wcsicmp(pwszNextToken, L"true"))
|
|
fOn=TRUE;
|
|
else
|
|
{
|
|
if(0 == _wcsicmp(pwszNextToken, L"false"))
|
|
fOn=FALSE;
|
|
else
|
|
goto InvalidArgErr;
|
|
}
|
|
|
|
cParam++;
|
|
|
|
//set the registry value
|
|
if(!SetSoftPubKey(SoftPubFlags[iIndex-1],
|
|
fOn))
|
|
{
|
|
hr=INITPKI_HRESULT_FROM_WIN32(GetLastError());
|
|
goto SetKeyErr;
|
|
}
|
|
|
|
//reset the value for dwIndex
|
|
iIndex=-1;
|
|
}
|
|
}
|
|
|
|
//we have to have even number of parameters
|
|
if( (0 != (cParam %2)) || (0 == cParam))
|
|
goto InvalidArgErr;
|
|
|
|
hr=S_OK;
|
|
|
|
CommonReturn:
|
|
|
|
if(pwszCopy)
|
|
LocalFree((HLOCAL)pwszCopy);
|
|
|
|
return hr;
|
|
|
|
ErrorReturn:
|
|
|
|
goto CommonReturn;
|
|
|
|
SET_ERROR(InvalidArgErr, E_INVALIDARG);
|
|
SET_ERROR_VAR(SetKeyErr, hr);
|
|
SET_ERROR(MemoryErr, E_OUTOFMEMORY);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: DllMain
|
|
//
|
|
// Synopsis:
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
BOOL WINAPI DllMain(HMODULE hInstDLL, DWORD fdwReason, LPVOID lpvReserved)
|
|
{
|
|
switch ( fdwReason )
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
hModule = hInstDLL;
|
|
break;
|
|
}
|
|
|
|
return( TRUE );
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: DllInstall
|
|
//
|
|
// Synopsis: dll installation entry point
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STDAPI DllInstall (BOOL fRegister, LPCSTR pszCommand)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HRESULT hr2;
|
|
LPWSTR pwszCommand=NULL;
|
|
|
|
if ( fRegister == FALSE )
|
|
{
|
|
return( E_NOTIMPL );
|
|
}
|
|
|
|
switch ( *pszCommand )
|
|
{
|
|
//letter S stands for setreg input parameters
|
|
//the command line should look like following:
|
|
//S 1 TRUE 2 FALSE 3 FALSE ...
|
|
//pszCommand is ACTUALLY LPWSTR for BOTH
|
|
//NT5, NT4 and Win95.
|
|
case 'S':
|
|
case 's':
|
|
pwszCommand=(LPWSTR)pszCommand;
|
|
|
|
if(wcslen(pwszCommand) <= 2)
|
|
{
|
|
hr=E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
hr=InitRegistryValue((LPWSTR)(&(pwszCommand[1])));
|
|
}
|
|
|
|
break;
|
|
case 'M':
|
|
case 'm':
|
|
MoveCertificates( TRUE );
|
|
PurgeExpiringCertsFromStores();
|
|
break;
|
|
|
|
case 'U':
|
|
case 'u':
|
|
_AdjustPolicyFlags(psPolicySettings);
|
|
|
|
PurgeExpiringCertsFromStores();
|
|
|
|
// Ensure we have a registry entry for the Group Policy
|
|
// SystemCertificates. On NT 4.0 or Win98, we emulate NT 5.0 GPT
|
|
// notification by doing a RegNotifyChangeKeyValue on this
|
|
// registry key.
|
|
CreateKey(HKEY_CURRENT_USER, GROUP_POLICY_STORE_REGPATH);
|
|
|
|
// Before adding to CurrentUser, will check if the root or CA
|
|
// already exists in LocalMachine. If it exists in
|
|
// LocalMachine and also exists in CurrentUser, will delete it
|
|
// from CurrentUser instead of adding.
|
|
hr = AddCurrentUserRootCertificates();
|
|
hr2 = AddCurrentUserCACertificates();
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
hr2 = AddCurrentUserDisallowedCertificates();
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
|
|
// Protect the CurrentUser roots and purge any existing
|
|
// protected CurrentUser roots also in LocalMachine
|
|
//
|
|
// Note, once the roots are protected, all subsequent adds are
|
|
// done by a special service executing with System privileges.
|
|
// This special service does secure attention sequence (SAS) UI
|
|
// before doing the add.
|
|
//
|
|
// Note, subsequent purges are exempt from UI. ie, this function
|
|
// doesn't do any SAS UI.
|
|
I_CertProtectFunction(
|
|
CERT_PROT_PURGE_LM_ROOTS_FUNC_ID,
|
|
0, // dwFlags
|
|
NULL, // pwszIn
|
|
NULL, // pbIn
|
|
0, // cbIn
|
|
NULL, // ppbOut
|
|
NULL // pcbOut
|
|
);
|
|
|
|
UpdateMyKeyIdentifiers(CERT_SYSTEM_STORE_CURRENT_USER);
|
|
break;
|
|
|
|
case 'B':
|
|
case 'b':
|
|
case 'R':
|
|
case 'r':
|
|
case 'A':
|
|
case 'a':
|
|
// Initialize HKLM registry locations used by PKI to
|
|
// only give Everyone KEY_READ access. Also gives the
|
|
// IEDirtyFlags registry key KEY_SET_VALUE access for
|
|
// Everyone.
|
|
InitializeHKLMAcls();
|
|
|
|
// Ensure we have a registry entry for the IEDirtyFlags
|
|
// This key should have already been created
|
|
// by InitializeHKLMAcls() for NT. Ensure its also there
|
|
// for Win95 and Win98
|
|
CreateKey(HKEY_LOCAL_MACHINE, CERT_IE_DIRTY_FLAGS_REGPATH);
|
|
|
|
MoveCertificates(TRUE);
|
|
PurgeExpiringCertsFromStores();
|
|
|
|
// Ensure we have a registry entry for the Group Policy
|
|
// SystemCertificates. On NT 4.0 or Win98, we emulate NT 5.0 GPT
|
|
// notification by doing a RegNotifyChangeKeyValue on this
|
|
// registry key.
|
|
CreateKey(HKEY_LOCAL_MACHINE, GROUP_POLICY_STORE_REGPATH);
|
|
CreateKey(HKEY_CURRENT_USER, GROUP_POLICY_STORE_REGPATH);
|
|
|
|
// Ensure we have existing predefined stores for the LocalMachine
|
|
// Enterprise system stores. These stores are periodically updated
|
|
// from the DS by a system service. RegNotifyChangeKeyValue is
|
|
// used to signal clients about Enterprise store changes.
|
|
RegisterEnterpriseStores();
|
|
|
|
// Our goal is to get the roots and CAs into LocalMachine.
|
|
// Note previously, they were only copied to CurrentUser.
|
|
AddLocalMachineRootCertificates();
|
|
AddLocalMachineCACertificates();
|
|
AddLocalMachineDisallowedCertificates();
|
|
|
|
// If the above adds to LocalMachine failed, then, add
|
|
// to CurrentUser.
|
|
//
|
|
// Before adding to CurrentUser, will check if the root or CA
|
|
// already exists in LocalMachine. If it exists in
|
|
// LocalMachine and also exists in CurrentUser, will delete it
|
|
// from CurrentUser instead of adding.
|
|
hr = AddCurrentUserRootCertificates();
|
|
hr2 = AddCurrentUserCACertificates();
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
hr2 = AddCurrentUserDisallowedCertificates();
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
|
|
// Protect the CurrentUser roots and purge any existing
|
|
// protected CurrentUser roots also in LocalMachine
|
|
//
|
|
// Note, once the roots are protected, all subsequent adds are
|
|
// done by a special service executing with System privileges.
|
|
// This special service does secure attention sequence (SAS) UI
|
|
// before doing the add.
|
|
//
|
|
// Note, subsequent purges are exempt from UI. ie, this function
|
|
// doesn't do any SAS UI.
|
|
I_CertProtectFunction(
|
|
CERT_PROT_PURGE_LM_ROOTS_FUNC_ID,
|
|
0, // dwFlags
|
|
NULL, // pwszIn
|
|
NULL, // pbIn
|
|
0, // cbIn
|
|
NULL, // ppbOut
|
|
NULL // pcbOut
|
|
);
|
|
|
|
UpdateMyKeyIdentifiers(CERT_SYSTEM_STORE_CURRENT_USER);
|
|
UpdateMyKeyIdentifiers(CERT_SYSTEM_STORE_LOCAL_MACHINE);
|
|
|
|
CleanupRegistry();
|
|
hr2 = RegisterCryptoDlls(TRUE);
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
|
|
if (!I_CryptCatAdminMigrateToNewCatDB())
|
|
{
|
|
hr2 = HRESULT_FROM_WIN32(GetLastError());
|
|
}
|
|
else
|
|
{
|
|
hr2 = ERROR_SUCCESS;
|
|
}
|
|
if (hr == ERROR_SUCCESS)
|
|
hr = hr2;
|
|
|
|
break;
|
|
|
|
case 'C':
|
|
case 'c':
|
|
// Win9x migration post setup cleanup. Remove any migrated
|
|
// roots that exist in the AuthRoot store.
|
|
AddLocalMachineRootCertificates();
|
|
AddCurrentUserRootCertificates();
|
|
break;
|
|
|
|
case 'Z':
|
|
case 'z':
|
|
|
|
//
|
|
// This is for componentization install
|
|
//
|
|
|
|
pwszCommand=(LPWSTR)pszCommand;
|
|
|
|
if (_wcsicmp(pwszCommand, L"z CoreCertificateServices") == 0)
|
|
{
|
|
InitializeHKLMAcls();
|
|
AddLocalMachineRootCertificates();
|
|
AddLocalMachineCACertificates();
|
|
AddLocalMachineDisallowedCertificates();
|
|
|
|
if (!_LoadAndRegister("wintrust.dll", FALSE) ||
|
|
!_LoadAndRegister("mssign32.dll", FALSE) ||
|
|
!_LoadAndRegister("xenroll.dll", FALSE) ||
|
|
!_AdjustPolicyFlags(psPolicySettings))
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
RegisterWinlogonExtension("crypt32chain", "crypt32.dll",
|
|
"ChainWlxLogoffEvent");
|
|
RegisterWinlogonExtension("cryptnet", "cryptnet.dll",
|
|
"CryptnetWlxLogoffEvent");
|
|
|
|
RegisterCrypt32EventSource();
|
|
}
|
|
else if (_wcsicmp(pwszCommand, L"z CertificateUIServices") == 0)
|
|
{
|
|
if (!_LoadAndRegister("cryptui.dll", FALSE))
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
}
|
|
else if (_wcsicmp(pwszCommand, L"z CryptographicNetworkServices") == 0)
|
|
{
|
|
if (!_LoadAndRegister("cryptnet.dll", FALSE))
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
}
|
|
else if (_wcsicmp(pwszCommand, L"z CertificateUIExtensions") == 0)
|
|
{
|
|
if (!_LoadAndRegister("cryptext.dll", FALSE))
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = E_INVALIDARG;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
hr = E_INVALIDARG;
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
STDAPI DllRegisterServer(void)
|
|
{
|
|
return(DllInstall(TRUE, "A"));
|
|
}
|
|
|
|
STDAPI DllUnregisterServer(void)
|
|
{
|
|
return(UnregisterCryptoDlls());
|
|
}
|
|
|
|
BOOL WINAPI InitializePKI(void)
|
|
{
|
|
if (RegisterCryptoDlls(TRUE) != S_OK)
|
|
{
|
|
return(FALSE);
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|