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.
 
 
 
 
 
 

699 lines
23 KiB

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1996 - 1999
//
// File: certtrst.cpp
//
// Contents: Microsoft Internet Security Provider
//
// Functions: WintrustCertificateTrust
//
// *** local functions ***
// _WalkChain
// _IsLifetimeSigningCert
//
// History: 07-Jun-1997 pberkman created
//
//--------------------------------------------------------------------------
#include "global.hxx"
//
// support for MS test roots!!!!
//
static BYTE rgbTestRoot[] =
{
# include "certs\\mstest1.h"
};
static BYTE rgbTestRootCorrected[] =
{
# include "certs\\mstest2.h"
};
static BYTE rgbTestRootBeta1[] =
{
# include "certs\\mstestb1.h"
};
#define NTESTROOTS 3
static CERT_PUBLIC_KEY_INFO rgTestRootPublicKeyInfo[NTESTROOTS] =
{
{szOID_RSA_RSA, 0, NULL, sizeof(rgbTestRoot), rgbTestRoot, 0},
{szOID_RSA_RSA, 0, NULL, sizeof(rgbTestRootCorrected), rgbTestRootCorrected, 0},
{szOID_RSA_RSA, 0, NULL, sizeof(rgbTestRootBeta1), rgbTestRootBeta1, 0}
};
BOOL _WalkChain(
CRYPT_PROVIDER_DATA *pProvData,
DWORD idxSigner,
DWORD *pdwError,
BOOL fCounterSigner,
DWORD idxCounterSigner,
BOOL fTimeStamped,
BOOL *pfLifetimeSigning // IN OUT, only accessed for fTimeStamped
);
BOOL WINAPI _IsLifetimeSigningCert(
PCCERT_CONTEXT pCertContext
);
HRESULT WINAPI WintrustCertificateTrust(CRYPT_PROVIDER_DATA *pProvData)
{
if ((_ISINSTRUCT(CRYPT_PROVIDER_DATA, pProvData->cbStruct, fRecallWithState)) &&
(pProvData->fRecallWithState == TRUE))
{
return(S_OK);
}
DWORD dwError;
dwError = S_OK;
if (pProvData->csSigners < 1)
{
pProvData->padwTrustStepErrors[TRUSTERROR_STEP_FINAL_CERTPROV] = TRUST_E_NOSIGNATURE;
return(S_FALSE);
}
//
// loop through all signers
//
for (int i = 0; i < (int)pProvData->csSigners; i++)
{
BOOL fTimeStamped = FALSE;
BOOL fLifetimeSigning = FALSE;
if (pProvData->pasSigners[i].csCertChain < 1)
{
pProvData->pasSigners[i].dwError = TRUST_E_NO_SIGNER_CERT;
pProvData->padwTrustStepErrors[TRUSTERROR_STEP_FINAL_CERTPROV] = TRUST_E_NO_SIGNER_CERT;
continue;
}
// Check if timestamped.
if (0 < pProvData->pasSigners[i].csCounterSigners &&
(pProvData->pasSigners[i].pasCounterSigners[0].dwSignerType &
SGNR_TYPE_TIMESTAMP))
{
fTimeStamped = TRUE;
// See if LifeTime Signing has been enabled
if (pProvData->dwProvFlags & WTD_LIFETIME_SIGNING_FLAG)
{
fLifetimeSigning = TRUE;
}
else
{
// Check if the signer certificate has the LIFETIME_SIGNING
// EKU.
fLifetimeSigning = _IsLifetimeSigningCert(
pProvData->pasSigners[i].pasCertChain[0].pCert);
}
}
_WalkChain(pProvData, i, &dwError, FALSE, 0,
fTimeStamped, &fLifetimeSigning);
for (int i2 = 0; i2 < (int)pProvData->pasSigners[i].csCounterSigners; i2++)
{
if (pProvData->pasSigners[i].pasCounterSigners[i2].csCertChain < 1)
{
pProvData->pasSigners[i].pasCounterSigners[i2].dwError = TRUST_E_NO_SIGNER_CERT;
pProvData->padwTrustStepErrors[TRUSTERROR_STEP_FINAL_CERTPROV] = TRUST_E_COUNTER_SIGNER;
dwError = S_FALSE;
continue;
}
// If lifetime signing has been enabled, use current time instead
// of timestamp time.
if (fLifetimeSigning)
{
memcpy(&pProvData->pasSigners[i].pasCounterSigners[i2].sftVerifyAsOf,
&pProvData->sftSystemTime, sizeof(FILETIME));
}
_WalkChain(pProvData, i, &dwError, TRUE, i2, FALSE, NULL);
}
}
return(dwError);
}
HCERTCHAINENGINE GetChainEngine(
IN CRYPT_PROVIDER_DATA *pProvData
)
{
CERT_CHAIN_ENGINE_CONFIG Config;
HCERTSTORE hStore = NULL;
HCERTCHAINENGINE hChainEngine = NULL;
if (NULL == pProvData->pWintrustData ||
pProvData->pWintrustData->dwUnionChoice != WTD_CHOICE_CERT ||
!_ISINSTRUCT(WINTRUST_CERT_INFO,
pProvData->pWintrustData->pCert->cbStruct, dwFlags) ||
0 == (pProvData->pWintrustData->pCert->dwFlags &
(WTCI_DONT_OPEN_STORES | WTCI_OPEN_ONLY_ROOT)))
return NULL;
memset(&Config, 0, sizeof(Config));
Config.cbSize = sizeof(Config);
if (NULL == (hStore = CertOpenStore(
CERT_STORE_PROV_MEMORY,
0, // dwEncodingType
0, // hCryptProv
0, // dwFlags
NULL // pvPara
)))
goto OpenMemoryStoreError;
if (pProvData->pWintrustData->pCert->dwFlags & WTCI_DONT_OPEN_STORES)
Config.hRestrictedRoot = hStore;
Config.hRestrictedTrust = hStore;
Config.hRestrictedOther = hStore;
if (!CertCreateCertificateChainEngine(
&Config,
&hChainEngine
))
goto CreateChainEngineError;
CommonReturn:
CertCloseStore(hStore, 0);
return hChainEngine;
ErrorReturn:
hChainEngine = NULL;
goto CommonReturn;
TRACE_ERROR_EX(DBG_SS, OpenMemoryStoreError)
TRACE_ERROR_EX(DBG_SS, CreateChainEngineError)
}
HCERTSTORE GetChainAdditionalStore(
IN CRYPT_PROVIDER_DATA *pProvData
)
{
HCERTSTORE hStore = NULL;
if (0 == pProvData->chStores)
return NULL;
if (1 < pProvData->chStores) {
if (hStore = CertOpenStore(
CERT_STORE_PROV_COLLECTION,
0, // dwEncodingType
0, // hCryptProv
0, // dwFlags
NULL // pvPara
)) {
DWORD i;
for (i = 0; i < pProvData->chStores; i++)
CertAddStoreToCollection(
hStore,
pProvData->pahStores[i],
CERT_PHYSICAL_STORE_ADD_ENABLE_FLAG,
0 // dwPriority
);
}
} else
hStore = CertDuplicateStore(pProvData->pahStores[0]);
#if 0
CertSaveStore(
hStore,
PKCS_7_ASN_ENCODING | X509_ASN_ENCODING,
CERT_STORE_SAVE_AS_STORE,
CERT_STORE_SAVE_TO_FILENAME_A,
(void *) "C:\\temp\\wintrust.sto",
0 // dwFlags
);
#endif
return hStore;
}
BOOL UpdateCertProvChain(
IN OUT PCRYPT_PROVIDER_DATA pProvData,
IN DWORD idxSigner,
OUT DWORD *pdwError,
IN BOOL fCounterSigner,
IN DWORD idxCounterSigner,
IN PCRYPT_PROVIDER_SGNR pSgnr,
IN PCCERT_CHAIN_CONTEXT pChainContext
)
{
BOOL fTestCert = FALSE;
DWORD dwSgnrError = 0;
DWORD i;
// The chain better have at least the certificate we passed in
assert(0 < pChainContext->cChain &&
0 < pChainContext->rgpChain[0]->cElement);
for (i = 0; i < pChainContext->cChain; i++) {
DWORD j;
PCERT_SIMPLE_CHAIN pChain = pChainContext->rgpChain[i];
for (j = 0; j < pChain->cElement; j++) {
PCERT_CHAIN_ELEMENT pEle = pChain->rgpElement[j];
DWORD dwEleError = pEle->TrustStatus.dwErrorStatus;
DWORD dwEleInfo = pEle->TrustStatus.dwInfoStatus;
PCRYPT_PROVIDER_CERT pProvCert;
if (0 != i || 0 != j) {
if (!(pProvData->psPfns->pfnAddCert2Chain(
pProvData, idxSigner, fCounterSigner,
idxCounterSigner, pEle->pCertContext)))
{
pProvData->dwError = GetLastError();
dwSgnrError = TRUST_E_SYSTEM_ERROR;
goto CommonReturn;
}
}
//
// else
// Signer cert has already been added
pProvCert = &pSgnr->pasCertChain[pSgnr->csCertChain -1];
//DSIE: 12-Oct-2000 added to get pChainElement.
pProvCert->pChainElement = pEle;
pProvCert->fSelfSigned =
0 != (dwEleInfo & CERT_TRUST_IS_SELF_SIGNED) &&
0 == (dwEleError & CERT_TRUST_IS_NOT_SIGNATURE_VALID);
pProvCert->fTrustedRoot =
pProvCert->fSelfSigned &&
i == pChainContext->cChain - 1 &&
j == pChain->cElement - 1 &&
0 == (dwEleError & CERT_TRUST_IS_UNTRUSTED_ROOT);
if (pProvCert->fSelfSigned) {
// Check if one of the "test" roots
DWORD k;
for (k = 0; k < NTESTROOTS; k++)
{
if (CertComparePublicKeyInfo(
pProvData->dwEncoding,
&pProvCert->pCert->pCertInfo->SubjectPublicKeyInfo,
&rgTestRootPublicKeyInfo[k]))
{
pProvCert->fTestCert = TRUE;
fTestCert = TRUE;
if (pProvData->dwRegPolicySettings & WTPF_TRUSTTEST)
pProvCert->fTrustedRoot = TRUE;
break;
}
}
}
// First Element in all but the first simple chain
pProvCert->fTrustListSignerCert = (0 < i && 0 == j);
pProvCert->fIsCyclic = (0 != (dwEleError & CERT_TRUST_IS_CYCLIC));
// Map to IE4Trust confidence
if (0 == (dwEleError & CERT_TRUST_IS_NOT_SIGNATURE_VALID))
pProvCert->dwConfidence |= CERT_CONFIDENCE_SIG;
if (0 == (dwEleError & CERT_TRUST_IS_NOT_TIME_VALID))
pProvCert->dwConfidence |= CERT_CONFIDENCE_TIME;
// On Sep 10, 1998 Trevor/Brian wanted time nesting checks to
// be disabled
// if (0 == (dwEleError & CERT_TRUST_IS_NOT_TIME_NESTED))
pProvCert->dwConfidence |= CERT_CONFIDENCE_TIMENEST;
if (0 != (dwEleInfo & CERT_TRUST_HAS_EXACT_MATCH_ISSUER))
pProvCert->dwConfidence |= CERT_CONFIDENCE_AUTHIDEXT;
if (0 == (dwEleError & CERT_TRUST_IS_NOT_SIGNATURE_VALID) &&
0 != (dwEleInfo & CERT_TRUST_HAS_EXACT_MATCH_ISSUER))
pProvCert->dwConfidence |= CERT_CONFIDENCE_HYGIENE;
if (pEle->pRevocationInfo) {
pProvCert->dwRevokedReason =
pEle->pRevocationInfo->dwRevocationResult;
}
// Update any signature or revocations errors
if (dwEleError & CERT_TRUST_IS_NOT_SIGNATURE_VALID) {
pProvCert->dwError = TRUST_E_CERT_SIGNATURE;
assert(pChainContext->TrustStatus.dwErrorStatus &
CERT_TRUST_IS_NOT_SIGNATURE_VALID);
dwSgnrError = TRUST_E_CERT_SIGNATURE;
} else if (dwEleError & CERT_TRUST_IS_REVOKED) {
pProvCert->dwError = CERT_E_REVOKED;
assert(pChainContext->TrustStatus.dwErrorStatus &
CERT_TRUST_IS_REVOKED);
if (0 == dwSgnrError ||
CERT_E_REVOCATION_FAILURE == dwSgnrError)
dwSgnrError = CERT_E_REVOKED;
} else if (dwEleError & CERT_TRUST_IS_OFFLINE_REVOCATION) {
// Ignore NO_CHECK errors
if (pProvData->pWintrustData->fdwRevocationChecks ==
WTD_REVOKE_WHOLECHAIN) {
pProvCert->dwError = CERT_E_REVOCATION_FAILURE;
assert(pChainContext->TrustStatus.dwErrorStatus &
CERT_TRUST_REVOCATION_STATUS_UNKNOWN);
if (0 == dwSgnrError)
dwSgnrError = CERT_E_REVOCATION_FAILURE;
}
}
// If last element in simple chain, check if it was in a
// CTL and update CryptProvData if it was.
if (j == pChain->cElement - 1 && pChain->pTrustListInfo &&
pChain->pTrustListInfo->pCtlContext) {
DWORD dwChainError = pChain->TrustStatus.dwErrorStatus;
// Note, don't need to AddRef since we already hold an
// AddRef on the ChainContext.
pProvCert->pCtlContext = pChain->pTrustListInfo->pCtlContext;
if (dwChainError & CERT_TRUST_CTL_IS_NOT_SIGNATURE_VALID) {
pProvCert->dwCtlError = TRUST_E_CERT_SIGNATURE;
dwSgnrError = TRUST_E_CERT_SIGNATURE;
} else if (dwChainError & CERT_TRUST_CTL_IS_NOT_TIME_VALID) {
if (0 == (pProvData->dwRegPolicySettings &
WTPF_IGNOREEXPIRATION))
pProvCert->dwCtlError = CERT_E_EXPIRED;
} else if (dwChainError &
CERT_TRUST_CTL_IS_NOT_VALID_FOR_USAGE) {
pProvCert->dwCtlError = CERT_E_WRONG_USAGE;
}
}
if (pProvData->psPfns->pfnCertCheckPolicy) {
if (! (*pProvData->psPfns->pfnCertCheckPolicy)(
pProvData, idxSigner, fCounterSigner, idxCounterSigner))
goto CommonReturn;
}
}
}
CommonReturn:
if (fTestCert) {
if (CERT_TRUST_IS_REVOKED == dwSgnrError ||
CERT_E_REVOCATION_FAILURE == dwSgnrError) {
// No revocation errors for "test" roots
dwSgnrError = 0;
// Loop through certs and remove any revocation error status
for (i = 0; i < pSgnr->csCertChain; i++) {
PCRYPT_PROVIDER_CERT pProvCert = &pSgnr->pasCertChain[i];
pProvCert->dwError = 0;
pProvCert->dwRevokedReason = 0;
}
}
}
if (CERT_E_REVOCATION_FAILURE == dwSgnrError &&
pProvData->pWintrustData->fdwRevocationChecks !=
WTD_REVOKE_WHOLECHAIN)
// Will check during Final Policy
dwSgnrError = 0;
if (dwSgnrError) {
pSgnr->dwError = dwSgnrError;
pProvData->padwTrustStepErrors[TRUSTERROR_STEP_FINAL_CERTPROV] =
dwSgnrError;
*pdwError = S_FALSE;
return FALSE;
} else
return TRUE;
}
BOOL _WalkChain(
CRYPT_PROVIDER_DATA *pProvData,
DWORD idxSigner,
DWORD *pdwError,
BOOL fCounterSigner,
DWORD idxCounterSigner,
BOOL fTimeStamped,
BOOL *pfLifetimeSigning // IN OUT, only accessed for fTimeStamped
)
{
BOOL fResult;
DWORD dwCreateChainFlags;
DWORD dwSgnrError = 0;
CRYPT_PROVIDER_SGNR *pSgnr; // not allocated
PCCERT_CONTEXT pCertContext; // not allocated
CERT_CHAIN_PARA ChainPara;
HCERTCHAINENGINE hChainEngine = NULL;
HCERTSTORE hAdditionalStore = NULL;
PCCERT_CHAIN_CONTEXT pChainContext = NULL;
LPSTR pszUsage = NULL;
if (fCounterSigner)
pSgnr = &pProvData->pasSigners[idxSigner].pasCounterSigners[
idxCounterSigner];
else
pSgnr = &pProvData->pasSigners[idxSigner];
assert(pSgnr);
//
// at this stage, the last cert in the chain "should be" the signers cert.
// eg: there should only be one cert in the chain from the Signature
// Provider
//
if (1 != pSgnr->csCertChain ||
(NULL == (pCertContext =
pSgnr->pasCertChain[pSgnr->csCertChain - 1].pCert))) {
dwSgnrError = TRUST_E_NO_SIGNER_CERT;
goto NoSignerCertError;
}
memset(&ChainPara, 0, sizeof(ChainPara));
ChainPara.cbSize = sizeof(ChainPara);
if (fCounterSigner && SGNR_TYPE_TIMESTAMP == pSgnr->dwSignerType) {
pszUsage = szOID_PKIX_KP_TIMESTAMP_SIGNING;
} else if(NULL != pProvData->pRequestUsage) {
ChainPara.RequestedUsage = *pProvData->pRequestUsage;
} else {
pszUsage = pProvData->pszUsageOID;
}
if ( (0 == (pProvData->dwProvFlags & WTD_NO_POLICY_USAGE_FLAG)) && pszUsage) {
ChainPara.RequestedUsage.dwType = USAGE_MATCH_TYPE_AND;
ChainPara.RequestedUsage.Usage.cUsageIdentifier = 1;
ChainPara.RequestedUsage.Usage.rgpszUsageIdentifier = &pszUsage;
}
hChainEngine = GetChainEngine(pProvData);
hAdditionalStore = GetChainAdditionalStore(pProvData);
dwCreateChainFlags = 0;
if (pProvData->dwProvFlags & CPD_REVOCATION_CHECK_NONE) {
;
} else if (pProvData->dwProvFlags & CPD_REVOCATION_CHECK_END_CERT) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_END_CERT;
} else if (pProvData->dwProvFlags & CPD_REVOCATION_CHECK_CHAIN) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN;
} else if (pProvData->dwProvFlags &
CPD_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT;
} else if (pProvData->dwProvFlags & WTD_REVOCATION_CHECK_NONE) {
;
} else if (pProvData->dwProvFlags & WTD_REVOCATION_CHECK_END_CERT) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_END_CERT;
} else if (pProvData->dwProvFlags & WTD_REVOCATION_CHECK_CHAIN) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN;
} else if (pProvData->dwProvFlags &
WTD_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT;
} else if (pProvData->pWintrustData->fdwRevocationChecks ==
WTD_REVOKE_WHOLECHAIN) {
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN;
} else if (fCounterSigner && SGNR_TYPE_TIMESTAMP == pSgnr->dwSignerType) {
if (0 == (pProvData->dwRegPolicySettings & WTPF_IGNOREREVOCATIONONTS))
// On 4-12-01 changed from END_CERT to EXCLUDE_ROOT
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT;
} else if (0 == (pProvData->dwRegPolicySettings & WTPF_IGNOREREVOKATION))
// On 4-12-01 changed from END_CERT to EXCLUDE_ROOT
dwCreateChainFlags = CERT_CHAIN_REVOCATION_CHECK_CHAIN_EXCLUDE_ROOT;
if (!(pProvData->pWintrustData->dwUnionChoice == WTD_CHOICE_CERT ||
pProvData->pWintrustData->dwUnionChoice == WTD_CHOICE_SIGNER))
// Certificate was obtained from either the message store, or one
// of our known stores
dwCreateChainFlags |= CERT_CHAIN_CACHE_END_CERT;
// else
// Can't implicitly cache a context passed to us
#if 0
{
HCERTSTORE hDebugStore;
if (hDebugStore = CertOpenSystemStoreA(0, "wintrust")) {
CertAddCertificateContextToStore(
hDebugStore,
pCertContext,
CERT_STORE_ADD_ALWAYS,
NULL
);
CertCloseStore(hDebugStore, 0);
}
}
#endif
FILETIME fileTime;
memset(&fileTime, 0, sizeof(fileTime));
if (fTimeStamped && *pfLifetimeSigning)
dwCreateChainFlags |= CERT_CHAIN_TIMESTAMP_TIME;
if (!CertGetCertificateChain (
hChainEngine,
pCertContext,
(memcmp(&pSgnr->sftVerifyAsOf, &fileTime, sizeof(fileTime)) == 0) ?
NULL : &pSgnr->sftVerifyAsOf,
hAdditionalStore,
&ChainPara,
dwCreateChainFlags,
NULL, // pvReserved,
&pChainContext
)) {
pProvData->dwError = GetLastError();
dwSgnrError = TRUST_E_SYSTEM_ERROR;
goto GetChainError;
}
if (fTimeStamped && !*pfLifetimeSigning) {
// See if resultant application policy has the LIFETIME_SIGNING OID
PCERT_ENHKEY_USAGE pAppUsage =
pChainContext->rgpChain[0]->rgpElement[0]->pApplicationUsage;
if (pAppUsage) {
DWORD i;
for (i = 0; i < pAppUsage->cUsageIdentifier; i++) {
if (0 == strcmp(pAppUsage->rgpszUsageIdentifier[i],
szOID_KP_LIFETIME_SIGNING)) {
*pfLifetimeSigning = TRUE;
break;
}
}
}
if (*pfLifetimeSigning) {
CertFreeCertificateChain(pChainContext);
pChainContext = NULL;
dwCreateChainFlags |= CERT_CHAIN_TIMESTAMP_TIME;
if (!CertGetCertificateChain (
hChainEngine,
pCertContext,
(memcmp(&pSgnr->sftVerifyAsOf, &fileTime, sizeof(fileTime)) == 0) ?
NULL : &pSgnr->sftVerifyAsOf,
hAdditionalStore,
&ChainPara,
dwCreateChainFlags,
NULL, // pvReserved,
&pChainContext
)) {
pProvData->dwError = GetLastError();
dwSgnrError = TRUST_E_SYSTEM_ERROR;
goto GetChainError;
}
}
}
pSgnr->pChainContext = pChainContext;
if (pProvData->dwProvFlags & WTD_NO_IE4_CHAIN_FLAG) {
pProvData->dwProvFlags |= CPD_USE_NT5_CHAIN_FLAG;
fResult = TRUE;
} else
fResult = UpdateCertProvChain(
pProvData,
idxSigner,
pdwError,
fCounterSigner,
idxCounterSigner,
pSgnr,
pChainContext
);
CommonReturn:
if (hChainEngine)
CertFreeCertificateChainEngine(hChainEngine);
if (hAdditionalStore)
CertCloseStore(hAdditionalStore, 0);
return fResult;
ErrorReturn:
pSgnr->dwError = dwSgnrError;
pProvData->padwTrustStepErrors[TRUSTERROR_STEP_FINAL_CERTPROV] =
dwSgnrError;
*pdwError = S_FALSE;
fResult = FALSE;
goto CommonReturn;
TRACE_ERROR_EX(DBG_SS, NoSignerCertError)
TRACE_ERROR_EX(DBG_SS, GetChainError)
}
BOOL WINAPI _IsLifetimeSigningCert(
PCCERT_CONTEXT pCertContext
)
{
DWORD cbSize;
PCERT_ENHKEY_USAGE pCertEKU;
//
// see if the certificate has the proper enhanced key usage OID
//
cbSize = 0;
CertGetEnhancedKeyUsage(pCertContext,
CERT_FIND_EXT_ONLY_ENHKEY_USAGE_FLAG,
NULL,
&cbSize);
if (cbSize == 0)
{
return(FALSE);
}
if (!(pCertEKU = (PCERT_ENHKEY_USAGE)new BYTE[cbSize]))
{
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
return(FALSE);
}
if (!(CertGetEnhancedKeyUsage(pCertContext,
CERT_FIND_EXT_ONLY_ENHKEY_USAGE_FLAG,
pCertEKU,
&cbSize)))
{
delete pCertEKU;
return(FALSE);
}
for (int i = 0; i < (int)pCertEKU->cUsageIdentifier; i++)
{
if (strcmp(pCertEKU->rgpszUsageIdentifier[i], szOID_KP_LIFETIME_SIGNING) == 0)
{
delete pCertEKU;
return(TRUE);
}
}
delete pCertEKU;
return(FALSE);
}