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.
3467 lines
81 KiB
3467 lines
81 KiB
//+--------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1996 - 1999
|
|
//
|
|
// File: backup.cpp
|
|
//
|
|
// Contents: Cert Server wrapper routines
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#include <pch.cpp>
|
|
|
|
#pragma hdrstop
|
|
|
|
#include <esent.h>
|
|
|
|
#include "certdb.h"
|
|
#include "cscsp.h"
|
|
#include "certlibp.h"
|
|
|
|
#define __dwFILE__ __dwFILE_CERTLIB_BACKUP_CPP__
|
|
|
|
#define _64k (64 * 1024)
|
|
|
|
|
|
DWORD
|
|
_64kBlocks(
|
|
IN DWORD nFileSizeHigh,
|
|
IN DWORD nFileSizeLow)
|
|
{
|
|
LARGE_INTEGER li;
|
|
|
|
li.HighPart = nFileSizeHigh;
|
|
li.LowPart = nFileSizeLow;
|
|
return((DWORD) ((li.QuadPart + _64k - 1) / _64k));
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myLargeAlloc(
|
|
OUT DWORD *pcbLargeAlloc,
|
|
OUT BYTE **ppbLargeAlloc)
|
|
{
|
|
HRESULT hr;
|
|
|
|
// at 512k the server begins doing efficient backups
|
|
*pcbLargeAlloc = 512 * 1024;
|
|
*ppbLargeAlloc = (BYTE *) VirtualAlloc(
|
|
NULL,
|
|
*pcbLargeAlloc,
|
|
MEM_COMMIT,
|
|
PAGE_READWRITE);
|
|
if (NULL == *ppbLargeAlloc)
|
|
{
|
|
// couldn't alloc a large chunk? Try 64k...
|
|
|
|
*pcbLargeAlloc = _64k;
|
|
*ppbLargeAlloc = (BYTE *) VirtualAlloc(
|
|
NULL,
|
|
*pcbLargeAlloc,
|
|
MEM_COMMIT,
|
|
PAGE_READWRITE);
|
|
if (NULL == *ppbLargeAlloc)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "VirtualAlloc");
|
|
}
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
// Files to look for when checking for an existing DB, AND
|
|
// Files to delete when clearing out a DB or DB Log directory:
|
|
// Do NOT delete certsrv.mdb from Cert server 1.0!
|
|
|
|
WCHAR const * const g_apwszDBFileMatchPatterns[] =
|
|
{
|
|
L"res*.log",
|
|
TEXT(szDBBASENAMEPARM) L"*.log", // "edb*.log"
|
|
TEXT(szDBBASENAMEPARM) L"*.chk", // "edb*.chk"
|
|
L"*" wszDBFILENAMEEXT, // "*.edb"
|
|
NULL
|
|
};
|
|
|
|
|
|
HRESULT
|
|
myDeleteDBFilesInDir(
|
|
IN WCHAR const *pwszDir)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR const * const *ppwsz;
|
|
|
|
for (ppwsz = g_apwszDBFileMatchPatterns; NULL != *ppwsz; ppwsz++)
|
|
{
|
|
hr = myDeleteFilePattern(pwszDir, *ppwsz, FALSE);
|
|
_JumpIfError(hr, error, "myDeleteFilePattern");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
DoFilesExistInDir(
|
|
IN WCHAR const *pwszDir,
|
|
IN WCHAR const *pwszPattern,
|
|
OUT BOOL *pfFilesExist,
|
|
OPTIONAL OUT WCHAR **ppwszFileInUse)
|
|
{
|
|
HRESULT hr;
|
|
HANDLE hf = INVALID_HANDLE_VALUE;
|
|
WCHAR *pwszFindPattern = NULL;
|
|
WIN32_FIND_DATA wfd;
|
|
|
|
*pfFilesExist = FALSE;
|
|
if (NULL != ppwszFileInUse)
|
|
{
|
|
*ppwszFileInUse = NULL;
|
|
}
|
|
|
|
hr = myBuildPathAndExt(pwszDir, pwszPattern, NULL, &pwszFindPattern);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hf = FindFirstFile(pwszFindPattern, &wfd);
|
|
if (INVALID_HANDLE_VALUE == hf)
|
|
{
|
|
hr = S_OK;
|
|
goto error;
|
|
}
|
|
do
|
|
{
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
//printf("File: %ws\n", wfd.cFileName);
|
|
*pfFilesExist = TRUE;
|
|
|
|
if (NULL != ppwszFileInUse)
|
|
{
|
|
WCHAR *pwszFile;
|
|
|
|
hr = myBuildPathAndExt(pwszDir, wfd.cFileName, NULL, &pwszFile);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
if (myIsFileInUse(pwszFile))
|
|
{
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIB,
|
|
"DoFilesExistInDir: File In Use: %ws\n",
|
|
pwszFile));
|
|
|
|
*ppwszFileInUse = pwszFile;
|
|
hr = S_OK;
|
|
goto error;
|
|
}
|
|
LocalFree(pwszFile);
|
|
}
|
|
|
|
} while (FindNextFile(hf, &wfd));
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (INVALID_HANDLE_VALUE != hf)
|
|
{
|
|
FindClose(hf);
|
|
}
|
|
if (NULL != pwszFindPattern)
|
|
{
|
|
LocalFree(pwszFindPattern);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myDoDBFilesExistInDir(
|
|
IN WCHAR const *pwszDir,
|
|
OUT BOOL *pfFilesExist,
|
|
OPTIONAL OUT WCHAR **ppwszFileInUse)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR const * const *ppwsz;
|
|
|
|
*pfFilesExist = FALSE;
|
|
if (NULL != ppwszFileInUse)
|
|
{
|
|
*ppwszFileInUse = NULL;
|
|
}
|
|
|
|
hr = S_OK;
|
|
for (ppwsz = g_apwszDBFileMatchPatterns; NULL != *ppwsz; ppwsz++)
|
|
{
|
|
BOOL fFilesExist;
|
|
|
|
hr = DoFilesExistInDir(
|
|
pwszDir,
|
|
*ppwsz,
|
|
&fFilesExist,
|
|
ppwszFileInUse);
|
|
_JumpIfError(hr, error, "DoFilesExistInDir");
|
|
|
|
if (fFilesExist)
|
|
{
|
|
*pfFilesExist = TRUE;
|
|
}
|
|
if (NULL != ppwszFileInUse && NULL != *ppwszFileInUse)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
CSASSERT(S_OK == hr);
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
DoDBFilesExistInRegDir(
|
|
IN WCHAR const *pwszRegName,
|
|
OUT BOOL *pfFilesExist,
|
|
OPTIONAL OUT WCHAR **ppwszFileInUse)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszDir = NULL;
|
|
|
|
*pfFilesExist = FALSE;
|
|
if (NULL != ppwszFileInUse)
|
|
{
|
|
*ppwszFileInUse = NULL;
|
|
}
|
|
|
|
hr = myGetCertRegStrValue(NULL, NULL, NULL, pwszRegName, &pwszDir);
|
|
if (S_OK != hr)
|
|
{
|
|
if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr)
|
|
{
|
|
// reg entry doesn't exist, that's fine
|
|
goto done;
|
|
}
|
|
_JumpError(hr, error, "myGetCertRegStrValue");
|
|
}
|
|
|
|
hr = myDoDBFilesExistInDir(pwszDir, pfFilesExist, ppwszFileInUse);
|
|
_JumpIfError(hr, error, "myDoDBFilesExistInDir");
|
|
|
|
done:
|
|
hr = S_OK;
|
|
error:
|
|
if (NULL != pwszDir)
|
|
{
|
|
LocalFree(pwszDir);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
BuildDBFileName(
|
|
IN WCHAR const *pwszSanitizedName,
|
|
OUT WCHAR **ppwszDBFile)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszDir = NULL;
|
|
|
|
*ppwszDBFile = NULL;
|
|
|
|
// get existing db path
|
|
|
|
hr = myGetCertRegStrValue(NULL, NULL, NULL, wszREGDBDIRECTORY, &pwszDir);
|
|
_JumpIfError(hr, error, "myGetCertRegStrValue");
|
|
|
|
// form existing db file path
|
|
hr = myBuildPathAndExt(
|
|
pwszDir,
|
|
pwszSanitizedName,
|
|
wszDBFILENAMEEXT,
|
|
ppwszDBFile);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
error:
|
|
if (NULL != pwszDir)
|
|
{
|
|
LocalFree(pwszDir);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
WCHAR const * const g_apwszDBRegNames[] =
|
|
{
|
|
wszREGDBDIRECTORY,
|
|
wszREGDBLOGDIRECTORY,
|
|
wszREGDBSYSDIRECTORY,
|
|
wszREGDBTEMPDIRECTORY,
|
|
NULL
|
|
};
|
|
|
|
|
|
// Verify that the DB and DB Log directories in the registry contain existing
|
|
// DB files, to decide whether the DB could be reused by cert server setup.
|
|
// Also see if any of the DB files are in use -- we don't want to point to the
|
|
// same directory as the DS DB and trash the DS, for example.
|
|
|
|
HRESULT
|
|
myDoDBFilesExist(
|
|
IN WCHAR const *pwszSanitizedName,
|
|
OUT BOOL *pfFilesExist,
|
|
OPTIONAL OUT WCHAR **ppwszFileInUse)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR const * const *ppwsz;
|
|
WCHAR *pwszDBFile = NULL;
|
|
|
|
*pfFilesExist = FALSE;
|
|
if (NULL != ppwszFileInUse)
|
|
{
|
|
*ppwszFileInUse = NULL;
|
|
}
|
|
|
|
// this is very primitive, just check for existence
|
|
|
|
// get existing db file path
|
|
|
|
hr = BuildDBFileName(pwszSanitizedName, &pwszDBFile);
|
|
if (S_OK == hr)
|
|
{
|
|
// If the main DB file doesn't exist, there's no point in continuing!
|
|
|
|
if (!myDoesFileExist(pwszDBFile))
|
|
{
|
|
CSASSERT(S_OK == hr);
|
|
goto error;
|
|
}
|
|
*pfFilesExist = TRUE;
|
|
|
|
if (NULL != ppwszFileInUse && myIsFileInUse(pwszDBFile))
|
|
{
|
|
*ppwszFileInUse = pwszDBFile;
|
|
pwszDBFile = NULL;
|
|
CSASSERT(S_OK == hr);
|
|
goto error;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_PrintError(hr, "BuildDBFileName");
|
|
}
|
|
|
|
for (ppwsz = g_apwszDBRegNames; NULL != *ppwsz; ppwsz++)
|
|
{
|
|
BOOL fFilesExist;
|
|
|
|
hr = DoDBFilesExistInRegDir(*ppwsz, &fFilesExist, ppwszFileInUse);
|
|
_JumpIfError(hr, error, "DoDBFilesExistInRegDir");
|
|
|
|
if (fFilesExist)
|
|
{
|
|
*pfFilesExist = TRUE;
|
|
}
|
|
if (NULL != ppwszFileInUse && NULL != *ppwszFileInUse)
|
|
{
|
|
CSASSERT(S_OK == hr);
|
|
goto error;
|
|
}
|
|
}
|
|
CSASSERT(S_OK == hr);
|
|
|
|
error:
|
|
if (NULL != pwszDBFile)
|
|
{
|
|
LocalFree(pwszDBFile);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
BackupCopyDBFile(
|
|
IN HCSBC hcsbc,
|
|
IN WCHAR const *pwszDBFile,
|
|
IN WCHAR const *pwszBackupFile,
|
|
IN DWORD dwPercentCompleteBase,
|
|
IN DWORD dwPercentCompleteDelta,
|
|
OUT DWORD *pdwPercentComplete)
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hr2;
|
|
HANDLE hFileBackup = INVALID_HANDLE_VALUE;
|
|
BOOL fOpen = FALSE;
|
|
LARGE_INTEGER licbFile;
|
|
DWORD cbRead;
|
|
DWORD cbWritten;
|
|
DWORD dwPercentCompleteCurrent;
|
|
DWORD ReadLoopMax;
|
|
DWORD ReadLoopCurrent;
|
|
DWORD cbLargeAlloc;
|
|
BYTE *pbLargeAlloc = NULL;
|
|
|
|
hr = myLargeAlloc(&cbLargeAlloc, &pbLargeAlloc);
|
|
_JumpIfError(hr, error, "myLargeAlloc");
|
|
|
|
//printf("Copy %ws to %ws\n", pwszDBFile, pwszBackupFile);
|
|
|
|
hr = CertSrvBackupOpenFile(hcsbc, pwszDBFile, cbLargeAlloc, &licbFile);
|
|
_JumpIfError(hr, error, "CertSrvBackupOpenFile");
|
|
|
|
fOpen = TRUE;
|
|
|
|
hFileBackup = CreateFile(
|
|
pwszBackupFile,
|
|
GENERIC_WRITE,
|
|
0,
|
|
NULL,
|
|
CREATE_NEW,
|
|
0,
|
|
NULL);
|
|
if (hFileBackup == INVALID_HANDLE_VALUE)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "CreateFile", pwszBackupFile);
|
|
}
|
|
|
|
dwPercentCompleteCurrent = dwPercentCompleteBase;
|
|
ReadLoopMax =
|
|
(DWORD) ((licbFile.QuadPart + cbLargeAlloc - 1) / cbLargeAlloc);
|
|
|
|
//printf("BackupDBFile: Percent per Read = %u, read count = %u\n", dwPercentCompleteDelta / ReadLoopMax, ReadLoopMax);
|
|
|
|
ReadLoopCurrent = 0;
|
|
|
|
while (0 != licbFile.QuadPart)
|
|
{
|
|
hr = CertSrvBackupRead(hcsbc, pbLargeAlloc, cbLargeAlloc, &cbRead);
|
|
_JumpIfError(hr, error, "CertSrvBackupRead");
|
|
|
|
//printf("CertSrvBackupRead(%x)\n", cbRead);
|
|
|
|
if (!WriteFile(hFileBackup, pbLargeAlloc, cbRead, &cbWritten, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "WriteFile", pwszBackupFile);
|
|
}
|
|
if (cbWritten != cbRead)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_HANDLE_EOF);
|
|
_JumpErrorStr(hr, error, "WriteFile", pwszBackupFile);
|
|
}
|
|
licbFile.QuadPart -= cbRead;
|
|
|
|
ReadLoopCurrent++;
|
|
|
|
dwPercentCompleteCurrent =
|
|
dwPercentCompleteBase +
|
|
(ReadLoopCurrent * dwPercentCompleteDelta) / ReadLoopMax;
|
|
CSASSERT(dwPercentCompleteCurrent <= dwPercentCompleteBase + dwPercentCompleteDelta);
|
|
CSASSERT(*pdwPercentComplete <= dwPercentCompleteCurrent);
|
|
*pdwPercentComplete = dwPercentCompleteCurrent;
|
|
//printf("BackupDBFile: PercentComplete = %u\n", *pdwPercentComplete);
|
|
}
|
|
CSASSERT(*pdwPercentComplete <= dwPercentCompleteBase + dwPercentCompleteDelta);
|
|
*pdwPercentComplete = dwPercentCompleteBase + dwPercentCompleteDelta;
|
|
//printf("BackupDBFile: PercentComplete = %u (EOF)\n", *pdwPercentComplete);
|
|
|
|
error:
|
|
if (INVALID_HANDLE_VALUE != hFileBackup)
|
|
{
|
|
CloseHandle(hFileBackup);
|
|
}
|
|
if (fOpen)
|
|
{
|
|
hr2 = CertSrvBackupClose(hcsbc);
|
|
_PrintIfError(hr2, "CertSrvBackupClose");
|
|
}
|
|
if (NULL != pbLargeAlloc)
|
|
{
|
|
VirtualFree(pbLargeAlloc, 0, MEM_RELEASE);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
BackupDBFileList(
|
|
IN HCSBC hcsbc,
|
|
IN BOOL fDBFiles,
|
|
IN WCHAR const *pwszDir,
|
|
OUT DWORD *pdwPercentComplete)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszzList = NULL;
|
|
WCHAR const *pwsz;
|
|
DWORD cfile;
|
|
DWORD cb;
|
|
WCHAR const *pwszFile;
|
|
WCHAR wszPath[MAX_PATH];
|
|
DWORD dwPercentCompleteCurrent;
|
|
DWORD dwPercentComplete1File;
|
|
|
|
if (fDBFiles)
|
|
{
|
|
hr = CertSrvBackupGetDatabaseNames(hcsbc, &pwszzList, &cb);
|
|
_JumpIfError(hr, error, "CertSrvBackupGetDatabaseNames");
|
|
}
|
|
else
|
|
{
|
|
hr = CertSrvBackupGetBackupLogs(hcsbc, &pwszzList, &cb);
|
|
_JumpIfError(hr, error, "CertSrvBackupGetBackupLogs");
|
|
}
|
|
|
|
// prefix complains this might happen, then deref'd below
|
|
if (pwszzList == NULL)
|
|
{
|
|
hr = E_UNEXPECTED;
|
|
_JumpError(hr, error, "BackupDBFileList");
|
|
}
|
|
|
|
cfile = 0;
|
|
for (pwsz = pwszzList; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
|
|
{
|
|
cfile++;
|
|
}
|
|
if (0 != cfile)
|
|
{
|
|
dwPercentCompleteCurrent = 0;
|
|
dwPercentComplete1File = 100 / cfile;
|
|
//printf("BackupDBFileList: Percent per File = %u\n", dwPercentComplete1File);
|
|
for (pwsz = pwszzList; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
|
|
{
|
|
pwszFile = wcsrchr(pwsz, L'\\');
|
|
if (NULL == pwszFile)
|
|
{
|
|
pwszFile = pwsz;
|
|
}
|
|
else
|
|
{
|
|
pwszFile++;
|
|
}
|
|
if (wcslen(pwszDir) + 1 + wcslen(pwszFile) >= ARRAYSIZE(wszPath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszPath", pwszDir);
|
|
}
|
|
wcscpy(wszPath, pwszDir);
|
|
wcscat(wszPath, L"\\");
|
|
wcscat(wszPath, pwszFile);
|
|
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIBI,
|
|
"BackupDBFileList: %x %ws -> %ws\n",
|
|
*pwsz,
|
|
&pwsz[1],
|
|
wszPath));
|
|
|
|
hr = BackupCopyDBFile(
|
|
hcsbc,
|
|
&pwsz[1],
|
|
wszPath,
|
|
dwPercentCompleteCurrent,
|
|
dwPercentComplete1File,
|
|
pdwPercentComplete);
|
|
_JumpIfError(hr, error, "BackupCopyDBFile");
|
|
|
|
dwPercentCompleteCurrent += dwPercentComplete1File;
|
|
CSASSERT(*pdwPercentComplete == dwPercentCompleteCurrent);
|
|
//printf("BackupDBFileList: PercentComplete = %u\n", *pdwPercentComplete);
|
|
}
|
|
}
|
|
CSASSERT(*pdwPercentComplete <= 100);
|
|
*pdwPercentComplete = 100;
|
|
//printf("BackupDBFileList: PercentComplete = %u (END)\n", *pdwPercentComplete);
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pwszzList)
|
|
{
|
|
CertSrvBackupFree(pwszzList);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
#define wszBSSTARDOTSTAR L"\\*.*"
|
|
#define wszBSSTAR L"\\*"
|
|
|
|
BOOL
|
|
myIsDirEmpty(
|
|
IN WCHAR const *pwszDir)
|
|
{
|
|
HRESULT hr;
|
|
HANDLE hf;
|
|
WIN32_FIND_DATA wfd;
|
|
WCHAR wszpath[MAX_PATH];
|
|
BOOL fEmpty = TRUE;
|
|
|
|
if (wcslen(pwszDir) + WSZARRAYSIZE(wszBSSTARDOTSTAR) >= ARRAYSIZE(wszpath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszDir);
|
|
}
|
|
wcscpy(wszpath, pwszDir);
|
|
wcscat(wszpath, wszBSSTARDOTSTAR);
|
|
|
|
hf = FindFirstFile(wszpath, &wfd);
|
|
if (INVALID_HANDLE_VALUE != hf)
|
|
{
|
|
do {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
fEmpty = FALSE;
|
|
//printf("File: %ws\n", wfd.cFileName);
|
|
break;
|
|
|
|
} while (FindNextFile(hf, &wfd));
|
|
FindClose(hf);
|
|
}
|
|
error:
|
|
return(fEmpty);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myForceDirEmpty(
|
|
IN WCHAR const *pwszDir)
|
|
{
|
|
HRESULT hr;
|
|
HANDLE hf;
|
|
WIN32_FIND_DATA wfd;
|
|
WCHAR *pwszFile;
|
|
WCHAR wszpath[MAX_PATH];
|
|
DWORD cwcBase;
|
|
|
|
cwcBase = wcslen(pwszDir);
|
|
if (cwcBase + WSZARRAYSIZE(wszBSSTARDOTSTAR) >= ARRAYSIZE(wszpath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszDir);
|
|
}
|
|
wcscpy(wszpath, pwszDir);
|
|
wcscat(wszpath, wszBSSTARDOTSTAR);
|
|
cwcBase++;
|
|
pwszFile = &wszpath[cwcBase];
|
|
|
|
hf = FindFirstFile(wszpath, &wfd);
|
|
if (INVALID_HANDLE_VALUE == hf)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpIfError(hr, error, "FindFirstFile");
|
|
}
|
|
hr = S_OK;
|
|
do {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
if (cwcBase + wcslen(wfd.cFileName) >= ARRAYSIZE(wszpath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_PrintErrorStr(hr, "wszpath", pwszDir);
|
|
continue;
|
|
}
|
|
wcscpy(pwszFile, wfd.cFileName);
|
|
//printf("File: %ws\n", wszpath);
|
|
DeleteFile(wszpath);
|
|
} while (FindNextFile(hf, &wfd));
|
|
FindClose(hf);
|
|
_JumpIfErrorStr(hr, error, "wszpath", pwszDir);
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
BOOL
|
|
myIsDirectory(IN WCHAR const *pwszDirectoryPath)
|
|
{
|
|
WIN32_FILE_ATTRIBUTE_DATA data;
|
|
|
|
return(
|
|
GetFileAttributesEx(pwszDirectoryPath, GetFileExInfoStandard, &data) &&
|
|
(FILE_ATTRIBUTE_DIRECTORY & data.dwFileAttributes));
|
|
}
|
|
|
|
|
|
BOOL
|
|
myIsFileInUse(
|
|
IN WCHAR const *pwszFile)
|
|
{
|
|
BOOL fInUse = FALSE;
|
|
HANDLE hFile;
|
|
|
|
hFile = CreateFile(
|
|
pwszFile,
|
|
GENERIC_WRITE, // dwDesiredAccess
|
|
0, // no share
|
|
NULL, // lpSecurityAttributes
|
|
OPEN_EXISTING, // open only & fail if doesn't exist
|
|
0, // dwFlagAndAttributes
|
|
NULL); // hTemplateFile
|
|
if (INVALID_HANDLE_VALUE == hFile)
|
|
{
|
|
if (ERROR_SHARING_VIOLATION == GetLastError())
|
|
{
|
|
fInUse = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CloseHandle(hFile);
|
|
}
|
|
return(fInUse);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myCreateBackupDir(
|
|
IN WCHAR const *pwszDir,
|
|
IN BOOL fForceOverWrite)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (!myIsDirectory(pwszDir))
|
|
{
|
|
if (!CreateDirectory(pwszDir, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
if (HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS) != hr)
|
|
{
|
|
_JumpErrorStr(hr, error, "CreateDirectory", pwszDir);
|
|
}
|
|
} // else dir created successfully
|
|
} // else dir already exists
|
|
|
|
if (!myIsDirEmpty(pwszDir))
|
|
{
|
|
if (!fForceOverWrite)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIR_NOT_EMPTY);
|
|
_JumpErrorStr(hr, error, "myIsDirEmpty", pwszDir);
|
|
}
|
|
hr = myForceDirEmpty(pwszDir);
|
|
_JumpIfErrorStr(hr, error, "myForceDirEmpty", pwszDir);
|
|
|
|
if (!myIsDirEmpty(pwszDir))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIR_NOT_EMPTY);
|
|
_JumpErrorStr(hr, error, "myIsDirEmpty", pwszDir);
|
|
}
|
|
} // else is empty
|
|
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
// if Flags & CDBBACKUP_VERIFYONLY, create and verify the target directory is empty
|
|
|
|
HRESULT
|
|
myBackupDB(
|
|
IN WCHAR const *pwszConfig,
|
|
IN DWORD Flags,
|
|
IN WCHAR const *pwszBackupDir,
|
|
OPTIONAL OUT DBBACKUPPROGRESS *pdbp)
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hr2;
|
|
BOOL fServerOnline;
|
|
HCSBC hcsbc;
|
|
BOOL fBegin = FALSE;
|
|
WCHAR *pwszPathDBDir = NULL;
|
|
WCHAR *pwszDATFile = NULL;
|
|
WCHAR *pwszzFileList = NULL;
|
|
DWORD cbList;
|
|
DBBACKUPPROGRESS dbp;
|
|
LONG grbitJet;
|
|
LONG BackupFlags;
|
|
BOOL fImpersonating = FALSE;
|
|
|
|
hcsbc = NULL;
|
|
if (NULL == pwszConfig)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "NULL pwszConfig");
|
|
}
|
|
|
|
if (NULL == pdbp)
|
|
{
|
|
pdbp = &dbp;
|
|
}
|
|
ZeroMemory(pdbp, sizeof(*pdbp));
|
|
|
|
if (!ImpersonateSelf(SecurityImpersonation))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "ImpersonateSelf");
|
|
}
|
|
fImpersonating = TRUE;
|
|
|
|
hr = myEnablePrivilege(SE_BACKUP_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
|
|
if (NULL == pwszBackupDir)
|
|
{
|
|
hr = E_POINTER;
|
|
_JumpError(hr, error, "NULL parm");
|
|
}
|
|
if (~CDBBACKUP_BACKUPVALID & Flags)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Flags");
|
|
}
|
|
|
|
if (!myIsDirectory(pwszBackupDir))
|
|
{
|
|
if (!CreateDirectory(pwszBackupDir, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
if (HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS) != hr)
|
|
{
|
|
_JumpError(hr, error, "CreateDirectory");
|
|
}
|
|
}
|
|
}
|
|
|
|
hr = myBuildPathAndExt(
|
|
pwszBackupDir,
|
|
wszDBBACKUPSUBDIR,
|
|
NULL,
|
|
&pwszPathDBDir);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = myCreateBackupDir(
|
|
pwszPathDBDir,
|
|
(CDBBACKUP_OVERWRITE & Flags)? TRUE : FALSE);
|
|
_JumpIfError(hr, error, "myCreateBackupDir");
|
|
|
|
//if (NULL != pwszConfig)
|
|
if (0 == (Flags & CDBBACKUP_VERIFYONLY))
|
|
{
|
|
hr = CertSrvIsServerOnline(pwszConfig, &fServerOnline);
|
|
_JumpIfError(hr, error, "CertSrvIsServerOnline");
|
|
|
|
//printf("Cert Server Online -> %d\n", fServerOnline);
|
|
|
|
if (!fServerOnline)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(RPC_S_SERVER_UNAVAILABLE);
|
|
_JumpError(hr, error, "CertSrvIsServerOnline");
|
|
}
|
|
|
|
BackupFlags = CSBACKUP_TYPE_FULL;
|
|
grbitJet = 0;
|
|
if (CDBBACKUP_INCREMENTAL & Flags)
|
|
{
|
|
grbitJet |= JET_bitBackupIncremental;
|
|
BackupFlags = CSBACKUP_TYPE_LOGS_ONLY;
|
|
}
|
|
if (CDBBACKUP_KEEPOLDLOGS & Flags)
|
|
{
|
|
// JetBeginExternalBackup can't handle setting this bit
|
|
// grbitJet |= JET_bitKeepOldLogs;
|
|
}
|
|
|
|
hr = CertSrvBackupPrepare(pwszConfig, grbitJet, BackupFlags, &hcsbc);
|
|
_JumpIfError(hr, error, "CertSrvBackupPrepare");
|
|
|
|
fBegin = TRUE;
|
|
|
|
if (0 == (CDBBACKUP_INCREMENTAL & Flags))
|
|
{
|
|
hr = CertSrvRestoreGetDatabaseLocations(hcsbc, &pwszzFileList, &cbList);
|
|
_JumpIfError(hr, error, "CertSrvRestoreGetDatabaseLocations");
|
|
|
|
hr = myBuildPathAndExt(
|
|
pwszPathDBDir,
|
|
wszDBBACKUPCERTBACKDAT,
|
|
NULL,
|
|
&pwszDATFile);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = EncodeToFileW(
|
|
pwszDATFile,
|
|
(BYTE const *) pwszzFileList,
|
|
cbList,
|
|
CRYPT_STRING_BINARY);
|
|
_JumpIfError(hr, error, "EncodeToFileW");
|
|
|
|
hr = BackupDBFileList(
|
|
hcsbc,
|
|
TRUE,
|
|
pwszPathDBDir,
|
|
&pdbp->dwDBPercentComplete);
|
|
_JumpIfError(hr, error, "BackupDBFileList(DB)");
|
|
}
|
|
else
|
|
{
|
|
pdbp->dwDBPercentComplete = 100;
|
|
}
|
|
//printf("DB Done: dwDBPercentComplete = %u\n", pdbp->dwDBPercentComplete);
|
|
|
|
hr = BackupDBFileList(
|
|
hcsbc,
|
|
FALSE,
|
|
pwszPathDBDir,
|
|
&pdbp->dwLogPercentComplete);
|
|
_JumpIfError(hr, error, "BackupDBFileList(Log)");
|
|
//printf("Log Done: dwLogPercentComplete = %u\n", pdbp->dwLogPercentComplete);
|
|
|
|
if (0 == (CDBBACKUP_KEEPOLDLOGS & Flags))
|
|
{
|
|
hr = CertSrvBackupTruncateLogs(hcsbc);
|
|
_JumpIfError(hr, error, "CertSrvBackupTruncateLogs");
|
|
}
|
|
pdbp->dwTruncateLogPercentComplete = 100;
|
|
//printf("Truncate Done: dwTruncateLogPercentComplete = %u\n", pdbp->dwTruncateLogPercentComplete);
|
|
}
|
|
|
|
error:
|
|
if (NULL != pwszzFileList)
|
|
{
|
|
CertSrvBackupFree(pwszzFileList);
|
|
}
|
|
if (fBegin)
|
|
{
|
|
hr2 = CertSrvBackupEnd(hcsbc);
|
|
_PrintIfError(hr2, "CertSrvBackupEnd");
|
|
if (S_OK == hr)
|
|
{
|
|
hr = hr2;
|
|
}
|
|
}
|
|
if (NULL != pwszDATFile)
|
|
{
|
|
LocalFree(pwszDATFile);
|
|
}
|
|
if (NULL != pwszPathDBDir)
|
|
{
|
|
LocalFree(pwszPathDBDir);
|
|
}
|
|
if (fImpersonating)
|
|
{
|
|
myEnablePrivilege(SE_BACKUP_NAME, FALSE);
|
|
RevertToSelf();
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
// Verify the backup file names only, and return the log file numeric range.
|
|
|
|
HRESULT
|
|
myVerifyBackupDirectory(
|
|
IN WCHAR const *pwszConfig,
|
|
IN DWORD Flags,
|
|
IN WCHAR const *pwszPathDBDir,
|
|
OUT DWORD *plogMin,
|
|
OUT DWORD *plogMax,
|
|
OUT DWORD *pc64kDBBlocks, // 64k blocks in DB files to be restored
|
|
OUT DWORD *pc64kLogBlocks) // 64k blocks in Log files to be restored
|
|
{
|
|
HRESULT hr;
|
|
HANDLE hf = INVALID_HANDLE_VALUE;
|
|
WIN32_FIND_DATA wfd;
|
|
WCHAR wszpath[2 * MAX_PATH];
|
|
WCHAR wszfile[MAX_PATH];
|
|
BOOL fSawEDBFile = FALSE;
|
|
BOOL fSawDatFile = FALSE;
|
|
DWORD cLogFiles = 0;
|
|
WCHAR *pwszCA;
|
|
WCHAR *pwszRevertCA = NULL;
|
|
WCHAR *pwszSanitizedCA = NULL;
|
|
WCHAR *pwszExt;
|
|
WCHAR *pwsz;
|
|
DWORD log;
|
|
|
|
*plogMin = MAXDWORD;
|
|
*plogMax = 0;
|
|
*pc64kDBBlocks = 0;
|
|
*pc64kLogBlocks = 0;
|
|
wszpath[0] = L'\0';
|
|
|
|
pwszCA = wcschr(pwszConfig, L'\\');
|
|
if (NULL != pwszCA)
|
|
{
|
|
pwszCA++; // point to CA Name
|
|
|
|
hr = myRevertSanitizeName(pwszCA, &pwszRevertCA);
|
|
_JumpIfError(hr, error, "myRevertSanitizeName");
|
|
|
|
hr = mySanitizeName(pwszRevertCA, &pwszSanitizedCA);
|
|
_JumpIfError(hr, error, "mySanitizeName");
|
|
}
|
|
|
|
if (wcslen(pwszPathDBDir) + WSZARRAYSIZE(wszBSSTARDOTSTAR) >= ARRAYSIZE(wszpath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszPathDBDir);
|
|
}
|
|
wcscpy(wszpath, pwszPathDBDir);
|
|
wcscat(wszpath, wszBSSTARDOTSTAR);
|
|
|
|
hf = FindFirstFile(wszpath, &wfd);
|
|
if (INVALID_HANDLE_VALUE == hf)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpError(hr, error, "missing backup files");
|
|
}
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIRECTORY);
|
|
do {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
//printf("File: %ws\n", wfd.cFileName);
|
|
|
|
if (wcslen(wfd.cFileName) >= ARRAYSIZE(wszfile))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszPathDBDir);
|
|
}
|
|
wcscpy(wszfile, wfd.cFileName);
|
|
|
|
pwszExt = wcsrchr(wszfile, L'.');
|
|
if (NULL == pwszExt)
|
|
{
|
|
_JumpError(hr, error, "file missing extension");
|
|
}
|
|
*pwszExt++ = L'\0';
|
|
|
|
if (0 == mylstrcmpiS(pwszExt, &wszLOGFILENAMEEXT[1]))
|
|
{
|
|
if (0 != _wcsnicmp(wszfile, wszDBBASENAMEPARM, 3))
|
|
{
|
|
_JumpErrorStr(hr, error, "bad log prefix", wfd.cFileName);
|
|
}
|
|
for (pwsz = &wszfile[3]; L'\0' != *pwsz; pwsz++)
|
|
{
|
|
if (!iswxdigit(*pwsz))
|
|
{
|
|
_JumpErrorStr(hr, error, "bad name digit", wfd.cFileName);
|
|
}
|
|
}
|
|
log = wcstoul(&wszfile[3], NULL, 16);
|
|
if (log > *plogMax)
|
|
{
|
|
//printf("Log %x: max = %x -> %x\n", log, *plogMax, log);
|
|
*plogMax = log;
|
|
}
|
|
if (log < *plogMin)
|
|
{
|
|
//printf("Log %x: min = %x -> %x\n", log, *plogMin, log);
|
|
*plogMin = log;
|
|
}
|
|
*pc64kLogBlocks += _64kBlocks(wfd.nFileSizeHigh, wfd.nFileSizeLow);
|
|
cLogFiles++;
|
|
}
|
|
else
|
|
if (0 == mylstrcmpiS(pwszExt, &wszDBFILENAMEEXT[1]))
|
|
{
|
|
if (fSawEDBFile)
|
|
{
|
|
_JumpError(hr, error, "multiple *.edb files");
|
|
}
|
|
if (NULL != pwszSanitizedCA &&
|
|
0 != mylstrcmpiL(wszfile, pwszSanitizedCA))
|
|
{
|
|
_PrintErrorStr(hr, "expected base name", pwszSanitizedCA);
|
|
_JumpErrorStr(hr, error, "base name mismatch", wfd.cFileName);
|
|
}
|
|
*pc64kDBBlocks += _64kBlocks(wfd.nFileSizeHigh, wfd.nFileSizeLow);
|
|
fSawEDBFile = TRUE;
|
|
}
|
|
else
|
|
if (0 == mylstrcmpiS(pwszExt, &wszDATFILENAMEEXT[1]))
|
|
{
|
|
if (fSawDatFile)
|
|
{
|
|
_JumpError(hr, error, "multiple *.dat files");
|
|
}
|
|
if (LSTRCMPIS(wfd.cFileName, wszDBBACKUPCERTBACKDAT))
|
|
{
|
|
_JumpErrorStr(hr, error, "unexpected file", wfd.cFileName);
|
|
}
|
|
fSawDatFile = TRUE;
|
|
}
|
|
else
|
|
{
|
|
_JumpErrorStr(hr, error, "unexpected extension", wfd.cFileName);
|
|
}
|
|
} while (FindNextFile(hf, &wfd));
|
|
|
|
//printf("clog=%u: %u - %u edb=%u\n", cLogFiles, *plogMin, *plogMax, fSawEDBFile);
|
|
|
|
if (0 == cLogFiles)
|
|
{
|
|
_JumpError(hr, error, "missing log file(s)");
|
|
}
|
|
if (0 == (CDBBACKUP_INCREMENTAL & Flags))
|
|
{
|
|
if (!fSawEDBFile || !fSawDatFile)
|
|
{
|
|
_JumpError(hr, error, "missing full backup file(s)");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (fSawEDBFile || fSawDatFile)
|
|
{
|
|
_JumpError(hr, error, "unexpected incremental backup file(s)");
|
|
}
|
|
}
|
|
|
|
if (*plogMax - *plogMin + 1 != cLogFiles)
|
|
{
|
|
_JumpError(hr, error, "missing log file(s)");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pwszRevertCA)
|
|
{
|
|
LocalFree(pwszRevertCA);
|
|
}
|
|
if (NULL != pwszSanitizedCA)
|
|
{
|
|
LocalFree(pwszSanitizedCA);
|
|
}
|
|
if (INVALID_HANDLE_VALUE != hf)
|
|
{
|
|
FindClose(hf);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myGetRegUNCDBDir(
|
|
IN HKEY hkey,
|
|
IN WCHAR const *pwszReg,
|
|
OPTIONAL IN WCHAR const *pwszServer,
|
|
IN WCHAR const **ppwszUNCDir)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwType;
|
|
DWORD cb;
|
|
WCHAR *pwszDir = NULL;
|
|
WCHAR *pwszUNCDir;
|
|
|
|
*ppwszUNCDir = NULL;
|
|
hr = RegQueryValueEx(hkey, pwszReg, NULL, &dwType, NULL, &cb);
|
|
if (S_OK != hr)
|
|
{
|
|
hr = myHError(hr);
|
|
_JumpErrorStr(hr, error, "RegQueryValueEx", pwszReg);
|
|
}
|
|
|
|
pwszDir = (WCHAR *) LocalAlloc(LMEM_FIXED, cb);
|
|
if (NULL == pwszDir)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
|
|
hr = RegQueryValueEx(hkey, pwszReg, NULL, &dwType, (BYTE *) pwszDir, &cb);
|
|
if (S_OK != hr)
|
|
{
|
|
hr = myHError(hr);
|
|
_JumpErrorStr(hr, error, "RegQueryValueEx", pwszReg);
|
|
}
|
|
|
|
hr = myConvertLocalPathToUNC(pwszServer, pwszDir, &pwszUNCDir);
|
|
_JumpIfError(hr, error, "myConvertLocalPathToUNC");
|
|
|
|
*ppwszUNCDir = pwszUNCDir;
|
|
|
|
error:
|
|
if (NULL != pwszDir)
|
|
{
|
|
LocalFree(pwszDir);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myCopyUNCPath(
|
|
IN WCHAR const *pwszIn,
|
|
OPTIONAL IN WCHAR const *pwszDnsName,
|
|
OUT WCHAR const **ppwszOut)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszOut;
|
|
WCHAR const *pwsz;
|
|
|
|
*ppwszOut = NULL;
|
|
|
|
if (L'\\' != pwszIn[0] || L'\\' != pwszIn[1])
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "bad parm");
|
|
}
|
|
if (NULL == pwszDnsName)
|
|
{
|
|
hr = myConvertUNCPathToLocal(pwszIn, &pwszOut);
|
|
_JumpIfError(hr, error, "myConvertUNCPathToLocal");
|
|
}
|
|
else
|
|
{
|
|
pwsz = wcschr(&pwszIn[2], L'\\');
|
|
if (NULL == pwsz)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "bad parm");
|
|
}
|
|
pwszOut = (WCHAR *) LocalAlloc(
|
|
LMEM_FIXED,
|
|
(2 + wcslen(pwszDnsName) + wcslen(pwsz) + 1) * sizeof(WCHAR));
|
|
if (NULL == pwszOut)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
wcscpy(pwszOut, L"\\\\");
|
|
wcscat(pwszOut, pwszDnsName);
|
|
wcscat(pwszOut, pwsz);
|
|
}
|
|
*ppwszOut = pwszOut;
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myGetDBPaths(
|
|
IN WCHAR const *pwszConfig,
|
|
OPTIONAL IN WCHAR const *pwszLogPath,
|
|
OPTIONAL IN WCHAR const *pwszzFileList,
|
|
OUT WCHAR const **ppwszDBDir,
|
|
OUT WCHAR const **ppwszLogDir,
|
|
OUT WCHAR const **ppwszSystemDir)
|
|
{
|
|
HRESULT hr;
|
|
HKEY hkey = NULL;
|
|
WCHAR *pwszDnsName = NULL;
|
|
WCHAR *pwszRegPath = NULL;
|
|
WCHAR *pwszDBDir = NULL;
|
|
WCHAR const *pwsz;
|
|
WCHAR const *pwszT;
|
|
BOOL fLocal;
|
|
|
|
*ppwszDBDir = NULL;
|
|
*ppwszLogDir = NULL;
|
|
*ppwszSystemDir = NULL;
|
|
|
|
hr = myIsConfigLocal(pwszConfig, NULL, &fLocal);
|
|
_JumpIfError(hr, error, "myIsConfigLocal");
|
|
|
|
if (fLocal)
|
|
{
|
|
pwszConfig = NULL;
|
|
}
|
|
else
|
|
{
|
|
hr = myGetMachineDnsName(&pwszDnsName);
|
|
_JumpIfError(hr, error, "myGetMachineDnsName");
|
|
}
|
|
|
|
hr = myRegOpenRelativeKey(
|
|
fLocal? NULL : pwszConfig,
|
|
L"",
|
|
0,
|
|
&pwszRegPath,
|
|
NULL, // ppwszName
|
|
&hkey);
|
|
_JumpIfErrorStr(hr, error, "myRegOpenRelativeKey", pwszConfig);
|
|
|
|
// Find old database path:
|
|
|
|
pwszT = NULL;
|
|
pwsz = NULL;
|
|
if (NULL != pwszzFileList)
|
|
{
|
|
for (pwsz = pwszzFileList; L'\0' != *pwsz; pwsz += wcslen(pwsz) + 1)
|
|
{
|
|
if (CSBFT_CERTSERVER_DATABASE == *pwsz)
|
|
{
|
|
pwsz++;
|
|
pwszT = wcsrchr(pwsz, L'\\');
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (NULL != pwszT)
|
|
{
|
|
DWORD cwc = SAFE_SUBTRACT_POINTERS(pwszT, pwsz);
|
|
|
|
pwszDBDir = (WCHAR *) LocalAlloc(LMEM_FIXED, (cwc + 1) * sizeof(WCHAR));
|
|
if (NULL == pwszDBDir)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
CopyMemory(pwszDBDir, pwsz, cwc * sizeof(WCHAR));
|
|
pwszDBDir[cwc] = L'\0';
|
|
|
|
hr = myCopyUNCPath(pwszDBDir, pwszDnsName, ppwszDBDir);
|
|
_JumpIfError(hr, error, "myCopyUNCPath");
|
|
}
|
|
else
|
|
{
|
|
hr = myGetRegUNCDBDir(hkey, wszREGDBDIRECTORY, pwszDnsName, ppwszDBDir);
|
|
_JumpIfError(hr, error, "myGetRegUNCDBDir");
|
|
}
|
|
|
|
if (NULL != pwszLogPath)
|
|
{
|
|
hr = myCopyUNCPath(pwszLogPath, pwszDnsName, ppwszLogDir);
|
|
_JumpIfError(hr, error, "myCopyUNCPath");
|
|
}
|
|
else
|
|
{
|
|
hr = myGetRegUNCDBDir(
|
|
hkey,
|
|
wszREGDBLOGDIRECTORY,
|
|
pwszDnsName,
|
|
ppwszLogDir);
|
|
_JumpIfError(hr, error, "myGetRegUNCDBDir");
|
|
}
|
|
|
|
hr = myGetRegUNCDBDir(
|
|
hkey,
|
|
wszREGDBSYSDIRECTORY,
|
|
pwszDnsName,
|
|
ppwszSystemDir);
|
|
_JumpIfError(hr, error, "myGetRegUNCDBDir");
|
|
|
|
error:
|
|
if (S_OK != hr)
|
|
{
|
|
if (NULL != *ppwszDBDir)
|
|
{
|
|
LocalFree(const_cast<WCHAR *>(*ppwszDBDir));
|
|
*ppwszDBDir = NULL;
|
|
}
|
|
if (NULL != *ppwszLogDir)
|
|
{
|
|
LocalFree(const_cast<WCHAR *>(*ppwszLogDir));
|
|
*ppwszLogDir = NULL;
|
|
}
|
|
if (NULL != *ppwszSystemDir)
|
|
{
|
|
LocalFree(const_cast<WCHAR *>(*ppwszSystemDir));
|
|
*ppwszSystemDir = NULL;
|
|
}
|
|
}
|
|
if (NULL != hkey)
|
|
{
|
|
RegCloseKey(hkey);
|
|
}
|
|
if (NULL != pwszDBDir)
|
|
{
|
|
LocalFree(pwszDBDir);
|
|
}
|
|
if (NULL != pwszRegPath)
|
|
{
|
|
LocalFree(pwszRegPath);
|
|
}
|
|
if (NULL != pwszDnsName)
|
|
{
|
|
LocalFree(pwszDnsName);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
RestoreCopyFile(
|
|
IN BOOL fForceOverWrite,
|
|
IN WCHAR const *pwszSourceDir,
|
|
IN WCHAR const *pwszTargetDir,
|
|
IN WCHAR const *pwszFile,
|
|
IN DWORD nFileSizeHigh,
|
|
IN DWORD nFileSizeLow,
|
|
IN DWORD c64kBlocksTotal, // total file size
|
|
IN OUT DWORD *pc64kBlocksCurrent, // current file size sum
|
|
IN OUT DWORD *pdwPercentComplete)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszSource = NULL;
|
|
WCHAR *pwszTarget = NULL;
|
|
HANDLE hTarget = INVALID_HANDLE_VALUE;
|
|
HANDLE hSource = INVALID_HANDLE_VALUE;
|
|
LARGE_INTEGER licb;
|
|
LARGE_INTEGER licbRead;
|
|
DWORD cbRead;
|
|
DWORD cbWritten;
|
|
DWORD cbLargeAlloc;
|
|
BYTE *pbLargeAlloc = NULL;
|
|
DWORD c64kBlocksFile;
|
|
DWORD dwPercentComplete;
|
|
|
|
licb.HighPart = nFileSizeHigh;
|
|
licb.LowPart = nFileSizeLow;
|
|
|
|
hr = myBuildPathAndExt(pwszSourceDir, pwszFile, NULL, &pwszSource);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = myBuildPathAndExt(pwszTargetDir, pwszFile, NULL, &pwszTarget);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = myLargeAlloc(&cbLargeAlloc, &pbLargeAlloc);
|
|
_JumpIfError(hr, error, "myLargeAlloc");
|
|
|
|
hSource = CreateFile(
|
|
pwszSource,
|
|
GENERIC_READ,
|
|
0,
|
|
NULL,
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL);
|
|
if (hSource == INVALID_HANDLE_VALUE)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "CreateFile", pwszSource);
|
|
}
|
|
hTarget = CreateFile(
|
|
pwszTarget,
|
|
GENERIC_WRITE,
|
|
0,
|
|
NULL,
|
|
fForceOverWrite? CREATE_ALWAYS : CREATE_NEW,
|
|
0,
|
|
NULL);
|
|
if (hTarget == INVALID_HANDLE_VALUE)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "CreateFile", pwszTarget);
|
|
}
|
|
|
|
licbRead.QuadPart = 0;
|
|
c64kBlocksFile = 0;
|
|
while (licbRead.QuadPart < licb.QuadPart)
|
|
{
|
|
if (!ReadFile(hSource, pbLargeAlloc, cbLargeAlloc, &cbRead, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "ReadFile");
|
|
}
|
|
//printf("ReadFile(%x)\n", cbRead);
|
|
|
|
if (!WriteFile(hTarget, pbLargeAlloc, cbRead, &cbWritten, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpErrorStr(hr, error, "WriteFile", pwszTarget);
|
|
}
|
|
if (cbWritten != cbRead)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_HANDLE_EOF);
|
|
_JumpErrorStr(hr, error, "WriteFile", pwszTarget);
|
|
}
|
|
licbRead.QuadPart += cbRead;
|
|
|
|
c64kBlocksFile = _64kBlocks(licbRead.HighPart, licbRead.LowPart);
|
|
dwPercentComplete =
|
|
(100 * (c64kBlocksFile + *pc64kBlocksCurrent)) / c64kBlocksTotal;
|
|
|
|
CSASSERT(*pdwPercentComplete <= dwPercentComplete);
|
|
*pdwPercentComplete = dwPercentComplete;
|
|
//printf("RestoreCopyFile0: PercentComplete = %u\n", *pdwPercentComplete);
|
|
}
|
|
*pc64kBlocksCurrent += c64kBlocksFile;
|
|
dwPercentComplete = (100 * *pc64kBlocksCurrent) / c64kBlocksTotal;
|
|
CSASSERT(*pdwPercentComplete <= dwPercentComplete);
|
|
*pdwPercentComplete = dwPercentComplete;
|
|
//printf("RestoreCopyFile1: PercentComplete = %u\n", *pdwPercentComplete);
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (INVALID_HANDLE_VALUE != hTarget)
|
|
{
|
|
CloseHandle(hTarget);
|
|
}
|
|
if (INVALID_HANDLE_VALUE != hSource)
|
|
{
|
|
CloseHandle(hSource);
|
|
}
|
|
if (NULL != pwszSource)
|
|
{
|
|
LocalFree(pwszSource);
|
|
}
|
|
if (NULL != pwszTarget)
|
|
{
|
|
LocalFree(pwszTarget);
|
|
}
|
|
if (NULL != pbLargeAlloc)
|
|
{
|
|
VirtualFree(pbLargeAlloc, 0, MEM_RELEASE);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
RestoreCopyFilePattern(
|
|
IN BOOL fForceOverWrite,
|
|
IN WCHAR const *pwszSourceDir,
|
|
IN WCHAR const *pwszTargetDir,
|
|
IN WCHAR const *pwszFilePattern,
|
|
IN DWORD c64kBlocksTotal, // total file size
|
|
IN OUT DWORD *pc64kBlocksCurrent, // current file size sum
|
|
IN OUT DWORD *pdwPercentComplete)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwszPattern = NULL;
|
|
HANDLE hf = INVALID_HANDLE_VALUE;
|
|
WIN32_FIND_DATA wfd;
|
|
|
|
hr = myBuildPathAndExt(pwszSourceDir, pwszFilePattern, NULL, &pwszPattern);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hf = FindFirstFile(pwszPattern, &wfd);
|
|
if (INVALID_HANDLE_VALUE == hf)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpErrorStr(hr, error, "missing source files", pwszPattern);
|
|
}
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIRECTORY);
|
|
do {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
//printf("File: %ws\n", wfd.cFileName);
|
|
hr = RestoreCopyFile(
|
|
fForceOverWrite,
|
|
pwszSourceDir, // source dir
|
|
pwszTargetDir, // target dir
|
|
wfd.cFileName,
|
|
wfd.nFileSizeHigh,
|
|
wfd.nFileSizeLow,
|
|
c64kBlocksTotal, // total file size
|
|
pc64kBlocksCurrent, // current file size sum
|
|
pdwPercentComplete);
|
|
_JumpIfError(hr, error, "RestoreCopyFile");
|
|
|
|
} while (FindNextFile(hf, &wfd));
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (INVALID_HANDLE_VALUE != hf)
|
|
{
|
|
FindClose(hf);
|
|
}
|
|
if (NULL != pwszPattern)
|
|
{
|
|
LocalFree(pwszPattern);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myRestoreDBFiles(
|
|
IN WCHAR const *pwszConfig,
|
|
IN DWORD Flags,
|
|
IN WCHAR const *pwszBackupDir,
|
|
OPTIONAL IN WCHAR const *pwszLogPath,
|
|
OPTIONAL IN WCHAR const *pwszzFileList, // NULL if incremental restore
|
|
IN DWORD c64kDBBlocks,
|
|
IN DWORD c64kLogBlocks,
|
|
OPTIONAL OUT DBBACKUPPROGRESS *pdbp)
|
|
{
|
|
HRESULT hr;
|
|
DWORD i;
|
|
#define IDIR_DB 0
|
|
#define IDIR_LOG 1
|
|
#define IDIR_SYSTEM 2
|
|
WCHAR const *apwszDirs[3] = { NULL, NULL, NULL };
|
|
DWORD c64kBlocksCurrent;
|
|
BOOL fForceOverWrite = 0 != (CDBBACKUP_OVERWRITE & Flags);
|
|
WCHAR *pwszFileInUse = NULL;
|
|
|
|
// Get DB, Log & System paths from registry
|
|
|
|
hr = myGetDBPaths(
|
|
pwszConfig,
|
|
pwszLogPath,
|
|
pwszzFileList,
|
|
&apwszDirs[IDIR_DB],
|
|
&apwszDirs[IDIR_LOG],
|
|
&apwszDirs[IDIR_SYSTEM]);
|
|
_JumpIfError(hr, error, "myGetDBPaths");
|
|
|
|
DBGPRINT((DBG_SS_CERTLIBI, "DBDir: %ws\n", apwszDirs[IDIR_DB]));
|
|
DBGPRINT((DBG_SS_CERTLIBI, "LogDir: %ws\n", apwszDirs[IDIR_LOG]));
|
|
DBGPRINT((DBG_SS_CERTLIBI, "SysDir: %ws\n", apwszDirs[IDIR_SYSTEM]));
|
|
|
|
CSASSERT((NULL == pwszzFileList) ^ (0 == (CDBBACKUP_INCREMENTAL & Flags)));
|
|
for (i = 0; i < ARRAYSIZE(apwszDirs); i++)
|
|
{
|
|
BOOL fFilesExist;
|
|
|
|
if (!myIsDirectory(apwszDirs[i]))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIRECTORY);
|
|
_JumpErrorStr(hr, error, "not a directory", apwszDirs[i]);
|
|
}
|
|
hr = myDoDBFilesExistInDir(apwszDirs[i], &fFilesExist, &pwszFileInUse);
|
|
_JumpIfError(hr, error, "myDoDBFilesExistInDir");
|
|
|
|
if (NULL != pwszFileInUse)
|
|
{
|
|
_PrintErrorStr(
|
|
HRESULT_FROM_WIN32(ERROR_BUSY),
|
|
"myDoDBFilesExistInDir",
|
|
pwszFileInUse);
|
|
}
|
|
if (!fFilesExist)
|
|
{
|
|
if (CDBBACKUP_INCREMENTAL & Flags)
|
|
{
|
|
// Incremental restore -- some DB files should already exist
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIRECTORY);
|
|
_JumpErrorStr(hr, error, "myDoDBFilesExistInDir", apwszDirs[i]);
|
|
}
|
|
}
|
|
else if (0 == (CDBBACKUP_INCREMENTAL & Flags))
|
|
{
|
|
// Full restore -- no DB files should exist yet
|
|
|
|
if (!fForceOverWrite)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIR_NOT_EMPTY);
|
|
_JumpErrorStr(
|
|
hr,
|
|
error,
|
|
"myDoDBFilesExistInDir",
|
|
NULL != pwszFileInUse? pwszFileInUse : apwszDirs[i]);
|
|
}
|
|
hr = myDeleteDBFilesInDir(apwszDirs[i]);
|
|
if (S_OK != hr)
|
|
{
|
|
_PrintErrorStr(hr, "myDeleteDBFilesInDir", apwszDirs[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
// copy files to appropriate target directories
|
|
|
|
if (0 == (CDBBACKUP_INCREMENTAL & Flags))
|
|
{
|
|
c64kBlocksCurrent = 0;
|
|
hr = RestoreCopyFilePattern(
|
|
fForceOverWrite,
|
|
pwszBackupDir, // source dir
|
|
apwszDirs[IDIR_DB], // target dir
|
|
L"*" wszDBFILENAMEEXT, // match pattern
|
|
c64kDBBlocks,
|
|
&c64kBlocksCurrent, // current total file size
|
|
&pdbp->dwDBPercentComplete);
|
|
_JumpIfError(hr, error, "RestoreCopyFile");
|
|
|
|
CSASSERT(c64kDBBlocks == c64kBlocksCurrent);
|
|
}
|
|
CSASSERT(100 >= pdbp->dwDBPercentComplete);
|
|
pdbp->dwDBPercentComplete = 100;
|
|
|
|
c64kBlocksCurrent = 0;
|
|
hr = RestoreCopyFilePattern(
|
|
fForceOverWrite,
|
|
pwszBackupDir, // source dir
|
|
apwszDirs[IDIR_LOG], // target dir
|
|
L"*" wszLOGFILENAMEEXT, // match pattern
|
|
c64kLogBlocks,
|
|
&c64kBlocksCurrent, // current total file size
|
|
&pdbp->dwLogPercentComplete);
|
|
_JumpIfError(hr, error, "RestoreCopyFile");
|
|
|
|
CSASSERT(c64kLogBlocks == c64kBlocksCurrent);
|
|
|
|
CSASSERT(100 >= pdbp->dwLogPercentComplete);
|
|
pdbp->dwLogPercentComplete = 100;
|
|
|
|
CSASSERT(100 >= pdbp->dwTruncateLogPercentComplete);
|
|
pdbp->dwTruncateLogPercentComplete = 100;
|
|
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pwszFileInUse)
|
|
{
|
|
LocalFree(pwszFileInUse);
|
|
}
|
|
for (i = 0; i < ARRAYSIZE(apwszDirs); i++)
|
|
{
|
|
if (NULL != apwszDirs[i])
|
|
{
|
|
LocalFree(const_cast<WCHAR *>(apwszDirs[i]));
|
|
}
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myDeleteRestoreInProgressKey(
|
|
IN WCHAR const *pwszConfig)
|
|
{
|
|
HRESULT hr;
|
|
HKEY hkey = NULL;
|
|
WCHAR *pwszRegPath = NULL;
|
|
|
|
hr = myRegOpenRelativeKey(
|
|
pwszConfig,
|
|
L"",
|
|
RORKF_CREATESUBKEYS,
|
|
&pwszRegPath,
|
|
NULL, // ppwszName
|
|
&hkey);
|
|
_JumpIfErrorStr(hr, error, "myRegOpenRelativeKey", pwszConfig);
|
|
|
|
hr = RegDeleteKey(hkey, wszREGKEYRESTOREINPROGRESS);
|
|
_JumpIfError(hr, error, "RegDeleteKey");
|
|
|
|
error:
|
|
if (NULL != hkey)
|
|
{
|
|
RegCloseKey(hkey);
|
|
}
|
|
if (NULL != pwszRegPath)
|
|
{
|
|
LocalFree(pwszRegPath);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
// If CDBBACKUP_VERIFYONLY, only verify the passed directory contains valid
|
|
// files. If pwszBackupDir is NULL, delete the RestoreInProgress registry key.
|
|
|
|
HRESULT
|
|
myRestoreDB(
|
|
IN WCHAR const *pwszConfig,
|
|
IN DWORD Flags,
|
|
OPTIONAL IN WCHAR const *pwszBackupDir,
|
|
OPTIONAL IN WCHAR const *pwszCheckPointFilePath,
|
|
OPTIONAL IN WCHAR const *pwszLogPath,
|
|
OPTIONAL IN WCHAR const *pwszBackupLogPath,
|
|
OPTIONAL OUT DBBACKUPPROGRESS *pdbp)
|
|
{
|
|
HRESULT hr;
|
|
HRESULT hr2;
|
|
WCHAR buf[MAX_PATH];
|
|
WCHAR *pwszPathDBDir = NULL;
|
|
WCHAR *pwszDATFile = NULL;
|
|
WCHAR *pwszzFileList = NULL;
|
|
DWORD cbList;
|
|
CSEDB_RSTMAP RstMap[1];
|
|
DWORD crstmap = 0;
|
|
WCHAR *pwszFile;
|
|
DWORD logMin;
|
|
DWORD logMax;
|
|
HCSBC hcsbc;
|
|
BOOL fBegin = FALSE;
|
|
BOOL fImpersonating = FALSE;
|
|
DBBACKUPPROGRESS dbp;
|
|
DWORD c64kDBBlocks; // 64k blocks in DB files to be restored
|
|
DWORD c64kLogBlocks; // 64k blocks in Log files to be restored
|
|
|
|
if (NULL == pdbp)
|
|
{
|
|
pdbp = &dbp;
|
|
}
|
|
ZeroMemory(pdbp, sizeof(*pdbp));
|
|
hcsbc = NULL;
|
|
|
|
if (!ImpersonateSelf(SecurityImpersonation))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "ImpersonateSelf");
|
|
}
|
|
fImpersonating = TRUE;
|
|
|
|
hr = myEnablePrivilege(SE_RESTORE_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
hr = myEnablePrivilege(SE_BACKUP_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
|
|
if (NULL == pwszConfig ||
|
|
((CDBBACKUP_VERIFYONLY & Flags) && NULL == pwszBackupDir))
|
|
{
|
|
hr = E_POINTER;
|
|
_JumpError(hr, error, "NULL parm");
|
|
}
|
|
if (NULL != pwszBackupDir)
|
|
{
|
|
if (!GetFullPathName(pwszBackupDir, ARRAYSIZE(buf), buf, &pwszFile))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "GetFullPathName");
|
|
}
|
|
hr = myBuildPathAndExt(buf, wszDBBACKUPSUBDIR, NULL, &pwszPathDBDir);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = myVerifyBackupDirectory(
|
|
pwszConfig,
|
|
Flags,
|
|
pwszPathDBDir,
|
|
&logMin,
|
|
&logMax,
|
|
&c64kDBBlocks,
|
|
&c64kLogBlocks);
|
|
_JumpIfError(hr, error, "myVerifyBackupDirectory");
|
|
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIBI,
|
|
"c64kBlocks=%u+%u\n",
|
|
c64kDBBlocks,
|
|
c64kLogBlocks));
|
|
|
|
if (0 == (CDBBACKUP_INCREMENTAL & Flags))
|
|
{
|
|
hr = myBuildPathAndExt(
|
|
pwszPathDBDir,
|
|
wszDBBACKUPCERTBACKDAT,
|
|
NULL,
|
|
&pwszDATFile);
|
|
_JumpIfError(hr, error, "myBuildPathAndExt");
|
|
|
|
hr = DecodeFileW(
|
|
pwszDATFile,
|
|
(BYTE **) &pwszzFileList,
|
|
&cbList,
|
|
CRYPT_STRING_BINARY);
|
|
_JumpIfError(hr, error, "DecodeFileW");
|
|
|
|
if (2 * sizeof(WCHAR) >= cbList ||
|
|
L'\0' != pwszzFileList[cbList/sizeof(WCHAR) - 1] ||
|
|
L'\0' != pwszzFileList[cbList/sizeof(WCHAR) - 2])
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "pwszzFileList malformed");
|
|
}
|
|
RstMap[0].pwszDatabaseName = pwszzFileList;
|
|
RstMap[0].pwszNewDatabaseName = pwszzFileList;
|
|
crstmap = 1;
|
|
}
|
|
if (0 == (CDBBACKUP_VERIFYONLY & Flags))
|
|
{
|
|
hr = myRestoreDBFiles(
|
|
pwszConfig,
|
|
Flags,
|
|
pwszPathDBDir,
|
|
pwszLogPath,
|
|
pwszzFileList,
|
|
c64kDBBlocks,
|
|
c64kLogBlocks,
|
|
pdbp);
|
|
_JumpIfError(hr, error, "myRestoreDBFiles");
|
|
|
|
hr = CertSrvRestorePrepare(pwszConfig, CSRESTORE_TYPE_FULL, &hcsbc);
|
|
_JumpIfError(hr, error, "CertSrvRestorePrepare");
|
|
|
|
fBegin = TRUE;
|
|
|
|
hr = CertSrvRestoreRegister(
|
|
hcsbc,
|
|
pwszCheckPointFilePath,
|
|
pwszLogPath,
|
|
0 == crstmap? NULL : RstMap,
|
|
crstmap,
|
|
pwszBackupLogPath,
|
|
logMin,
|
|
logMax);
|
|
|
|
// When running only as backup operator, we don't have rights
|
|
// in the registry and CertSrvRestoreRegister fails with access
|
|
// denied. We try to mark for restore through a file.
|
|
|
|
if (E_ACCESSDENIED == hr)
|
|
{
|
|
hr = CertSrvRestoreRegisterThroughFile(
|
|
hcsbc,
|
|
pwszCheckPointFilePath,
|
|
pwszLogPath,
|
|
0 == crstmap? NULL : RstMap,
|
|
crstmap,
|
|
pwszBackupLogPath,
|
|
logMin,
|
|
logMax);
|
|
_JumpIfError(hr, error, "CertSrvRestoreRegisterThroughFile");
|
|
}
|
|
else
|
|
{
|
|
_JumpIfError(hr, error, "CertSrvRestoreRegister");
|
|
|
|
hr = CertSrvRestoreRegisterComplete(hcsbc, S_OK);
|
|
_JumpIfError(hr, error, "CertSrvRestoreRegisterComplete");
|
|
}
|
|
}
|
|
}
|
|
else if (0 == (CDBBACKUP_VERIFYONLY & Flags))
|
|
{
|
|
hr = myDeleteRestoreInProgressKey(pwszConfig);
|
|
_JumpIfError(hr, error, "myDeleteRestoreInProgressKey");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (fBegin)
|
|
{
|
|
hr2 = CertSrvRestoreEnd(hcsbc);
|
|
_PrintIfError(hr2, "CertSrvBackupEnd");
|
|
if (S_OK == hr)
|
|
{
|
|
hr = hr2;
|
|
}
|
|
}
|
|
if (NULL != pwszzFileList)
|
|
{
|
|
LocalFree(pwszzFileList);
|
|
}
|
|
if (NULL != pwszDATFile)
|
|
{
|
|
LocalFree(pwszDATFile);
|
|
}
|
|
if (NULL != pwszPathDBDir)
|
|
{
|
|
LocalFree(pwszPathDBDir);
|
|
}
|
|
if (fImpersonating)
|
|
{
|
|
myEnablePrivilege(SE_BACKUP_NAME, FALSE);
|
|
myEnablePrivilege(SE_RESTORE_NAME, FALSE);
|
|
RevertToSelf();
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
typedef BOOL (WINAPI FNPFXEXPORTCERTSTOREEX)(
|
|
IN HCERTSTORE hStore,
|
|
IN OUT CRYPT_DATA_BLOB* pPFX,
|
|
IN LPCWSTR szPassword,
|
|
IN VOID *pvReserved,
|
|
IN DWORD dwFlags);
|
|
|
|
FNPFXEXPORTCERTSTOREEX PFXExportCertStoreOld;
|
|
|
|
BOOL WINAPI
|
|
PFXExportCertStoreOld(
|
|
IN HCERTSTORE hStore,
|
|
IN OUT CRYPT_DATA_BLOB *ppfx,
|
|
IN WCHAR const *pwszPassword,
|
|
IN VOID *, // pvReserved
|
|
IN DWORD dwFlags)
|
|
{
|
|
return(PFXExportCertStore(hStore, ppfx, pwszPassword, dwFlags));
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myPFXExportCertStore(
|
|
IN HCERTSTORE hStore,
|
|
OUT CRYPT_DATA_BLOB *ppfx,
|
|
IN WCHAR const *pwszPassword,
|
|
IN BOOL fEnhancedStrength,
|
|
IN DWORD dwFlags)
|
|
{
|
|
HRESULT hr;
|
|
FNPFXEXPORTCERTSTOREEX *pfn;
|
|
char const *pszFunc;
|
|
|
|
if (fEnhancedStrength)
|
|
{
|
|
pfn = PFXExportCertStoreEx;
|
|
pszFunc = "PFXExportCertStoreEx";
|
|
}
|
|
else
|
|
{
|
|
pfn = PFXExportCertStoreOld;
|
|
pszFunc = "PFXExportCertStoreOld";
|
|
}
|
|
|
|
ppfx->pbData = NULL;
|
|
if (!(*pfn)(hStore, ppfx, pwszPassword, NULL, dwFlags))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, pszFunc);
|
|
}
|
|
ppfx->pbData = (BYTE *) LocalAlloc(LMEM_FIXED, ppfx->cbData);
|
|
if (NULL == ppfx->pbData)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "no memory for PFX blob");
|
|
}
|
|
if (!(*pfn)(hStore, ppfx, pwszPassword, NULL, dwFlags))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, pszFunc);
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
HRESULT
|
|
myAddChainToMemoryStore(
|
|
IN HCERTSTORE hMemoryStore,
|
|
IN CERT_CONTEXT const *pCertContext,
|
|
IN DWORD dwmsTimeout)
|
|
{
|
|
HRESULT hr;
|
|
DWORD i;
|
|
CERT_CHAIN_CONTEXT const *pCertChainContext = NULL;
|
|
CERT_CHAIN_PARA CertChainPara;
|
|
CERT_SIMPLE_CHAIN *pSimpleChain;
|
|
|
|
ZeroMemory(&CertChainPara, sizeof(CertChainPara));
|
|
CertChainPara.cbSize = sizeof(CertChainPara);
|
|
CertChainPara.dwUrlRetrievalTimeout = dwmsTimeout;
|
|
|
|
if (!CertGetCertificateChain(
|
|
HCCE_LOCAL_MACHINE,
|
|
pCertContext,
|
|
NULL,
|
|
NULL,
|
|
&CertChainPara,
|
|
0,
|
|
NULL,
|
|
&pCertChainContext))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CertGetCertificateChain");
|
|
}
|
|
|
|
// make sure there is at least 1 simple chain
|
|
|
|
if (0 == pCertChainContext->cChain)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpError(hr, error, "pCertChainContext->cChain");
|
|
}
|
|
|
|
pSimpleChain = pCertChainContext->rgpChain[0];
|
|
for (i = 0; i < pSimpleChain->cElement; i++)
|
|
{
|
|
if (!CertAddCertificateContextToStore(
|
|
hMemoryStore,
|
|
pSimpleChain->rgpElement[i]->pCertContext,
|
|
CERT_STORE_ADD_REPLACE_EXISTING,
|
|
NULL))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CertAddCertificateContextToStore");
|
|
}
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (pCertChainContext != NULL)
|
|
{
|
|
CertFreeCertificateChain(pCertChainContext);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SaveCACertChainToMemoryStore(
|
|
IN WCHAR const *pwszSanitizedName,
|
|
IN DWORD iCert,
|
|
IN HCERTSTORE hMyStore,
|
|
IN HCERTSTORE hTempMemoryStore,
|
|
IN DWORD dwmsTimeout)
|
|
{
|
|
HRESULT hr;
|
|
CERT_CONTEXT const *pccCA = NULL;
|
|
CRYPT_KEY_PROV_INFO *pkpi = NULL;
|
|
DWORD NameId;
|
|
|
|
hr = myFindCACertByHashIndex(
|
|
hMyStore,
|
|
pwszSanitizedName,
|
|
CSRH_CASIGCERT,
|
|
iCert,
|
|
&NameId,
|
|
&pccCA);
|
|
_JumpIfError(hr, error, "myFindCACertByHashIndex");
|
|
|
|
hr = myRepairCertKeyProviderInfo(pccCA, TRUE, &pkpi);
|
|
if (S_OK != hr)
|
|
{
|
|
if (CRYPT_E_NOT_FOUND != hr)
|
|
{
|
|
_JumpError(hr, error, "myRepairCertKeyProviderInfo");
|
|
}
|
|
}
|
|
else if (NULL != pkpi)
|
|
{
|
|
BOOL fMatchingKey;
|
|
|
|
hr = myVerifyPublicKey(
|
|
pccCA,
|
|
FALSE,
|
|
NULL, // pKeyProvInfo
|
|
NULL, // pPublicKeyInfo
|
|
&fMatchingKey);
|
|
if (S_OK != hr)
|
|
{
|
|
if (!IsHrSkipPrivateKey(hr))
|
|
{
|
|
_JumpError(hr, error, "myVerifyPublicKey");
|
|
}
|
|
_PrintError2(hr, "myVerifyPublicKey", hr);
|
|
}
|
|
else if (!fMatchingKey)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "Key doesn't match cert");
|
|
}
|
|
}
|
|
|
|
// Begin Chain Building
|
|
|
|
hr = myAddChainToMemoryStore(hTempMemoryStore, pccCA, dwmsTimeout);
|
|
_JumpIfError(hr, error, "myAddChainToMemoryStore");
|
|
|
|
// End Chain Building
|
|
|
|
error:
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
}
|
|
if (NULL != pccCA)
|
|
{
|
|
CertFreeCertificateContext(pccCA);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myCertServerExportPFX(
|
|
IN WCHAR const *pwszCA,
|
|
IN WCHAR const *pwszBackupDir,
|
|
IN WCHAR const *pwszPassword,
|
|
IN BOOL fEnhancedStrength,
|
|
IN BOOL fForceOverWrite,
|
|
IN BOOL fMustExportPrivateKeys,
|
|
IN DWORD dwmsTimeout,
|
|
OPTIONAL OUT WCHAR **ppwszPFXFile)
|
|
{
|
|
HRESULT hr;
|
|
HCERTSTORE hMyStore = NULL;
|
|
HCERTSTORE hTempMemoryStore = NULL;
|
|
CRYPT_DATA_BLOB pfx;
|
|
WCHAR *pwszPFXFile = NULL;
|
|
BOOL fImpersonating = FALSE;
|
|
WCHAR *pwszSanitizedCA = NULL;
|
|
WCHAR *pwszRevertCA = NULL;
|
|
DWORD cCACert;
|
|
DWORD cCACertSaved;
|
|
DWORD i;
|
|
|
|
pfx.pbData = NULL;
|
|
|
|
if (!ImpersonateSelf(SecurityImpersonation))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "ImpersonateSelf");
|
|
}
|
|
fImpersonating = TRUE;
|
|
|
|
hr = myEnablePrivilege(SE_BACKUP_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
|
|
if (NULL != ppwszPFXFile)
|
|
{
|
|
*ppwszPFXFile = NULL;
|
|
}
|
|
|
|
for (;;)
|
|
{
|
|
hr = mySanitizeName(pwszCA, &pwszSanitizedCA);
|
|
_JumpIfError(hr, error, "mySanitizeName");
|
|
|
|
// get CA cert count
|
|
hr = myGetCARegHashCount(pwszSanitizedCA, CSRH_CASIGCERT, &cCACert);
|
|
if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr &&
|
|
NULL == pwszRevertCA)
|
|
{
|
|
LocalFree(pwszSanitizedCA);
|
|
pwszSanitizedCA = NULL;
|
|
|
|
hr = myRevertSanitizeName(pwszCA, &pwszRevertCA);
|
|
_JumpIfError(hr, error, "myRevertSanitizeName");
|
|
|
|
pwszCA = pwszRevertCA;
|
|
continue;
|
|
}
|
|
_JumpIfError(hr, error, "myGetCARegHashCount");
|
|
|
|
if (NULL != pwszRevertCA)
|
|
{
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIB,
|
|
"myCertServerExportPFX called with Sanitized Name: %ws\n",
|
|
pwszSanitizedCA));
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (!myIsDirectory(pwszBackupDir))
|
|
{
|
|
if (!CreateDirectory(pwszBackupDir, NULL))
|
|
{
|
|
hr = myHLastError();
|
|
if (HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS) != hr)
|
|
{
|
|
_JumpError(hr, error, "CreateDirectory");
|
|
}
|
|
}
|
|
}
|
|
|
|
pwszPFXFile = (WCHAR *) LocalAlloc(
|
|
LMEM_FIXED,
|
|
(wcslen(pwszBackupDir) +
|
|
1 +
|
|
wcslen(pwszSanitizedCA) +
|
|
ARRAYSIZE(wszPFXFILENAMEEXT)) *
|
|
sizeof(WCHAR));
|
|
if (NULL == pwszPFXFile)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
wcscpy(pwszPFXFile, pwszBackupDir);
|
|
wcscat(pwszPFXFile, L"\\");
|
|
wcscat(pwszPFXFile, pwszSanitizedCA);
|
|
wcscat(pwszPFXFile, wszPFXFILENAMEEXT);
|
|
|
|
DBGPRINT((DBG_SS_CERTLIBI, "myCertServerExportPFX(%ws)\n", pwszPFXFile));
|
|
|
|
hMyStore = CertOpenStore(
|
|
CERT_STORE_PROV_SYSTEM_W,
|
|
X509_ASN_ENCODING,
|
|
NULL, // hProv
|
|
CERT_STORE_OPEN_EXISTING_FLAG |
|
|
CERT_STORE_ENUM_ARCHIVED_FLAG |
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE |
|
|
CERT_STORE_READONLY_FLAG,
|
|
wszMY_CERTSTORE);
|
|
if (NULL == hMyStore)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CertOpenStore");
|
|
}
|
|
|
|
hTempMemoryStore = CertOpenStore(
|
|
CERT_STORE_PROV_MEMORY,
|
|
X509_ASN_ENCODING,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
if (NULL == hTempMemoryStore)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CertOpenStore");
|
|
}
|
|
|
|
cCACertSaved = 0;
|
|
for (i = 0; i < cCACert; i++)
|
|
{
|
|
hr = SaveCACertChainToMemoryStore(
|
|
pwszSanitizedCA,
|
|
i,
|
|
hMyStore,
|
|
hTempMemoryStore,
|
|
dwmsTimeout);
|
|
_PrintIfError(hr, "SaveCACertChainToMemoryStore");
|
|
if (S_FALSE != hr)
|
|
{
|
|
_JumpIfError(hr, error, "SaveCACertChainToMemoryStore");
|
|
|
|
cCACertSaved++;
|
|
}
|
|
}
|
|
if (0 == cCACertSaved)
|
|
{
|
|
hr = CRYPT_E_NOT_FOUND;
|
|
_JumpError(hr, error, "SaveCACertChainToMemoryStore");
|
|
}
|
|
|
|
// done, have built entire chain for all CA Certs
|
|
|
|
// GemPlus returns NTE_BAD_TYPE instead of NTE_BAD_KEY, blowing up
|
|
// REPORT_NOT_ABLE* filtering. if they ever get this right, we can pass
|
|
// "[...] : EXPORT_PRIVATE_KEYS"
|
|
|
|
hr = myPFXExportCertStore(
|
|
hTempMemoryStore,
|
|
&pfx,
|
|
pwszPassword,
|
|
fEnhancedStrength,
|
|
fMustExportPrivateKeys?
|
|
(EXPORT_PRIVATE_KEYS | REPORT_NOT_ABLE_TO_EXPORT_PRIVATE_KEY) : 0);
|
|
_JumpIfError(hr, error, "myPFXExportCertStore");
|
|
|
|
hr = EncodeToFileW(
|
|
pwszPFXFile,
|
|
pfx.pbData,
|
|
pfx.cbData,
|
|
CRYPT_STRING_BINARY | (fForceOverWrite? DECF_FORCEOVERWRITE : 0));
|
|
_JumpIfError(hr, error, "EncodeToFileW");
|
|
|
|
if (NULL != ppwszPFXFile)
|
|
{
|
|
*ppwszPFXFile = pwszPFXFile;
|
|
pwszPFXFile = NULL;
|
|
}
|
|
|
|
error:
|
|
if (NULL != pwszSanitizedCA)
|
|
{
|
|
LocalFree(pwszSanitizedCA);
|
|
}
|
|
if (NULL != pwszRevertCA)
|
|
{
|
|
LocalFree(pwszRevertCA);
|
|
}
|
|
if (NULL != pwszPFXFile)
|
|
{
|
|
LocalFree(pwszPFXFile);
|
|
}
|
|
if (NULL != hMyStore)
|
|
{
|
|
CertCloseStore(hMyStore, CERT_CLOSE_STORE_CHECK_FLAG);
|
|
}
|
|
if (NULL != hTempMemoryStore)
|
|
{
|
|
CertCloseStore(hTempMemoryStore, CERT_CLOSE_STORE_CHECK_FLAG);
|
|
}
|
|
if (NULL != pfx.pbData)
|
|
{
|
|
LocalFree(pfx.pbData);
|
|
}
|
|
if (fImpersonating)
|
|
{
|
|
myEnablePrivilege(SE_BACKUP_NAME, FALSE);
|
|
RevertToSelf();
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
FindKeyUsage(
|
|
IN DWORD cExtension,
|
|
IN CERT_EXTENSION const *rgExtension,
|
|
OUT DWORD *pdwUsage)
|
|
{
|
|
HRESULT hr;
|
|
DWORD i;
|
|
CRYPT_BIT_BLOB *pblob = NULL;
|
|
|
|
*pdwUsage = 0;
|
|
for (i = 0; i < cExtension; i++)
|
|
{
|
|
CERT_EXTENSION const *pce;
|
|
|
|
pce = &rgExtension[i];
|
|
if (0 == strcmp(pce->pszObjId, szOID_KEY_USAGE))
|
|
{
|
|
DWORD cb;
|
|
|
|
// Decode CRYPT_BIT_BLOB:
|
|
|
|
if (!myDecodeObject(
|
|
X509_ASN_ENCODING,
|
|
X509_KEY_USAGE,
|
|
pce->Value.pbData,
|
|
pce->Value.cbData,
|
|
CERTLIB_USE_LOCALALLOC,
|
|
(VOID **) &pblob,
|
|
&cb))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "myDecodeObject");
|
|
}
|
|
if (1 > pblob->cbData || 8 < pblob->cUnusedBits)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
_JumpError(hr, error, "Key Usage Extension too small");
|
|
}
|
|
*pdwUsage = *pblob->pbData;
|
|
|
|
hr = S_OK;
|
|
goto error;
|
|
}
|
|
}
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpError(hr, error, "no Key Usage Extension");
|
|
|
|
error:
|
|
if (NULL != pblob)
|
|
{
|
|
LocalFree(pblob);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
mySetKeySpec(
|
|
IN CERT_CONTEXT const *pCert,
|
|
OUT DWORD *pdwKeySpec)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwKeyUsage;
|
|
|
|
*pdwKeySpec = AT_SIGNATURE;
|
|
hr = FindKeyUsage(
|
|
pCert->pCertInfo->cExtension,
|
|
pCert->pCertInfo->rgExtension,
|
|
&dwKeyUsage);
|
|
_JumpIfError(hr, error, "FindKeyUsage");
|
|
|
|
if (CERT_KEY_ENCIPHERMENT_KEY_USAGE & dwKeyUsage)
|
|
{
|
|
*pdwKeySpec = AT_KEYEXCHANGE;
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
|
|
// Ignore errors because the Key Usage extension may not exist:
|
|
hr = S_OK;
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myRepairKeyProviderInfo(
|
|
IN CERT_CONTEXT const *pCert,
|
|
IN BOOL fForceMachineKey,
|
|
IN OUT CRYPT_KEY_PROV_INFO *pkpi)
|
|
{
|
|
HRESULT hr;
|
|
BOOL fModified = FALSE;
|
|
|
|
if (0 == pkpi->dwProvType)
|
|
{
|
|
pkpi->dwProvType = PROV_RSA_FULL;
|
|
fModified = TRUE;
|
|
}
|
|
if (0 == pkpi->dwKeySpec)
|
|
{
|
|
hr = mySetKeySpec(pCert, &pkpi->dwKeySpec);
|
|
_JumpIfError(hr, error, "mySetKeySpec");
|
|
|
|
fModified = TRUE;
|
|
}
|
|
if (fForceMachineKey && 0 == (CRYPT_MACHINE_KEYSET & pkpi->dwFlags))
|
|
{
|
|
pkpi->dwFlags |= CRYPT_MACHINE_KEYSET;
|
|
fModified = TRUE;
|
|
}
|
|
if (fModified)
|
|
{
|
|
if (!CertSetCertificateContextProperty(
|
|
pCert,
|
|
CERT_KEY_PROV_INFO_PROP_ID,
|
|
0,
|
|
pkpi))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CertSetCertificateContextProperty");
|
|
}
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myRepairCertKeyProviderInfo(
|
|
IN CERT_CONTEXT const *pCert,
|
|
IN BOOL fForceMachineKey,
|
|
OPTIONAL OUT CRYPT_KEY_PROV_INFO **ppkpi)
|
|
{
|
|
HRESULT hr;
|
|
CRYPT_KEY_PROV_INFO *pkpi = NULL;
|
|
|
|
if (NULL != ppkpi)
|
|
{
|
|
*ppkpi = NULL;
|
|
}
|
|
|
|
hr = myCertGetKeyProviderInfo(pCert, &pkpi);
|
|
_JumpIfError2(hr, error, "myCertGetKeyProviderInfo", CRYPT_E_NOT_FOUND);
|
|
|
|
CSASSERT(NULL != pkpi);
|
|
|
|
hr = myRepairKeyProviderInfo(pCert, fForceMachineKey, pkpi);
|
|
_JumpIfError(hr, error, "myRepairKeyProviderInfo");
|
|
|
|
if (NULL != ppkpi)
|
|
{
|
|
*ppkpi = pkpi;
|
|
pkpi = NULL;
|
|
}
|
|
|
|
error:
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myGetChainArrayFromStore(
|
|
IN HCERTSTORE hStore,
|
|
IN BOOL fCAChain,
|
|
IN BOOL fUserStore,
|
|
OPTIONAL OUT WCHAR **ppwszCommonName,
|
|
IN OUT DWORD *pcRestoreChain,
|
|
OPTIONAL OUT RESTORECHAIN *paRestoreChain)
|
|
{
|
|
HRESULT hr;
|
|
CERT_CONTEXT const *pCert = NULL;
|
|
WCHAR *pwszCommonName = NULL;
|
|
CERT_CHAIN_PARA ChainParams;
|
|
CRYPT_KEY_PROV_INFO *pkpi = NULL;
|
|
DWORD iRestoreChain = 0;
|
|
|
|
if (NULL != ppwszCommonName)
|
|
{
|
|
*ppwszCommonName = NULL;
|
|
}
|
|
if (NULL != paRestoreChain)
|
|
{
|
|
ZeroMemory(paRestoreChain, *pcRestoreChain * sizeof(paRestoreChain[0]));
|
|
}
|
|
|
|
// Look for certificates with keys. There should be at least one.
|
|
|
|
for (;;)
|
|
{
|
|
BOOL fMatchingKey;
|
|
WCHAR *pwszCommonNameT;
|
|
CERT_CHAIN_CONTEXT const *pChain;
|
|
DWORD NameId;
|
|
|
|
pCert = CertEnumCertificatesInStore(hStore, pCert);
|
|
if (NULL == pCert)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
pkpi = NULL;
|
|
}
|
|
hr = myRepairCertKeyProviderInfo(pCert, !fUserStore, &pkpi);
|
|
if (S_OK != hr)
|
|
{
|
|
if (CRYPT_E_NOT_FOUND == hr)
|
|
{
|
|
continue;
|
|
}
|
|
_JumpError(hr, error, "myRepairCertKeyProviderInfo");
|
|
}
|
|
if (NULL == pkpi || NULL == pkpi->pwszContainerName)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
hr = myVerifyPublicKey(
|
|
pCert,
|
|
CERT_V1 == pCert->pCertInfo->dwVersion,
|
|
pkpi, // pKeyProvInfo
|
|
NULL, // pPublicKeyInfo
|
|
&fMatchingKey);
|
|
_JumpIfError(hr, error, "myVerifyPublicKey");
|
|
|
|
if (!fMatchingKey)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "Key doesn't match cert");
|
|
}
|
|
|
|
hr = myGetCertSubjectCommonName(pCert, &pwszCommonNameT);
|
|
_JumpIfError(hr, error, "myGetCertSubjectCommonName");
|
|
|
|
if (NULL == pwszCommonName)
|
|
{
|
|
pwszCommonName = pwszCommonNameT;
|
|
}
|
|
else
|
|
{
|
|
if (0 != lstrcmp(pwszCommonName, pwszCommonNameT))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_PrintErrorStr(hr, "first CommonName", pwszCommonName);
|
|
_JumpErrorStr(hr, error, "multiple CommonNames", pwszCommonNameT);
|
|
}
|
|
LocalFree(pwszCommonNameT);
|
|
}
|
|
if (fCAChain)
|
|
{
|
|
hr = myGetNameId(pCert, &NameId);
|
|
_PrintIfError(hr, "myGetNameId");
|
|
}
|
|
else
|
|
{
|
|
NameId = 0;
|
|
}
|
|
|
|
if (NULL != paRestoreChain && iRestoreChain >= *pcRestoreChain)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpError(hr, error, "Chain array full");
|
|
}
|
|
|
|
ZeroMemory(&ChainParams, sizeof(ChainParams));
|
|
ChainParams.cbSize = sizeof(ChainParams);
|
|
ChainParams.RequestedUsage.dwType = USAGE_MATCH_TYPE_AND;
|
|
|
|
// Get the chain and verify the cert:
|
|
|
|
if (!CertGetCertificateChain(
|
|
HCCE_LOCAL_MACHINE, // hChainEngine
|
|
pCert, // pCertContext
|
|
NULL, // pTime
|
|
hStore, // hAdditionalStore
|
|
&ChainParams, // pChainPara
|
|
0, // dwFlags
|
|
NULL, // pvReserved
|
|
&pChain)) // ppChainContext
|
|
{
|
|
hr = myHLastError();
|
|
_JumpIfError(hr, error, "CertGetCertificateChain");
|
|
}
|
|
if (NULL != paRestoreChain)
|
|
{
|
|
paRestoreChain[iRestoreChain].pChain = pChain;
|
|
paRestoreChain[iRestoreChain].NameId = NameId;
|
|
}
|
|
else
|
|
{
|
|
CertFreeCertificateChain(pChain);
|
|
}
|
|
iRestoreChain++;
|
|
}
|
|
if (NULL != ppwszCommonName)
|
|
{
|
|
*ppwszCommonName = pwszCommonName;
|
|
pwszCommonName = NULL;
|
|
}
|
|
*pcRestoreChain = iRestoreChain;
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (S_OK != hr && NULL != paRestoreChain)
|
|
{
|
|
for (iRestoreChain = 0; iRestoreChain < *pcRestoreChain; iRestoreChain++)
|
|
{
|
|
if (NULL != paRestoreChain[iRestoreChain].pChain)
|
|
{
|
|
CertFreeCertificateChain(paRestoreChain[iRestoreChain].pChain);
|
|
paRestoreChain[iRestoreChain].pChain = NULL;
|
|
}
|
|
}
|
|
}
|
|
if (NULL != pwszCommonName)
|
|
{
|
|
LocalFree(pwszCommonName);
|
|
}
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
}
|
|
if (NULL != pCert)
|
|
{
|
|
CertFreeCertificateContext(pCert);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myCopyKeys(
|
|
IN CRYPT_KEY_PROV_INFO const *pkpi,
|
|
IN WCHAR const *pwszOldContainer,
|
|
IN WCHAR const *pwszNewContainer,
|
|
IN WCHAR const *pwszNewCSP,
|
|
IN BOOL fOldUserKey,
|
|
IN BOOL fNewUserKey,
|
|
IN BOOL fNewProtect,
|
|
IN BOOL fForceOverWrite)
|
|
{
|
|
HRESULT hr;
|
|
HCRYPTPROV hProvOld = NULL;
|
|
HCRYPTKEY hKeyOld = NULL;
|
|
HCRYPTPROV hProvNew = NULL;
|
|
HCRYPTKEY hKeyNew = NULL;
|
|
CRYPT_BIT_BLOB PrivateKey;
|
|
BOOL fKeyContainerNotFound = FALSE;
|
|
|
|
ZeroMemory(&PrivateKey, sizeof(PrivateKey));
|
|
|
|
if (!myCertSrvCryptAcquireContext(
|
|
&hProvOld,
|
|
pwszOldContainer,
|
|
pkpi->pwszProvName,
|
|
pkpi->dwProvType,
|
|
pkpi->dwFlags,
|
|
!fOldUserKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "myCertSrvCryptAcquireContext");
|
|
}
|
|
if (!CryptGetUserKey(hProvOld, pkpi->dwKeySpec, &hKeyOld))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptGetUserKey");
|
|
}
|
|
hr = myCryptExportPrivateKey(
|
|
hKeyOld,
|
|
&PrivateKey.pbData,
|
|
&PrivateKey.cbData);
|
|
_JumpIfError(hr, error, "myCryptExportPrivateKey");
|
|
|
|
if (myCertSrvCryptAcquireContext(
|
|
&hProvNew,
|
|
pwszNewContainer,
|
|
pwszNewCSP,
|
|
pkpi->dwProvType,
|
|
pkpi->dwFlags,
|
|
!fNewUserKey))
|
|
{
|
|
if (!fForceOverWrite)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_EXISTS);
|
|
_JumpErrorStr(hr, error, "Key Container Exists", pwszNewContainer);
|
|
}
|
|
|
|
// Delete the target key container
|
|
|
|
CryptReleaseContext(hProvNew, 0);
|
|
if (myCertSrvCryptAcquireContext(
|
|
&hProvNew,
|
|
pwszNewContainer,
|
|
pwszNewCSP,
|
|
pkpi->dwProvType,
|
|
pkpi->dwFlags | CRYPT_DELETEKEYSET,
|
|
!fNewUserKey))
|
|
{
|
|
fKeyContainerNotFound = TRUE;
|
|
}
|
|
hProvNew = NULL;
|
|
}
|
|
else
|
|
{
|
|
fKeyContainerNotFound = TRUE;
|
|
}
|
|
|
|
if (!myCertSrvCryptAcquireContext(
|
|
&hProvNew,
|
|
pwszNewContainer,
|
|
pwszNewCSP,
|
|
pkpi->dwProvType,
|
|
pkpi->dwFlags |
|
|
(fKeyContainerNotFound? CRYPT_NEWKEYSET : 0),
|
|
!fNewUserKey))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "myCertSrvCryptAcquireContext");
|
|
}
|
|
|
|
if (!CryptImportKey(
|
|
hProvNew,
|
|
PrivateKey.pbData,
|
|
PrivateKey.cbData,
|
|
NULL, // HCRYPTKEY hPubKey
|
|
CRYPT_EXPORTABLE | (fNewProtect? CRYPT_USER_PROTECTED : 0),
|
|
&hKeyNew))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "CryptImportKey");
|
|
}
|
|
|
|
error:
|
|
if (NULL != PrivateKey.pbData)
|
|
{
|
|
SecureZeroMemory(PrivateKey.pbData, PrivateKey.cbData); // Key material
|
|
LocalFree(PrivateKey.pbData);
|
|
}
|
|
if (NULL != hKeyNew)
|
|
{
|
|
CryptDestroyKey(hKeyNew);
|
|
}
|
|
if (NULL != hProvNew)
|
|
{
|
|
CryptReleaseContext(hProvNew, 0);
|
|
}
|
|
if (NULL != hKeyOld)
|
|
{
|
|
CryptDestroyKey(hKeyOld);
|
|
}
|
|
if (NULL != hProvOld)
|
|
{
|
|
CryptReleaseContext(hProvOld, 0);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myImportChainAndKeys(
|
|
IN WCHAR const *pwszSanitizedCA,
|
|
IN DWORD iCert,
|
|
IN DWORD iKey,
|
|
IN BOOL fForceOverWrite,
|
|
IN CERT_CHAIN_CONTEXT const *pChain,
|
|
OPTIONAL OUT CERT_CONTEXT const **ppccNewestCA)
|
|
{
|
|
HRESULT hr;
|
|
CRYPT_KEY_PROV_INFO *pkpi = NULL;
|
|
CERT_CHAIN_ELEMENT **ppChainElement;
|
|
WCHAR *pwszKeyContainerName = NULL;
|
|
|
|
hr = myAllocIndexedName(
|
|
pwszSanitizedCA,
|
|
iKey,
|
|
MAXDWORD,
|
|
&pwszKeyContainerName);
|
|
_JumpIfError(hr, error, "myAllocIndexedName");
|
|
|
|
ppChainElement = pChain->rgpChain[0]->rgpElement;
|
|
|
|
hr = myCertGetKeyProviderInfo(ppChainElement[0]->pCertContext, &pkpi);
|
|
_JumpIfError(hr, error, "myCertGetKeyProviderInfo");
|
|
|
|
if (iCert == iKey)
|
|
{
|
|
hr = myCopyKeys(
|
|
pkpi,
|
|
pkpi->pwszContainerName, // pwszOldContainer
|
|
pwszKeyContainerName, // pwszNewContainer
|
|
pkpi->pwszProvName, // pwszNewCSP
|
|
FALSE, // fOldUserKey
|
|
FALSE, // fNewUserKey
|
|
FALSE, // fNewProtect
|
|
fForceOverWrite);
|
|
_JumpIfError(hr, error, "myCopyKeys");
|
|
}
|
|
|
|
pkpi->pwszContainerName = pwszKeyContainerName;
|
|
|
|
hr = mySaveChainAndKeys(
|
|
pChain->rgpChain[0],
|
|
wszMY_CERTSTORE,
|
|
CERT_SYSTEM_STORE_LOCAL_MACHINE |
|
|
CERT_STORE_BACKUP_RESTORE_FLAG,
|
|
pkpi,
|
|
ppccNewestCA);
|
|
_JumpIfError(hr, error, "mySaveChainAndKeys");
|
|
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
}
|
|
if (NULL != pwszKeyContainerName)
|
|
{
|
|
LocalFree(pwszKeyContainerName);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
FindPFXInBackupDir(
|
|
IN WCHAR const *pwszBackupDir,
|
|
OUT WCHAR **ppwszPFXFile)
|
|
{
|
|
HRESULT hr;
|
|
HANDLE hf;
|
|
WIN32_FIND_DATA wfd;
|
|
WCHAR wszpath[MAX_PATH];
|
|
WCHAR wszfile[MAX_PATH];
|
|
DWORD cFile = 0;
|
|
|
|
*ppwszPFXFile = NULL;
|
|
|
|
if (wcslen(pwszBackupDir) +
|
|
WSZARRAYSIZE(wszBSSTAR) +
|
|
WSZARRAYSIZE(wszPFXFILENAMEEXT) >= ARRAYSIZE(wszpath))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszBackupDir);
|
|
}
|
|
wcscpy(wszpath, pwszBackupDir);
|
|
wcscat(wszpath, wszBSSTAR);
|
|
wcscat(wszpath, wszPFXFILENAMEEXT);
|
|
|
|
hf = FindFirstFile(wszpath, &wfd);
|
|
if (INVALID_HANDLE_VALUE != hf)
|
|
{
|
|
do {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
|
{
|
|
continue;
|
|
}
|
|
cFile++;
|
|
if (wcslen(wfd.cFileName) >= ARRAYSIZE(wszfile))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpErrorStr(hr, error, "wszpath", pwszBackupDir);
|
|
}
|
|
wcscpy(wszfile, wfd.cFileName);
|
|
//printf("File: %ws\n", wszfile);
|
|
break;
|
|
|
|
} while (FindNextFile(hf, &wfd));
|
|
FindClose(hf);
|
|
}
|
|
if (0 == cFile)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpError(hr, error, "no *.p12 files");
|
|
}
|
|
if (1 < cFile)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_DIRECTORY);
|
|
_JumpError(hr, error, "Too many *.p12 files");
|
|
}
|
|
|
|
*ppwszPFXFile = (WCHAR *) LocalAlloc(
|
|
LMEM_FIXED,
|
|
(wcslen(pwszBackupDir) +
|
|
1 +
|
|
wcslen(wszfile) +
|
|
1) * sizeof(WCHAR));
|
|
if (NULL == *ppwszPFXFile)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
wcscpy(*ppwszPFXFile, pwszBackupDir);
|
|
wcscat(*ppwszPFXFile, L"\\");
|
|
wcscat(*ppwszPFXFile, wszfile);
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
// Return TRUE if pcc is newer than pcc2
|
|
|
|
BOOL
|
|
IsCACertNewer(
|
|
IN CERT_CONTEXT const *pcc,
|
|
IN DWORD NameId,
|
|
IN CERT_CONTEXT const *pcc2,
|
|
IN DWORD NameId2)
|
|
{
|
|
BOOL fNewer = FALSE;
|
|
CERT_INFO const *pci = pcc->pCertInfo;
|
|
CERT_INFO const *pci2 = pcc2->pCertInfo;
|
|
|
|
if (MAXDWORD != NameId && MAXDWORD != NameId2)
|
|
{
|
|
if (CANAMEIDTOICERT(NameId) > CANAMEIDTOICERT(NameId2))
|
|
{
|
|
fNewer = TRUE;
|
|
}
|
|
}
|
|
else
|
|
if (CompareFileTime(&pci->NotAfter, &pci2->NotAfter) > 0)
|
|
{
|
|
fNewer = TRUE;
|
|
}
|
|
|
|
#if 0
|
|
HRESULT hr;
|
|
WCHAR *pwszDate = NULL;
|
|
WCHAR *pwszDate2 = NULL;
|
|
|
|
hr = myGMTFileTimeToWszLocalTime(&pci->NotAfter, &pwszDate);
|
|
_PrintIfError(hr, "myGMTFileTimeToWszLocalTime");
|
|
|
|
hr = myGMTFileTimeToWszLocalTime(&pci2->NotAfter, &pwszDate2);
|
|
_PrintIfError(hr, "myGMTFileTimeToWszLocalTime");
|
|
|
|
printf(
|
|
"%u.%u %ws is %wsnewer than %u.%u %ws\n",
|
|
CANAMEIDTOICERT(NameId),
|
|
CANAMEIDTOIKEY(NameId),
|
|
pwszDate,
|
|
fNewer? L"" : L"NOT ",
|
|
CANAMEIDTOICERT(NameId2),
|
|
CANAMEIDTOIKEY(NameId2),
|
|
pwszDate2);
|
|
|
|
if (NULL != pwszDate) LocalFree(pwszDate);
|
|
if (NULL != pwszDate2) LocalFree(pwszDate2);
|
|
#endif
|
|
|
|
return(fNewer);
|
|
}
|
|
|
|
|
|
#if 0
|
|
VOID
|
|
DumpChainArray(
|
|
IN char const *psz,
|
|
IN DWORD cCACert,
|
|
IN OUT RESTORECHAIN *paRestoreChain)
|
|
{
|
|
HRESULT hr;
|
|
DWORD i;
|
|
|
|
printf("\n%hs:\n", psz);
|
|
for (i = 0; i < cCACert; i++)
|
|
{
|
|
WCHAR *pwszDate;
|
|
|
|
hr = myGMTFileTimeToWszLocalTime(
|
|
&paRestoreChain[i].pChain->rgpChain[0]->rgpElement[0]->pCertContext->pCertInfo->NotBefore,
|
|
&pwszDate);
|
|
_PrintIfError(hr, "myGMTFileTimeToWszLocalTime");
|
|
|
|
printf(
|
|
" %u: %u.%u %ws",
|
|
i,
|
|
CANAMEIDTOICERT(paRestoreChain[i].NameId),
|
|
CANAMEIDTOIKEY(paRestoreChain[i].NameId),
|
|
pwszDate);
|
|
|
|
|
|
if (NULL != pwszDate) LocalFree(pwszDate);
|
|
|
|
hr = myGMTFileTimeToWszLocalTime(
|
|
&paRestoreChain[i].pChain->rgpChain[0]->rgpElement[0]->pCertContext->pCertInfo->NotAfter,
|
|
&pwszDate);
|
|
_PrintIfError(hr, "myGMTFileTimeToWszLocalTime");
|
|
|
|
printf(" -- %ws\n", pwszDate);
|
|
|
|
if (NULL != pwszDate) LocalFree(pwszDate);
|
|
}
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
|
|
|
|
HRESULT
|
|
SortCACerts(
|
|
IN DWORD cCACert,
|
|
IN OUT RESTORECHAIN *paRestoreChain)
|
|
{
|
|
HRESULT hr;
|
|
DWORD i;
|
|
DWORD j;
|
|
|
|
#if 0
|
|
DumpChainArray("Start", cCACert, paRestoreChain);
|
|
#endif
|
|
|
|
for (i = 0; i < cCACert; i++)
|
|
{
|
|
for (j = i + 1; j < cCACert; j++)
|
|
{
|
|
CERT_CHAIN_CONTEXT const *pChain;
|
|
DWORD NameId;
|
|
DWORD NameId2;
|
|
CERT_CONTEXT const *pcc;
|
|
CERT_CONTEXT const *pcc2;
|
|
|
|
pChain = paRestoreChain[i].pChain;
|
|
NameId = paRestoreChain[i].NameId;
|
|
NameId2 = paRestoreChain[j].NameId;
|
|
|
|
pcc = pChain->rgpChain[0]->rgpElement[0]->pCertContext;
|
|
pcc2 = paRestoreChain[j].pChain->rgpChain[0]->rgpElement[0]->pCertContext;
|
|
|
|
#if 0
|
|
printf(
|
|
"%u(%u.%u) %u(%u.%u): ",
|
|
i,
|
|
CANAMEIDTOIKEY(NameId),
|
|
CANAMEIDTOICERT(NameId),
|
|
j,
|
|
CANAMEIDTOIKEY(NameId2),
|
|
CANAMEIDTOICERT(NameId2));
|
|
#endif
|
|
|
|
if (IsCACertNewer(pcc, NameId, pcc2, NameId2))
|
|
{
|
|
paRestoreChain[i] = paRestoreChain[j];
|
|
paRestoreChain[j].pChain = pChain;
|
|
paRestoreChain[j].NameId = NameId;
|
|
}
|
|
}
|
|
}
|
|
#if 0
|
|
DumpChainArray("End", cCACert, paRestoreChain);
|
|
#endif
|
|
|
|
hr = S_OK;
|
|
|
|
//error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
#define cwcGUIDKEYS 38
|
|
|
|
HRESULT
|
|
myDeleteGuidKeys(
|
|
IN HCERTSTORE hStorePFX,
|
|
IN BOOL fMachineKeySet)
|
|
{
|
|
HRESULT hr;
|
|
CERT_CONTEXT const *pCert = NULL;
|
|
CRYPT_KEY_PROV_INFO *pkpi = NULL;
|
|
|
|
// Look for certificates with keys, and delete all key containers with
|
|
// names that look like GUIDs.
|
|
|
|
for (;;)
|
|
{
|
|
HCRYPTPROV hProv;
|
|
|
|
pCert = CertEnumCertificatesInStore(hStorePFX, pCert);
|
|
if (NULL == pCert)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
pkpi = NULL;
|
|
}
|
|
hr = myRepairCertKeyProviderInfo(pCert, FALSE, &pkpi);
|
|
if (S_OK == hr &&
|
|
NULL != pkpi->pwszContainerName &&
|
|
wcLBRACE == pkpi->pwszContainerName[0] &&
|
|
cwcGUIDKEYS == wcslen(pkpi->pwszContainerName) &&
|
|
wcRBRACE == pkpi->pwszContainerName[cwcGUIDKEYS - 1])
|
|
{
|
|
if (myCertSrvCryptAcquireContext(
|
|
&hProv,
|
|
pkpi->pwszContainerName,
|
|
pkpi->pwszProvName,
|
|
pkpi->dwProvType,
|
|
pkpi->dwFlags | CRYPT_DELETEKEYSET,
|
|
fMachineKeySet))
|
|
{
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIBI,
|
|
"myDeleteGuidKeys(%ws, %ws)\n",
|
|
fMachineKeySet? L"Machine" : L"User",
|
|
pkpi->pwszContainerName));
|
|
}
|
|
}
|
|
}
|
|
hr = S_OK;
|
|
|
|
//error:
|
|
if (NULL != pkpi)
|
|
{
|
|
LocalFree(pkpi);
|
|
}
|
|
if (NULL != pCert)
|
|
{
|
|
CertFreeCertificateContext(pCert);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
myCertServerImportPFX(
|
|
IN WCHAR const *pwszBackupDirOrPFXFile,
|
|
IN WCHAR const *pwszPassword,
|
|
IN BOOL fForceOverWrite,
|
|
OPTIONAL OUT WCHAR **ppwszCommonName,
|
|
OPTIONAL OUT WCHAR **ppwszPFXFile,
|
|
OPTIONAL OUT CERT_CONTEXT const **ppccNewestCA)
|
|
{
|
|
HRESULT hr;
|
|
CRYPT_DATA_BLOB pfx;
|
|
HCERTSTORE hStorePFX = NULL;
|
|
WCHAR *pwszCommonName = NULL;
|
|
WCHAR *pwszSanitizedName = NULL;
|
|
RESTORECHAIN *paRestoreChain = NULL;
|
|
WCHAR *pwszPFXFile = NULL;
|
|
DWORD FileAttr;
|
|
BOOL fImpersonating = FALSE;
|
|
DWORD cCACert;
|
|
DWORD iCert;
|
|
|
|
pfx.pbData = NULL;
|
|
cCACert = 0;
|
|
|
|
if (NULL != ppwszCommonName)
|
|
{
|
|
*ppwszCommonName = NULL;
|
|
}
|
|
if (NULL != ppwszPFXFile)
|
|
{
|
|
*ppwszPFXFile = NULL;
|
|
}
|
|
if (NULL != ppccNewestCA)
|
|
{
|
|
*ppccNewestCA = NULL;
|
|
}
|
|
|
|
if (!ImpersonateSelf(SecurityImpersonation))
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "ImpersonateSelf");
|
|
}
|
|
fImpersonating = TRUE;
|
|
|
|
hr = myEnablePrivilege(SE_RESTORE_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
|
|
hr = myEnablePrivilege(SE_BACKUP_NAME, TRUE);
|
|
_JumpIfError(hr, error, "myEnablePrivilege");
|
|
|
|
FileAttr = GetFileAttributes(pwszBackupDirOrPFXFile);
|
|
if (MAXDWORD == FileAttr)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "GetFileAttributes");
|
|
}
|
|
|
|
if (FILE_ATTRIBUTE_DIRECTORY & FileAttr)
|
|
{
|
|
hr = FindPFXInBackupDir(pwszBackupDirOrPFXFile, &pwszPFXFile);
|
|
_JumpIfError(hr, error, "FindPFXInBackupDir");
|
|
}
|
|
else
|
|
{
|
|
hr = myDupString(pwszBackupDirOrPFXFile, &pwszPFXFile);
|
|
_JumpIfError(hr, error, "myDupString");
|
|
}
|
|
|
|
hr = DecodeFileW(pwszPFXFile, &pfx.pbData, &pfx.cbData, CRYPT_STRING_ANY);
|
|
_JumpIfError(hr, error, "DecodeFileW");
|
|
|
|
CSASSERT(NULL != pfx.pbData);
|
|
|
|
if (!PFXIsPFXBlob(&pfx))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "PFXIsPFXBlob");
|
|
}
|
|
|
|
hStorePFX = myPFXImportCertStore(
|
|
&pfx,
|
|
pwszPassword,
|
|
CRYPT_EXPORTABLE | CRYPT_MACHINE_KEYSET);
|
|
if (NULL == hStorePFX)
|
|
{
|
|
hr = myHLastError();
|
|
_JumpError(hr, error, "myPFXImportCertStore");
|
|
}
|
|
|
|
hr = myGetChainArrayFromStore(
|
|
hStorePFX,
|
|
TRUE, // fCAChain
|
|
FALSE, // fUserStore
|
|
&pwszCommonName,
|
|
&cCACert,
|
|
NULL);
|
|
_JumpIfError(hr, error, "myGetChainArrayFromStore");
|
|
|
|
if (0 == cCACert)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(CRYPT_E_SELF_SIGNED);
|
|
_JumpError(hr, error, "myGetChainArrayFromStore <no chain>");
|
|
}
|
|
|
|
paRestoreChain = (RESTORECHAIN *) LocalAlloc(
|
|
LMEM_FIXED | LMEM_ZEROINIT,
|
|
cCACert * sizeof(paRestoreChain[0]));
|
|
if (NULL == paRestoreChain)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
|
|
hr = myGetChainArrayFromStore(
|
|
hStorePFX,
|
|
TRUE, // fCAChain
|
|
FALSE, // fUserStore
|
|
NULL,
|
|
&cCACert,
|
|
paRestoreChain);
|
|
_JumpIfError(hr, error, "myGetChainArrayFromStore");
|
|
|
|
hr = SortCACerts(cCACert, paRestoreChain);
|
|
_JumpIfError(hr, error, "SortCACerts");
|
|
|
|
hr = mySanitizeName(pwszCommonName, &pwszSanitizedName);
|
|
_JumpIfError(hr, error, "mySanitizeName");
|
|
|
|
for (iCert = 0; iCert < cCACert; iCert++)
|
|
{
|
|
CERT_CHAIN_CONTEXT const *pChain = paRestoreChain[iCert].pChain;
|
|
DWORD iKey;
|
|
CERT_PUBLIC_KEY_INFO *pPublicKeyInfo;
|
|
WCHAR *pwszDN;
|
|
|
|
if (1 > pChain->cChain)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "No Chain Context");
|
|
}
|
|
|
|
// Compute iKey by comparing this public key to the public keys
|
|
// of all certs in the array already processed.
|
|
|
|
pPublicKeyInfo = &pChain->rgpChain[0]->rgpElement[0]->pCertContext->pCertInfo->SubjectPublicKeyInfo;
|
|
|
|
for (iKey = 0; iKey < iCert; iKey++)
|
|
{
|
|
if (CertComparePublicKeyInfo(
|
|
X509_ASN_ENCODING,
|
|
pPublicKeyInfo,
|
|
&paRestoreChain[iKey].pChain->rgpChain[0]->rgpElement[0]->pCertContext->pCertInfo->SubjectPublicKeyInfo))
|
|
{
|
|
// by design, CertComparePublicKeyInfo doesn't set last error!
|
|
|
|
break;
|
|
}
|
|
}
|
|
pwszDN = NULL;
|
|
hr = myCertNameToStr(
|
|
X509_ASN_ENCODING,
|
|
&pChain->rgpChain[0]->rgpElement[0]->pCertContext->pCertInfo->Subject,
|
|
CERT_X500_NAME_STR | CERT_NAME_STR_REVERSE_FLAG,
|
|
&pwszDN);
|
|
_PrintIfError(hr, "myCertNameToStr");
|
|
DBGPRINT((
|
|
DBG_SS_CERTLIB,
|
|
"Import: %u.%u -- %u.%u: %ws\n",
|
|
iCert,
|
|
iKey,
|
|
CANAMEIDTOICERT(paRestoreChain[iCert].NameId),
|
|
CANAMEIDTOIKEY(paRestoreChain[iCert].NameId),
|
|
pwszDN));
|
|
if (NULL != pwszDN)
|
|
{
|
|
LocalFree(pwszDN);
|
|
}
|
|
|
|
// Retrieve the cert context for the newest CA cert chain in the PFX
|
|
// we are importing. We must return a cert context with the new
|
|
// key prov info, not the PFX cert context with a GUID key container.
|
|
|
|
hr = myImportChainAndKeys(
|
|
pwszSanitizedName,
|
|
iCert,
|
|
iKey,
|
|
fForceOverWrite,
|
|
pChain,
|
|
iCert + 1 == cCACert? ppccNewestCA : NULL);
|
|
_JumpIfError(hr, error, "myImportChainAndKeys");
|
|
}
|
|
|
|
if (NULL != ppwszCommonName)
|
|
{
|
|
*ppwszCommonName = pwszCommonName;
|
|
pwszCommonName = NULL;
|
|
}
|
|
if (NULL != ppwszPFXFile)
|
|
{
|
|
*ppwszPFXFile = pwszPFXFile;
|
|
pwszPFXFile = NULL;
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
if (NULL != paRestoreChain)
|
|
{
|
|
for (iCert = 0; iCert < cCACert; iCert++)
|
|
{
|
|
if (NULL != paRestoreChain[iCert].pChain)
|
|
{
|
|
CertFreeCertificateChain(paRestoreChain[iCert].pChain);
|
|
}
|
|
}
|
|
LocalFree(paRestoreChain);
|
|
}
|
|
if (NULL != pwszPFXFile)
|
|
{
|
|
LocalFree(pwszPFXFile);
|
|
}
|
|
if (NULL != pwszCommonName)
|
|
{
|
|
LocalFree(pwszCommonName);
|
|
}
|
|
if (NULL != pwszSanitizedName)
|
|
{
|
|
LocalFree(pwszSanitizedName);
|
|
}
|
|
if (NULL != hStorePFX)
|
|
{
|
|
myDeleteGuidKeys(hStorePFX, TRUE);
|
|
CertCloseStore(hStorePFX, CERT_CLOSE_STORE_CHECK_FLAG);
|
|
}
|
|
if (NULL != pfx.pbData)
|
|
{
|
|
LocalFree(pfx.pbData);
|
|
}
|
|
if (fImpersonating)
|
|
{
|
|
myEnablePrivilege(SE_RESTORE_NAME, FALSE);
|
|
myEnablePrivilege(SE_BACKUP_NAME, FALSE);
|
|
RevertToSelf();
|
|
}
|
|
return(hr);
|
|
}
|