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.
1012 lines
23 KiB
1012 lines
23 KiB
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
//
|
|
// Copyright (C) Microsoft Corporation, 1997 - 1999
|
|
//
|
|
// File: cscsp.cpp
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#include <pch.cpp>
|
|
|
|
#pragma hdrstop
|
|
|
|
#include <tchar.h>
|
|
#include "initcert.h"
|
|
#include "cscsp.h"
|
|
#include "csdisp.h"
|
|
|
|
#define __dwFILE__ __dwFILE_CERTLIB_CSCSP_CPP__
|
|
|
|
|
|
|
|
TCHAR const g_wszRegKeyCSP[] = wszREGKEYCSP;
|
|
TCHAR const g_wszRegKeyEncryptionCSP[] = wszREGKEYENCRYPTIONCSP;
|
|
TCHAR const g_wszRegProviderType[] = wszREGPROVIDERTYPE;
|
|
TCHAR const g_wszRegProvider[] = wszREGPROVIDER;
|
|
TCHAR const g_wszRegHashAlgorithm[] = wszHASHALGORITHM;
|
|
TCHAR const g_wszRegEncryptionAlgorithm[] = wszENCRYPTIONALGORITHM;
|
|
TCHAR const g_wszRegMachineKeyset[] = wszMACHINEKEYSET;
|
|
TCHAR const g_wszRegKeySize[] = wszREGKEYSIZE;
|
|
|
|
|
|
BOOL
|
|
myCertSrvCryptAcquireContext(
|
|
OUT HCRYPTPROV *phProv,
|
|
IN WCHAR const *pwszContainer,
|
|
IN WCHAR const *pwszProvider,
|
|
IN DWORD dwProvType,
|
|
IN DWORD dwFlags,
|
|
IN BOOL fMachineKeyset)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (fMachineKeyset)
|
|
{
|
|
dwFlags |= CRYPT_MACHINE_KEYSET;
|
|
}
|
|
if (!CryptAcquireContext(
|
|
phProv,
|
|
pwszContainer,
|
|
pwszProvider,
|
|
dwProvType,
|
|
dwFlags))
|
|
{
|
|
hr = myHLastError();
|
|
_PrintErrorStr2(
|
|
hr,
|
|
(CRYPT_MACHINE_KEYSET & dwFlags)?
|
|
"CryptAcquireContext(Machine)" : "CryptAcquireContext(User)",
|
|
pwszProvider,
|
|
(CRYPT_DELETEKEYSET & dwFlags)? hr : S_OK);
|
|
_JumpErrorStr2(
|
|
hr,
|
|
error,
|
|
(CRYPT_MACHINE_KEYSET & dwFlags)?
|
|
"CryptAcquireContext(Machine)" : "CryptAcquireContext(User)",
|
|
pwszContainer,
|
|
(CRYPT_DELETEKEYSET & dwFlags)? hr : S_OK);
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (S_OK != hr)
|
|
{
|
|
SetLastError(hr);
|
|
}
|
|
return(S_OK == hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myGetCertSrvCSP(
|
|
IN BOOL fEncryptionCSP,
|
|
IN WCHAR const *pwszSanitizedCAName,
|
|
OUT DWORD *pdwProvType,
|
|
OUT WCHAR **ppwszProvName,
|
|
OUT ALG_ID *pidAlg,
|
|
OUT BOOL *pfMachineKeyset,
|
|
OPTIONAL OUT DWORD *pdwKeySize)
|
|
{
|
|
HRESULT hr;
|
|
HKEY hCertSrvCSPKey = NULL;
|
|
DWORD dwValueType;
|
|
DWORD dwValueSize;
|
|
WCHAR *pwszPath = NULL;
|
|
WCHAR *pwszProvName = NULL;
|
|
WCHAR const *pwszRegKeyCSP;
|
|
DWORD cwc;
|
|
|
|
if (NULL != ppwszProvName)
|
|
{
|
|
*ppwszProvName = NULL;
|
|
}
|
|
if (NULL != pdwKeySize)
|
|
{
|
|
*pdwKeySize = 0;
|
|
}
|
|
if (NULL == pwszSanitizedCAName ||
|
|
NULL == pdwProvType ||
|
|
NULL == ppwszProvName ||
|
|
NULL == pidAlg ||
|
|
NULL == pfMachineKeyset)
|
|
{
|
|
hr = E_POINTER;
|
|
_JumpError(hr, error, "NULL parm");
|
|
}
|
|
|
|
pwszRegKeyCSP = fEncryptionCSP? g_wszRegKeyEncryptionCSP : g_wszRegKeyCSP;
|
|
cwc = WSZARRAYSIZE(wszREGKEYCONFIGPATH_BS) +
|
|
wcslen(pwszSanitizedCAName) +
|
|
1 +
|
|
wcslen(pwszRegKeyCSP);
|
|
pwszPath = (WCHAR *) LocalAlloc(LMEM_FIXED, (cwc + 1) * sizeof(WCHAR));
|
|
if (NULL == pwszPath)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
|
|
wcscpy(pwszPath, wszREGKEYCONFIGPATH_BS);
|
|
wcscat(pwszPath, pwszSanitizedCAName);
|
|
wcscat(pwszPath, L"\\");
|
|
wcscat(pwszPath, pwszRegKeyCSP);
|
|
CSASSERT(cwc == wcslen(pwszPath));
|
|
|
|
hr = RegOpenKeyEx(
|
|
HKEY_LOCAL_MACHINE,
|
|
pwszPath,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
&hCertSrvCSPKey);
|
|
_JumpIfErrorStr2(hr, error, "RegOpenKeyEx", pwszPath, ERROR_FILE_NOT_FOUND);
|
|
|
|
dwValueSize = 0;
|
|
hr = RegQueryValueEx(
|
|
hCertSrvCSPKey,
|
|
g_wszRegProvider,
|
|
0,
|
|
&dwValueType,
|
|
NULL,
|
|
&dwValueSize);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
if (REG_SZ != dwValueType)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Invalid reg type");
|
|
}
|
|
|
|
pwszProvName = (WCHAR *) LocalAlloc(
|
|
LMEM_FIXED,
|
|
dwValueSize + sizeof(WCHAR));
|
|
if (NULL == pwszProvName)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
hr = RegQueryValueEx(
|
|
hCertSrvCSPKey,
|
|
g_wszRegProvider,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pwszProvName,
|
|
&dwValueSize);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
pwszProvName[dwValueSize / sizeof(WCHAR)] = L'\0';
|
|
|
|
|
|
dwValueSize = sizeof(*pdwProvType);
|
|
hr = RegQueryValueEx(hCertSrvCSPKey,
|
|
g_wszRegProviderType,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pdwProvType,
|
|
&dwValueSize);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
if (REG_DWORD != dwValueType)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Invalid reg type");
|
|
}
|
|
|
|
dwValueSize = sizeof(*pidAlg);
|
|
hr = RegQueryValueEx(
|
|
hCertSrvCSPKey,
|
|
fEncryptionCSP?
|
|
g_wszRegEncryptionAlgorithm :
|
|
g_wszRegHashAlgorithm,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pidAlg,
|
|
&dwValueSize);
|
|
if (S_OK != hr && fEncryptionCSP)
|
|
{
|
|
_PrintErrorStr2(
|
|
hr,
|
|
"RegQueryValueEx",
|
|
g_wszRegEncryptionAlgorithm,
|
|
ERROR_FILE_NOT_FOUND);
|
|
dwValueSize = sizeof(*pidAlg);
|
|
hr = RegQueryValueEx(
|
|
hCertSrvCSPKey,
|
|
g_wszRegHashAlgorithm,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pidAlg,
|
|
&dwValueSize);
|
|
}
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
if (REG_DWORD != dwValueType)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Invalid reg type");
|
|
}
|
|
|
|
dwValueSize = sizeof(*pfMachineKeyset);
|
|
hr = RegQueryValueEx(hCertSrvCSPKey,
|
|
g_wszRegMachineKeyset,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pfMachineKeyset,
|
|
&dwValueSize);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
if (REG_DWORD != dwValueType)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Invalid reg type");
|
|
}
|
|
|
|
if (NULL != pdwKeySize)
|
|
{
|
|
dwValueSize = sizeof(*pdwKeySize);
|
|
hr = RegQueryValueEx(hCertSrvCSPKey,
|
|
g_wszRegKeySize,
|
|
0,
|
|
&dwValueType,
|
|
(BYTE *) pdwKeySize,
|
|
&dwValueSize);
|
|
_JumpIfError(hr, error, "RegQueryValueEx");
|
|
|
|
if (REG_DWORD != dwValueType)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Invalid reg type");
|
|
}
|
|
}
|
|
|
|
*ppwszProvName = pwszProvName;
|
|
pwszProvName = NULL;
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pwszProvName)
|
|
{
|
|
LocalFree(pwszProvName);
|
|
}
|
|
if (NULL != pwszPath)
|
|
{
|
|
LocalFree(pwszPath);
|
|
}
|
|
if (NULL != hCertSrvCSPKey)
|
|
{
|
|
RegCloseKey(hCertSrvCSPKey);
|
|
}
|
|
return(myHError(hr));
|
|
}
|
|
|
|
|
|
//+------------------------------------------------------------------------
|
|
//
|
|
// Function: myGetSigningOID( . . . . )
|
|
//
|
|
// Synopsis: Determine algorithm identifer for cert creation
|
|
// Arguments: csp_provider_type, hash_algorithm_id.
|
|
// Returns: object identifier
|
|
//
|
|
//-------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
myGetSigningOID(
|
|
OPTIONAL IN HCRYPTPROV hProv, // hProv OR pwszProvName & dwProvType
|
|
OPTIONAL IN WCHAR const *pwszProvName,
|
|
OPTIONAL IN DWORD dwProvType,
|
|
IN ALG_ID idHashAlg,
|
|
OUT CHAR **ppszAlgId)
|
|
{
|
|
HRESULT hr;
|
|
|
|
PCCRYPT_OID_INFO pcOIDInfo; //don't free it
|
|
ALG_ID aidAlgKey[] = {idHashAlg, 0};
|
|
HCRYPTPROV hProvT = NULL;
|
|
int i;
|
|
DWORD dwFlags;
|
|
DWORD cbData;
|
|
PROV_ENUMALGS enumalgs;
|
|
BOOL fFoundSignID = FALSE;
|
|
|
|
*ppszAlgId = NULL;
|
|
if (NULL == hProv)
|
|
{
|
|
CSASSERT(NULL != pwszProvName);
|
|
|
|
if (!myCertSrvCryptAcquireContext(
|
|
&hProvT,
|
|
NULL,
|
|
pwszProvName,
|
|
dwProvType,
|
|
CRYPT_VERIFYCONTEXT,
|
|
FALSE))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "myCertSrvCryptAcquireContext", pwszProvName);
|
|
}
|
|
hProv = hProvT;
|
|
}
|
|
|
|
// find public key id
|
|
|
|
dwFlags = CRYPT_FIRST;
|
|
for (i = 0; ; i++)
|
|
{
|
|
cbData = sizeof(enumalgs);
|
|
if (!CryptGetProvParam(
|
|
hProv,
|
|
PP_ENUMALGS,
|
|
(BYTE *) &enumalgs,
|
|
&cbData,
|
|
dwFlags))
|
|
{
|
|
hr = myHLastError();
|
|
if (HRESULT_FROM_WIN32(ERROR_NO_MORE_ITEMS) == hr)
|
|
{
|
|
// out of for loop
|
|
break;
|
|
}
|
|
_JumpError(hr, error, "CryptGetProvParam");
|
|
}
|
|
if (ALG_CLASS_SIGNATURE == GET_ALG_CLASS(enumalgs.aiAlgid))
|
|
{
|
|
fFoundSignID = TRUE;
|
|
aidAlgKey[1] = enumalgs.aiAlgid;
|
|
break;
|
|
}
|
|
dwFlags = 0;
|
|
}
|
|
|
|
if (fFoundSignID)
|
|
{
|
|
pcOIDInfo = CryptFindOIDInfo(
|
|
CRYPT_OID_INFO_SIGN_KEY,
|
|
aidAlgKey,
|
|
CRYPT_SIGN_ALG_OID_GROUP_ID); // only signing
|
|
if (NULL == pcOIDInfo)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "unsupported signing algorithm");
|
|
}
|
|
hr = myDupStringA(pcOIDInfo->pszOID, ppszAlgId);
|
|
_JumpIfError(hr, error, "myDupStringA");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != hProvT)
|
|
{
|
|
CryptReleaseContext(hProvT, 0);
|
|
}
|
|
CSASSERT(S_OK == hr || FAILED(hr));
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myValidateKeyForSigning(
|
|
IN HCRYPTPROV hProv,
|
|
OPTIONAL IN CERT_PUBLIC_KEY_INFO const *pPublicKeyInfo,
|
|
IN ALG_ID algId)
|
|
{
|
|
HRESULT hr;
|
|
BYTE abRandom[64];
|
|
HCRYPTKEY hKey = NULL;
|
|
HCRYPTHASH hHash = NULL;
|
|
BYTE *pbSignature = NULL;
|
|
DWORD cbSignature;
|
|
|
|
// create a supported hash
|
|
if (!CryptCreateHash(hProv, algId, 0, 0, &hHash))
|
|
{
|
|
hr = myHLastError();
|
|
DBGPRINT((DBG_SS_ERROR, "algId = %x\n", algId));
|
|
_JumpError(hr, error, "CryptCreateHash");
|
|
}
|
|
// create some random data
|
|
if (!CryptGenRandom(hProv, ARRAYSIZE(abRandom), abRandom))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGenRandom");
|
|
}
|
|
if (!CryptHashData(hHash, abRandom, ARRAYSIZE(abRandom), 0))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptHashData");
|
|
}
|
|
// sign the hash, get size first
|
|
if (!CryptSignHash(hHash, AT_SIGNATURE, NULL, 0, NULL, &cbSignature))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptSignHash");
|
|
}
|
|
pbSignature = (BYTE *) LocalAlloc(LMEM_FIXED, cbSignature);
|
|
if (NULL == pbSignature)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
if (!CryptSignHash(hHash, AT_SIGNATURE, NULL, 0, pbSignature, &cbSignature))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptSignHash");
|
|
}
|
|
|
|
if (NULL != pPublicKeyInfo)
|
|
{
|
|
// import public key into provider
|
|
|
|
if (!CryptImportPublicKeyInfo(
|
|
hProv,
|
|
X509_ASN_ENCODING,
|
|
const_cast<CERT_PUBLIC_KEY_INFO *>(pPublicKeyInfo),
|
|
&hKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptImportPublicKeyInfo");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// get public key from container
|
|
|
|
if (!CryptGetUserKey(hProv, AT_SIGNATURE, &hKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGetUserKey");
|
|
}
|
|
}
|
|
|
|
if (!CryptVerifySignature(hHash, pbSignature, cbSignature, hKey, NULL, 0))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptVerifySignature");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != hHash)
|
|
{
|
|
CryptDestroyHash(hHash);
|
|
}
|
|
if (NULL != hKey)
|
|
{
|
|
CryptDestroyKey(hKey);
|
|
}
|
|
if (NULL != pbSignature)
|
|
{
|
|
LocalFree(pbSignature);
|
|
}
|
|
CSASSERT(S_OK == hr || FAILED(hr));
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myValidateKeyForEncrypting(
|
|
IN HCRYPTPROV hProv,
|
|
IN CERT_PUBLIC_KEY_INFO const *pPublicKeyInfo,
|
|
IN ALG_ID algId)
|
|
{
|
|
HRESULT hr;
|
|
HCRYPTKEY hKeyPub = NULL;
|
|
HCRYPTKEY hKeySym = NULL;
|
|
HCRYPTKEY hKeyPri = NULL;
|
|
BYTE *pbKeySym = NULL;
|
|
DWORD cbKeySym;
|
|
BYTE abRandom[64];
|
|
BYTE *pbEncrypted = NULL;
|
|
BYTE *pbDecrypted = NULL;
|
|
DWORD cbEncrypted;
|
|
DWORD cbDecrypted;
|
|
|
|
// import public key into provider
|
|
|
|
if (!CryptImportPublicKeyInfo(
|
|
hProv,
|
|
X509_ASN_ENCODING,
|
|
const_cast<CERT_PUBLIC_KEY_INFO *>(pPublicKeyInfo),
|
|
&hKeyPub))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptImportPublicKeyInfo");
|
|
}
|
|
|
|
// Generate a default sized symmetric session key
|
|
|
|
if (!CryptGenKey(hProv, algId, CRYPT_EXPORTABLE, &hKeySym))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGenKey");
|
|
}
|
|
|
|
// create some random data
|
|
|
|
if (!CryptGenRandom(hProv, ARRAYSIZE(abRandom), abRandom))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGenRandom");
|
|
}
|
|
#if 1
|
|
DBGPRINT((DBG_SS_CERTLIBI, "Plain text size = %d\n", ARRAYSIZE(abRandom)));
|
|
DBGDUMPHEX((
|
|
DBG_SS_CERTLIBI,
|
|
DH_NOADDRESS | DH_NOTABPREFIX | 8,
|
|
abRandom,
|
|
ARRAYSIZE(abRandom)));
|
|
#endif
|
|
hr = myCryptEncrypt(
|
|
hKeySym,
|
|
abRandom,
|
|
ARRAYSIZE(abRandom),
|
|
&pbEncrypted,
|
|
&cbEncrypted);
|
|
_JumpIfError(hr, error, "myCryptEncrypt");
|
|
|
|
#if 1
|
|
DBGPRINT((DBG_SS_CERTLIBI, "Encrypted size = %d\n", cbEncrypted));
|
|
DBGDUMPHEX((
|
|
DBG_SS_CERTLIBI,
|
|
DH_NOADDRESS | DH_NOTABPREFIX | 8,
|
|
pbEncrypted,
|
|
cbEncrypted));
|
|
#endif
|
|
|
|
hr = myCryptExportKey(hKeySym, hKeyPub, SIMPLEBLOB, 0, &pbKeySym, &cbKeySym);
|
|
_JumpIfError(hr, error, "myCryptExportKey");
|
|
|
|
CryptDestroyKey(hKeySym);
|
|
hKeySym = NULL;
|
|
|
|
#if 1
|
|
DBGPRINT((DBG_SS_CERTLIBI, "SIMPLEBLOB:\n"));
|
|
DBGDUMPHEX((
|
|
DBG_SS_CERTLIBI,
|
|
DH_NOADDRESS | DH_NOTABPREFIX | DH_PRIVATEDATA | 8,
|
|
pbKeySym,
|
|
cbKeySym));
|
|
#endif
|
|
|
|
// get private key from container and import session key blob
|
|
|
|
if (!CryptGetUserKey(hProv, AT_KEYEXCHANGE, &hKeyPri))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGetUserKey");
|
|
}
|
|
if (!CryptImportKey(
|
|
hProv,
|
|
pbKeySym,
|
|
cbKeySym,
|
|
hKeyPri,
|
|
CRYPT_EXPORTABLE,
|
|
&hKeySym))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptImportKey");
|
|
}
|
|
hr = myCryptDecrypt(
|
|
hKeySym,
|
|
pbEncrypted,
|
|
cbEncrypted,
|
|
&pbDecrypted,
|
|
&cbDecrypted);
|
|
_JumpIfError(hr, error, "myCryptDecrypt");
|
|
|
|
#if 1
|
|
DBGPRINT((DBG_SS_CERTLIBI, "Decrypted size = %d\n", cbDecrypted));
|
|
DBGDUMPHEX((
|
|
DBG_SS_CERTLIBI,
|
|
DH_NOADDRESS | DH_NOTABPREFIX | 8,
|
|
pbDecrypted,
|
|
cbDecrypted));
|
|
#endif
|
|
|
|
if (ARRAYSIZE(abRandom) != cbDecrypted ||
|
|
0 != memcmp(abRandom, pbDecrypted, ARRAYSIZE(abRandom)))
|
|
{
|
|
hr = NTE_BAD_KEY;
|
|
_JumpIfError(hr, error, "Decrypted data mismatch");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != hKeyPub)
|
|
{
|
|
CryptDestroyKey(hKeyPub);
|
|
}
|
|
if (NULL != hKeySym)
|
|
{
|
|
CryptDestroyKey(hKeySym);
|
|
}
|
|
if (NULL != hKeyPri)
|
|
{
|
|
CryptDestroyKey(hKeyPri);
|
|
}
|
|
if (NULL != pbEncrypted)
|
|
{
|
|
LocalFree(pbEncrypted);
|
|
}
|
|
if (NULL != pbDecrypted)
|
|
{
|
|
LocalFree(pbDecrypted);
|
|
}
|
|
if (NULL != pbKeySym)
|
|
{
|
|
LocalFree(pbKeySym);
|
|
}
|
|
CSASSERT(S_OK == hr || FAILED(hr));
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myValidateSigningKey(
|
|
IN WCHAR const *pwszKeyContainerName,
|
|
IN WCHAR const *pwszProvName,
|
|
IN DWORD dwProvType,
|
|
IN BOOL fCryptSilent,
|
|
IN BOOL fMachineKeyset,
|
|
IN BOOL fForceSignatureTest,
|
|
IN OPTIONAL CERT_CONTEXT const *pcc,
|
|
IN OPTIONAL CERT_PUBLIC_KEY_INFO const *pPublicKeyInfo,
|
|
IN ALG_ID idAlg,
|
|
OPTIONAL OUT BOOL *pfSigningTestAttempted,
|
|
OPTIONAL OUT HCRYPTPROV *phProv)
|
|
{
|
|
HRESULT hr;
|
|
HCRYPTPROV hProv = NULL;
|
|
BOOL fCountSupported;
|
|
BOOL fCountEnabled;
|
|
BOOL fMatchingKey;
|
|
|
|
if (NULL != pfSigningTestAttempted)
|
|
{
|
|
*pfSigningTestAttempted = FALSE;
|
|
}
|
|
if (NULL != phProv)
|
|
{
|
|
*phProv = NULL;
|
|
}
|
|
if (!myCertSrvCryptAcquireContext(
|
|
&hProv,
|
|
pwszKeyContainerName,
|
|
pwszProvName,
|
|
dwProvType,
|
|
fCryptSilent? CRYPT_SILENT : 0,
|
|
fMachineKeyset))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(
|
|
hr,
|
|
error,
|
|
"myCertSrvCryptAcquireContext",
|
|
pwszKeyContainerName);
|
|
}
|
|
|
|
if (NULL == pPublicKeyInfo && NULL != pcc)
|
|
{
|
|
pPublicKeyInfo = &pcc->pCertInfo->SubjectPublicKeyInfo;
|
|
}
|
|
if (NULL != pPublicKeyInfo)
|
|
{
|
|
hr = myVerifyPublicKeyFromHProv(
|
|
hProv,
|
|
AT_SIGNATURE,
|
|
pcc,
|
|
NULL != pcc && CERT_V1 == pcc->pCertInfo->dwVersion,
|
|
pPublicKeyInfo,
|
|
&fMatchingKey);
|
|
_JumpIfError(hr, error, "myVerifyPublicKeyFromHProv");
|
|
|
|
if (!fMatchingKey)
|
|
{
|
|
hr = NTE_BAD_KEY;
|
|
_JumpError(hr, error, "fMatchingKey");
|
|
}
|
|
}
|
|
|
|
hr = myGetSigningKeyUsageCount(
|
|
hProv,
|
|
&fCountSupported,
|
|
&fCountEnabled,
|
|
NULL); // puliCount
|
|
_JumpIfError(hr, error, "myGetSigningKeyUsageCount");
|
|
|
|
if (fForceSignatureTest || !fCountSupported || !fCountEnabled)
|
|
{
|
|
// test signing
|
|
|
|
if (NULL != pfSigningTestAttempted)
|
|
{
|
|
*pfSigningTestAttempted = TRUE;
|
|
}
|
|
hr = myValidateKeyForSigning(hProv, pPublicKeyInfo, idAlg);
|
|
_JumpIfError(hr, error, "myValidateKeyForSigning");
|
|
}
|
|
if (NULL != phProv)
|
|
{
|
|
*phProv = hProv;
|
|
hProv = NULL;
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != hProv)
|
|
{
|
|
CryptReleaseContext(hProv, 0);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
GenerateTemporaryContainerName(
|
|
OUT WCHAR **ppwszKeyContainerName)
|
|
{
|
|
HRESULT hr;
|
|
GUID guid;
|
|
WCHAR awcbuf[MAX_PATH];
|
|
DWORD cb;
|
|
DWORD cwc;
|
|
|
|
myGenerateGuidSerialNumber(&guid);
|
|
|
|
wcscpy(awcbuf, L"KeyVerification--");
|
|
cwc = wcslen(awcbuf);
|
|
cb = sizeof(awcbuf) - cwc * sizeof(WCHAR);
|
|
hr = MultiByteIntegerToWszBuf(
|
|
FALSE,
|
|
sizeof(guid),
|
|
(BYTE const *) &guid,
|
|
&cb,
|
|
&awcbuf[cwc]);
|
|
_JumpIfError(hr, error, "MultiByteIntegerToWszBuf");
|
|
|
|
hr = myDupString(awcbuf, ppwszKeyContainerName);
|
|
_JumpIfError(hr, error, "myDupString");
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
VOID
|
|
DeleteRSAKeyContainer(
|
|
IN WCHAR const *pwszKeyContainerName,
|
|
IN BOOL fMachineKeyset)
|
|
{
|
|
HCRYPTPROV hProv;
|
|
DWORD dwFlags = CRYPT_DELETEKEYSET;
|
|
|
|
if (fMachineKeyset)
|
|
{
|
|
dwFlags |= CRYPT_MACHINE_KEYSET;
|
|
}
|
|
CryptAcquireContext(
|
|
&hProv,
|
|
pwszKeyContainerName,
|
|
NULL, // pwszProvName
|
|
PROV_RSA_FULL,
|
|
dwFlags);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myValidateKeyBlob(
|
|
IN BYTE const *pbKey,
|
|
IN DWORD cbKey,
|
|
IN CERT_PUBLIC_KEY_INFO const *pPublicKeyInfo,
|
|
IN BOOL fV1Cert,
|
|
OUT BOOL *pfSigningKey,
|
|
OPTIONAL OUT CRYPT_KEY_PROV_INFO *pkpi)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszKeyContainerName = NULL;
|
|
HCRYPTPROV hProv = NULL;
|
|
HCRYPTKEY hKey = NULL;
|
|
CRYPT_KEY_PROV_INFO kpi;
|
|
BOOL fMatchingKey;
|
|
|
|
// Use a NULL container name in a CRYPT_VERIFYCONTEXT to avoid persisting
|
|
// the private key -- unless the caller asked for the KeyProvInfo to be
|
|
// filled in and returned.
|
|
|
|
*pfSigningKey = FALSE;
|
|
|
|
if (NULL != pkpi)
|
|
{
|
|
hr = GenerateTemporaryContainerName(&pwszKeyContainerName);
|
|
_JumpIfError(hr, error, "GenerateTemporaryContainerName");
|
|
|
|
DeleteRSAKeyContainer(pwszKeyContainerName, FALSE);
|
|
|
|
DBGPRINT((DBG_SS_CERTLIBI, "Key Container: %ws\n", pwszKeyContainerName));
|
|
}
|
|
|
|
if (!CryptAcquireContext(
|
|
&hProv,
|
|
pwszKeyContainerName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
NULL == pkpi? CRYPT_VERIFYCONTEXT : CRYPT_NEWKEYSET))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptAcquireContext");
|
|
}
|
|
|
|
if (!CryptImportKey(
|
|
hProv,
|
|
pbKey,
|
|
cbKey,
|
|
NULL, // hPubKey
|
|
CRYPT_EXPORTABLE,
|
|
&hKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptImportKey");
|
|
}
|
|
CryptDestroyKey(hKey);
|
|
hKey = NULL;
|
|
|
|
ZeroMemory(&kpi, sizeof(kpi));
|
|
kpi.pwszContainerName = pwszKeyContainerName;
|
|
kpi.dwProvType = PROV_RSA_FULL;
|
|
kpi.dwKeySpec = AT_KEYEXCHANGE;
|
|
if (!CryptGetUserKey(hProv, kpi.dwKeySpec, &hKey))
|
|
{
|
|
hr = myHLastError();
|
|
_PrintError2(hr, "CryptGetUserKey", hr);
|
|
|
|
kpi.dwKeySpec = AT_SIGNATURE;
|
|
if (!CryptGetUserKey(hProv, kpi.dwKeySpec, &hKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGetUserKey");
|
|
}
|
|
*pfSigningKey = TRUE;
|
|
}
|
|
CryptDestroyKey(hKey);
|
|
hKey = NULL;
|
|
|
|
if (NULL != pkpi)
|
|
{
|
|
CryptReleaseContext(hProv, 0);
|
|
hProv = NULL;
|
|
|
|
if (!CryptAcquireContext(
|
|
&hProv,
|
|
pwszKeyContainerName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
0))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptAcquireContext");
|
|
}
|
|
}
|
|
if (AT_SIGNATURE == kpi.dwKeySpec)
|
|
{
|
|
hr = myValidateKeyForSigning(hProv, pPublicKeyInfo, CALG_SHA1);
|
|
_PrintIfError(hr, "myValidateKeyForSigning");
|
|
}
|
|
else
|
|
{
|
|
hr = myValidateKeyForEncrypting(hProv, pPublicKeyInfo, CALG_RC4);
|
|
_PrintIfError(hr, "myValidateKeyForEncrypting");
|
|
}
|
|
if (S_OK != hr)
|
|
{
|
|
_JumpError(hr, error, "Key Validation");
|
|
}
|
|
|
|
hr = myVerifyPublicKeyFromHProv(
|
|
hProv,
|
|
kpi.dwKeySpec,
|
|
NULL, // pCert
|
|
fV1Cert,
|
|
pPublicKeyInfo,
|
|
&fMatchingKey);
|
|
_JumpIfError(hr, error, "myVerifyPublicKeyFromHProv");
|
|
|
|
if (!fMatchingKey)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "Key doesn't match cert");
|
|
}
|
|
if (NULL != pkpi)
|
|
{
|
|
*pkpi = kpi;
|
|
pwszKeyContainerName = NULL;
|
|
}
|
|
|
|
error:
|
|
if (NULL != hKey)
|
|
{
|
|
CryptDestroyKey(hKey);
|
|
}
|
|
if (NULL != hProv)
|
|
{
|
|
CryptReleaseContext(hProv, 0);
|
|
}
|
|
if (NULL != pwszKeyContainerName)
|
|
{
|
|
DeleteRSAKeyContainer(pwszKeyContainerName, FALSE);
|
|
LocalFree(pwszKeyContainerName);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
/*
|
|
* myEnumProviders
|
|
*
|
|
* Purpose:
|
|
* Enumerate the providers.
|
|
*
|
|
* Parameters:
|
|
* IN dwIndex - Index to the providers to enumerate
|
|
* IN pdwReserved - Reserved for future use
|
|
* IN dwFlags - Flags parameter
|
|
* OUT pdwProvType - The type of the provider
|
|
* OUT ppwszProvName - Name of the enumerated provider
|
|
*/
|
|
|
|
HRESULT
|
|
myEnumProviders(
|
|
IN DWORD dwIndex,
|
|
IN DWORD *pdwReserved,
|
|
IN DWORD dwFlags,
|
|
OUT DWORD *pdwProvType,
|
|
OUT WCHAR **ppwszProvName)
|
|
{
|
|
HRESULT hr;
|
|
char *pszProvName = NULL;
|
|
DWORD cbProvName;
|
|
|
|
*ppwszProvName = NULL;
|
|
|
|
for (;;)
|
|
{
|
|
if (!CryptEnumProvidersA(
|
|
dwIndex,
|
|
pdwReserved,
|
|
dwFlags,
|
|
pdwProvType,
|
|
pszProvName,
|
|
&cbProvName))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError2(hr, error, "CryptEnumProvidersA", hr);
|
|
}
|
|
if (NULL != pszProvName)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// allocate ansi string buffer
|
|
|
|
pszProvName = (char *) LocalAlloc(LMEM_FIXED, cbProvName);
|
|
if (NULL == pszProvName)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
}
|
|
|
|
if (!myConvertSzToWsz(ppwszProvName, pszProvName, -1))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "myConvertSzToWsz");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pszProvName)
|
|
{
|
|
LocalFree(pszProvName);
|
|
}
|
|
return(hr);
|
|
}
|