Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

761 lines
23 KiB

// iop.cpp -- Definition of CIOP
// (c) Copyright Schlumberger Technology Corp., unpublished work, created
// 2000. This computer program includes Confidential, Proprietary
// Information and is a Trade Secret of Schlumberger Technology Corp. All
// use, disclosure, and/or reproduction is prohibited unless authorized
// in writing. All Rights Reserved.
#include <tchar.h>
#include <string>
#include <scuOsExc.h>
#include <scuOsVersion.h>
#include <scuArrayP.h>
#include "iop.h"
#include <aclapi.h>
#include "LockWrap.h"
using namespace std;
namespace
{
char g_szSLBRegistryPath[] = "SOFTWARE\\Schlumberger";
char g_szTerminalsName[] = "Smart Cards and Terminals";
char g_szCardName[] = "Smart Cards";
char g_szCrypto4KName[] = "Cryptoflex 4K";
char g_szOldCrypto8KName[] = "Cryptoflex 8K (no RSA key generation)";
char g_szNewCrypto8KName[] = "Cryptoflex 8K (with RSA key generation)";
char g_szCrypto8KV2Name[] = "Cryptoflex 8K (V2)";
char g_szAccessName[] = "Cyberflex Access 16K";
char g_sze_gateName[] = "Schlumberger Cryptoflex e-gate";
char g_szCrypto16KName[] = "Cryptoflex 16K";
char g_szAccessCampus[] = "Schlumberger Cyberflex Access Campus";
char g_szCryptoActivCard[] = "Schlumberger Cryptoflex ActivCard";
string
CardPath()
{
static string sPath = string(g_szSLBRegistryPath) +
string("\\") + string(g_szTerminalsName) + string("\\") +
string(g_szCardName);
return sPath;
}
#if defined(SLBIOP_WAIT_FOR_RM_STARTUP)
HANDLE GetSCResourceManagerStartedEvent(void)
{
typedef HANDLE (*LPCALAISACCESSEVENT)(void);
HANDLE hReturn = NULL;
try
{
HMODULE hWinScard = GetModuleHandle(TEXT("WINSCARD.DLL"));
if (NULL != hWinScard)
{
LPCALAISACCESSEVENT pfCalais =
(LPCALAISACCESSEVENT)GetProcAddress(hWinScard,
"SCardAccessStartedEvent");
if (NULL != pfCalais)
{
hReturn = (*pfCalais)();
}
}
}
catch (...)
{
hReturn = NULL;
}
return hReturn;
}
#endif // defined(SLBIOP_WAIT_FOR_RM_STARTUP)
}
namespace iop
{
CIOP::CIOP()
: m_hContext(NULL)
{
// Ensure that resorce manager is running, then Establish context
if (!CIOP::WaitForSCManager())
throw Exception(ccResourceManagerDisabled);
HRESULT hResult = SCardEstablishContext(SCARD_SCOPE_SYSTEM, NULL, NULL, &m_hContext);
if (SCARD_S_SUCCESS != hResult)
throw scu::OsException(hResult);
}
CIOP::~CIOP()
{
SCardReleaseContext(m_hContext);
}
CSmartCard *
CIOP::Connect(const char* szReaderName,
bool fExclusiveMode)
{
HRESULT hResult = NOERROR;
DWORD dwShare = (fExclusiveMode ? SCARD_SHARE_EXCLUSIVE : SCARD_SHARE_SHARED);
DWORD dwProtocol;
SCARDHANDLE hCard;
// Grab our Mutex. This is a hack around an RM bug.
CIOPLock TempLock(szReaderName); // This is ok as long as one do not try to do SCard locking
CIOPMutex tempMutex(&TempLock);
// Connect to the reader
hResult = SCardConnect(m_hContext, szReaderName, dwShare,
SCARD_PROTOCOL_T0, &hCard, &dwProtocol);
if (hResult != SCARD_S_SUCCESS)
throw scu::OsException(hResult);
// Get the ATR and determine card type
DWORD dwBufferLen = 0;
DWORD dwState;
BYTE bATR[CSmartCard::cMaxAtrLength];
DWORD dwATRLen = sizeof bATR / sizeof *bATR;
hResult = SCardStatus(hCard,NULL, &dwBufferLen, &dwState,
&dwProtocol, bATR, &dwATRLen);
if (hResult != SCARD_S_SUCCESS)
throw scu::OsException(hResult);
// Create a SmartCard of the right type.
CSmartCard *psc = CreateCard(bATR, dwATRLen, hCard, szReaderName, dwShare);
return psc;
}
// This function creates a smart card of the appropriate type
CSmartCard * CIOP::CreateCard(const BYTE* bATR,
const DWORD dwLength,
const SCARDHANDLE hCard,
const char* szReaderName,
const DWORD dwShareMode)
{
////////////////////////////////////
// Open path to registered keys //
////////////////////////////////////
HKEY hkCardKey;
HKEY hkTestKey;
RegOpenKeyEx(HKEY_LOCAL_MACHINE, CardPath().c_str(), NULL,
KEY_READ, &hkCardKey);
//////////////////////////////////////////////
// Enumerate subkeys to find an ATR match //
//////////////////////////////////////////////
FILETIME fileTime;
char szATR[] = "ATR";
char szMask[] = "ATR Mask";
char szCardType[] = "Card Type";
char sBuffer[MAX_PATH + 1];
BYTE bATRtest[CSmartCard::cMaxAtrLength];
BYTE bMask[CSmartCard::cMaxAtrLength];
BYTE type;
char szCardName[MAX_PATH + 1];
DWORD dwBufferSize = sizeof(sBuffer);
DWORD dwATRSize = sizeof bATRtest / sizeof *bATRtest;
DWORD dwMaskSize = sizeof bMask / sizeof *bMask;
DWORD dwTypeSize = 1;
DWORD index = 0;
LONG iRetVal = RegEnumKeyEx(hkCardKey, index, sBuffer,
&dwBufferSize, NULL, NULL, NULL,
&fileTime);
while (iRetVal == ERROR_SUCCESS)
{
strcpy(szCardName, sBuffer);
RegOpenKeyEx(hkCardKey, sBuffer, NULL, KEY_READ, &hkTestKey);
RegQueryValueEx(hkTestKey, szATR, NULL, NULL, bATRtest, &dwATRSize);
RegQueryValueEx(hkTestKey, szMask, NULL, NULL, bMask, &dwMaskSize);
RegQueryValueEx(hkTestKey, szCardType, NULL, NULL, &type, &dwTypeSize);
if (dwATRSize == dwLength)
{
scu::AutoArrayPtr<BYTE> aabMaskedATR(new BYTE[dwATRSize]);
for (DWORD count = 0; count < dwATRSize; count++)
aabMaskedATR[count] = bATR[count] & bMask[count];
if (!memcmp(aabMaskedATR.Get(), bATRtest, dwATRSize))
break;
}
index++;
dwBufferSize = sizeof(sBuffer);
dwATRSize = sizeof bATRtest / sizeof *bATRtest;
dwMaskSize = sizeof bMask / sizeof *bMask;
RegCloseKey(hkTestKey);
iRetVal = RegEnumKeyEx(hkCardKey, index, sBuffer, &dwBufferSize, NULL, NULL, NULL, &fileTime);
}
// if loop was broken, iRetVal is still ERROR_SUCCESS, and type holds correct card to use
CSmartCard *retVal = NULL;
if (iRetVal == ERROR_SUCCESS)
{
switch (type)
{
case CRYPTO_CARD: retVal = new CCryptoCard(hCard,
szReaderName,
m_hContext,
dwShareMode);
break;
case ACCESS_CARD: retVal = new CAccessCard(hCard,
szReaderName,
m_hContext,
dwShareMode);
break;
default: throw Exception(ccUnknownCard);
break;
}
}
// loop wasn't broken, i.e., ATR not found. Try to make an Access Card.
else
retVal = new CAccessCard(hCard, szReaderName, m_hContext,
dwShareMode);
retVal->setCardName(szCardName);
return retVal;
}
void
CIOP::ListReaders(char* szReadersList, int &iSizeOfList)
{
DWORD dwSize = static_cast<DWORD>(iSizeOfList);
LONG lRet;
lRet = SCardListReaders(m_hContext, NULL, szReadersList, &dwSize);
iSizeOfList = static_cast<int>(dwSize);
if (SCARD_S_SUCCESS != lRet)
throw scu::OsException(lRet);
}
void
CIOP::ListKnownCards(char* szCardList, int& iSizeOfList)
{
////////////////////////////////////
// Open path to registered keys //
////////////////////////////////////
LONG rv;
HKEY hkCardKey;
rv = RegOpenKeyEx(HKEY_LOCAL_MACHINE, CardPath().c_str(), NULL,
KEY_READ, &hkCardKey);
if(ERROR_SUCCESS != rv)
throw scu::OsException(rv);
///////////////////////////////////////////
// Enumerate subkeys to get card names //
///////////////////////////////////////////
FILETIME fileTime;
char sBuffer[1024];
DWORD dwBufferSize = sizeof sBuffer / sizeof *sBuffer;
int iTotalSize = 0;
int index = 0;
memset(sBuffer, 0, dwBufferSize);
scu::AutoArrayPtr<char> aaszCardListBuffer(new char[iSizeOfList]);
memset(aaszCardListBuffer.Get(), 0, iSizeOfList);
rv = RegEnumKeyEx(hkCardKey, index++, sBuffer, &dwBufferSize,
NULL, NULL, NULL, &fileTime);
while (rv == ERROR_SUCCESS)
{
if (iTotalSize + dwBufferSize <= iSizeOfList - 2) // spare two chars for trailing nulls
{
strcpy((aaszCardListBuffer.Get() + iTotalSize), sBuffer);
iTotalSize += dwBufferSize;
aaszCardListBuffer[iTotalSize++] = 0;
}
else
{
iTotalSize += dwBufferSize + 1;
}
dwBufferSize = sizeof sBuffer / sizeof *sBuffer;
memset(sBuffer, 0, dwBufferSize);
rv = RegEnumKeyEx(hkCardKey, index++, sBuffer, &dwBufferSize,
NULL, NULL, NULL, &fileTime);
}
bool fRetVal = (iTotalSize <= iSizeOfList - 1); // spare byte for final null terminator
if (fRetVal)
{
aaszCardListBuffer[iTotalSize++] = 0;
memcpy(szCardList, aaszCardListBuffer.Get(), iTotalSize);
}
else
iTotalSize++; // spare byte for final null terminator
iSizeOfList = iTotalSize;
rv = RegCloseKey(hkCardKey);
if (ERROR_SUCCESS != rv)
throw scu::OsException(rv);
}
void
CIOP::RegisterCard(const char* szCardName,
const BYTE* bATR,
BYTE bATRLength,
const BYTE* bATRMask,
BYTE bATRMaskLength,
const BYTE* bProperties,
cardType type)
{
HKEY hkCardKey;
DWORD dwCreateFlag;
BYTE bCardType = (BYTE)type;
char szATR[] = "ATR";
char szATRMask[] = "ATR Mask";
char szCardType[] = "Card Type";
char szProperties[] = "Properties";
string sCardRegPath(CardPath());
sCardRegPath.append("\\");
sCardRegPath.append(szCardName);
LONG rv = RegCreateKeyEx(HKEY_LOCAL_MACHINE,
sCardRegPath.c_str(), NULL, NULL, NULL,
KEY_ALL_ACCESS, NULL, &hkCardKey, &dwCreateFlag);
if(ERROR_SUCCESS!=rv)
throw scu::OsException(rv);
if (dwCreateFlag == REG_CREATED_NEW_KEY)
{
rv = RegSetValueEx(hkCardKey, szATR, NULL, REG_BINARY, bATR,
bATRLength);
if (ERROR_SUCCESS==rv)
{
rv = RegSetValueEx(hkCardKey, szATRMask, NULL,
REG_BINARY, bATRMask,
bATRMaskLength);
if (ERROR_SUCCESS==rv)
{
rv = RegSetValueEx(hkCardKey, szCardType, NULL,
REG_BINARY, &bCardType, 1);
if (ERROR_SUCCESS==rv)
rv = RegSetValueEx(hkCardKey, szProperties, NULL,
REG_BINARY, bProperties, 512);
}
}
}
LONG rv2 = RegCloseKey(hkCardKey);
if (ERROR_SUCCESS!=rv) // an error occured earlier
throw scu::OsException(rv);
if (ERROR_SUCCESS != rv2)
throw scu::OsException(rv2);
// return (dwCreateFlag == REG_CREATED_NEW_KEY);
}
void
CIOP::RegisterDefaultCards()
{
BYTE bMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
BYTE bAccessMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00 };
BYTE bCMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00 };
BYTE bMaskLength = 9;
BYTE bATRLength = 9;
BYTE bProperties[512];
memset(bProperties, 0, sizeof(bProperties));
// Register Cryptoflex 16K
BYTE b16KCryptoATR[] = { 0x3B, 0x95, 0x15, 0x40, 0xFF, 0x63,
0x01, 0x01, 0x00, 0x00 };
BYTE b16KCryptoMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0x00, 0x00 };
RegisterCard(g_szCrypto16KName, b16KCryptoATR,
sizeof b16KCryptoATR / sizeof *b16KCryptoATR, b16KCryptoMask,
sizeof b16KCryptoMask / sizeof *b16KCryptoMask,
bProperties, CRYPTO_CARD);
// Register e-gate
BYTE be_gateATR[] = { 0x3B, 0x95, 0x00, 0x40, 0xFF, 0x62,
0x01, 0x01, 0x00, 0x00 };
BYTE be_gateMask[] = { 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0x00, 0x00 };
RegisterCard(g_sze_gateName, be_gateATR,
sizeof be_gateATR / sizeof *be_gateATR, be_gateMask,
sizeof be_gateMask / sizeof *be_gateMask,
bProperties, CRYPTO_CARD);
// Register Cyberflex Access card
BYTE bAccessATR[] = { 0x3B, 0x16, 0x94, 0x81, 0x10, 0x06, 0x01, 0x00, 0x00 };
RegisterCard(g_szAccessName, bAccessATR, bATRLength, bAccessMask,
bMaskLength, bProperties, ACCESS_CARD);
// Register old Cryptoflex 8K card
BYTE bOldCryptoATR[] = { 0x3B, 0x85, 0x40, 0x20, 0x68, 0x01, 0x01, 0x00, 0x00 };
RegisterCard(g_szOldCrypto8KName, bOldCryptoATR, bATRLength, bMask,
bMaskLength, bProperties, CRYPTO_CARD);
// Register new Cryptoflex 8K card
BYTE bNewCryptoATR[] = { 0x3B, 0x85, 0x40, 0x20, 0x68, 0x01, 0x01, 0x05, 0x01 };
RegisterCard(g_szNewCrypto8KName, bNewCryptoATR, bATRLength, bMask,
bMaskLength, bProperties, CRYPTO_CARD);
// Register another new Cryptoflex 8K card
BYTE bCrypto8KV2ATR[] = { 0x3B, 0x95, 0x15, 0x40, 0x00, 0x68, 0x01, 0x02, 0x00, 0x00 };
BYTE bCrypto8KV2Mask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0xFF, 0x00, 0x00 };
RegisterCard(g_szCrypto8KV2Name, bCrypto8KV2ATR, sizeof(bCrypto8KV2ATR), bCrypto8KV2Mask,
sizeof(bCrypto8KV2Mask), bProperties, CRYPTO_CARD);
// Register Cryptoflex 4K card
BYTE b4KCryptoATR[] = { 0x3B, 0xE2, 0x00, 0x00, 0x40, 0x20, 0x49, 0x00 };
bATRLength = 8;
bMaskLength = 8;
RegisterCard(g_szCrypto4KName, b4KCryptoATR, bATRLength, bCMask,
bMaskLength, bProperties, CRYPTO_CARD);
// Register Cyberflex Access Campus
BYTE be_AccessCampusATR[] = { 0x3B, 0x23, 0x00, 0x35, 0x13, 0x80 };
BYTE be_AccessCampusMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
RegisterCard(g_szAccessCampus, be_AccessCampusATR,
sizeof be_AccessCampusATR / sizeof *be_AccessCampusATR,
be_AccessCampusMask,
sizeof be_AccessCampusMask / sizeof *be_AccessCampusMask,
bProperties, ACCESS_CARD);
// Register Cryptoflex ActivCard
BYTE bCryptoActivCardATR[] = { 0x3B, 0x05, 0x68, 0x01, 0x01,
0x02, 0x05 };
BYTE bCryptoActivCardMask[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF };
RegisterCard(g_szCryptoActivCard, bCryptoActivCardATR,
sizeof(bCryptoActivCardATR), bCryptoActivCardMask,
sizeof(bCryptoActivCardMask), bProperties, CRYPTO_CARD);
}
#if defined(SLBIOP_USE_SECURITY_ATTRIBUTES)
void
CIOP::InitIOPSecurityAttrs(CSecurityAttributes *psa)
{
DWORD dwRes;
PSID pEveryoneSID = NULL, pAdminSID = NULL;
PACL pACL = NULL;
PSECURITY_DESCRIPTOR pSD = NULL;
EXPLICIT_ACCESS ea;
SID_IDENTIFIER_AUTHORITY SIDAuthWorld = SECURITY_WORLD_SID_AUTHORITY;
SID_IDENTIFIER_AUTHORITY SIDAuthNT = SECURITY_NT_AUTHORITY;
bool fErrorFound = false;
// Create a well-known SID for the Everyone group.
if(!AllocateAndInitializeSid(&SIDAuthWorld, 1,
SECURITY_WORLD_RID,
0, 0, 0, 0, 0, 0, 0, &pEveryoneSID))
throw scu::OsException(GetLastError());
// Initialize an EXPLICIT_ACCESS structure for an ACE.
// The ACE will allow Everyone read access to the key.
ZeroMemory(&ea, sizeof(EXPLICIT_ACCESS));
ea.grfAccessPermissions = SPECIFIC_RIGHTS_ALL | STANDARD_RIGHTS_ALL;
ea.grfAccessMode = SET_ACCESS;
ea.grfInheritance= NO_INHERITANCE;
ea.Trustee.TrusteeForm = TRUSTEE_IS_SID;
ea.Trustee.TrusteeType = TRUSTEE_IS_WELL_KNOWN_GROUP;
ea.Trustee.ptstrName = (LPTSTR) pEveryoneSID;
#if 0
// Create a SID for the BUILTIN\Administrators group.
if (!AllocateAndInitializeSid(&SIDAuthNT, 2,
SECURITY_BUILTIN_DOMAIN_RID,
DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0,
0, 0, &pAdminSID))
throw scu::OsException(GetLastError());
// Initialize an EXPLICIT_ACCESS structure for an ACE.
// The ACE will allow the Administrators group full access to the key.
ea.grfAccessPermissions = SPECIFIC_RIGHTS_ALL | STANDARD_RIGHTS_ALL;
ea.grfAccessMode = SET_ACCESS;
ea.grfInheritance= NO_INHERITANCE;
ea.Trustee.TrusteeForm = TRUSTEE_IS_SID;
ea.Trustee.TrusteeType = TRUSTEE_IS_GROUP;
ea.Trustee.ptstrName = (LPTSTR) pAdminSID;
#endif // 0
// Create a new ACL that contains the new ACEs.
dwRes = SetEntriesInAcl(1, &ea, NULL, &pACL);
if (ERROR_SUCCESS != dwRes)
{
fErrorFound = true;
}
else
{
// Initialize a security descriptor.
pSD = (PSECURITY_DESCRIPTOR) LocalAlloc(LPTR, SECURITY_DESCRIPTOR_MIN_LENGTH);
if (pSD == NULL)
{
fErrorFound = true;
}
else if (!InitializeSecurityDescriptor(pSD, SECURITY_DESCRIPTOR_REVISION))
{
fErrorFound = true;
}
// Add the ACL to the security descriptor.
else if (!SetSecurityDescriptorDacl(pSD,
TRUE, // fDaclPresent flag
pACL,
FALSE)) // not a default DACL
{
fErrorFound = true;
}
else
{
if (!IsValidSecurityDescriptor(pSD))
{
fErrorFound = true;
}
else
{
// Initialize a security attributes structure.
psa->sa.nLength = sizeof(SECURITY_ATTRIBUTES);
psa->sa.lpSecurityDescriptor = pSD;
psa->sa.bInheritHandle = FALSE;
psa->pEveryoneSID = pEveryoneSID;
psa->pACL = pACL;
}
}
}
DWORD dwLastError = GetLastError();
if (true == fErrorFound)
{
if (NULL != pACL)
{
LocalFree(pACL);
pACL = NULL;
}
if (NULL != pSD)
{
LocalFree(pSD);
pSD = NULL;
}
if (NULL != pEveryoneSID)
{
FreeSid(pEveryoneSID);
pEveryoneSID = NULL;
}
throw scu::OsException(dwLastError);
}
#if 0
// Create a new ACL that contains the new ACEs.
dwRes = SetEntriesInAcl(1, &ea, NULL, &pACL);
if (ERROR_SUCCESS != dwRes)
throw scu::OsException(GetLastError());
// Initialize a security descriptor.
pSD = (PSECURITY_DESCRIPTOR) LocalAlloc(LPTR, SECURITY_DESCRIPTOR_MIN_LENGTH);
if (pSD == NULL)
throw scu::OsException(GetLastError());
if (!InitializeSecurityDescriptor(pSD, SECURITY_DESCRIPTOR_REVISION))
throw scu::OsException(GetLastError());
// Add the ACL to the security descriptor.
if (!SetSecurityDescriptorDacl(pSD,
TRUE, // fDaclPresent flag
pACL,
FALSE)) // not a default DACL
throw scu::OsException(GetLastError());
// Initialize a security attributes structure.
psa->nLength = sizeof(SECURITY_ATTRIBUTES);
psa->lpSecurityDescriptor = pSD;
psa->bInheritHandle = FALSE;
if (!IsValidSecurityDescriptor(pSD))
throw scu::OsException(GetLastError());
#endif
}
#endif // defined(SLBIOP_USE_SECURITY_ATTRIBUTES)
bool CIOP::WaitForSCManager()
{
#if defined(SLBIOP_WAIT_FOR_RM_STARTUP)
// Wait for the SCManager to start, time out at dwTimeout seconds.
HANDLE hStarted = GetSCResourceManagerStartedEvent();
if (hStarted)
{
if (WaitForSingleObject(hStarted, 60 * 1000) == WAIT_OBJECT_0)
return true;
}
return false;
#else // defined(SLBIOP_WAIT_FOR_RM_STARTUP)
return true;
#endif // defined(SLBIOP_WAIT_FOR_RM_STARTUP)
}
} // namespace iop
STDAPI DllGetVersion(DLLVERSIONINFO *dvi)
{
dvi->dwBuildNumber = 0;
dvi->dwMajorVersion = 0;
dvi->dwMinorVersion = 9;
return 0;
}
STDAPI DllRegisterServer()
{
// Ensure default cards are registered to the system
HRESULT hResult = ERROR_SUCCESS;
try
{
iop::CIOP::RegisterDefaultCards();
}
catch (scu::OsException const &rExc)
{
hResult = rExc.Cause();
}
return hResult;
}
STDAPI DllUnregisterServer()
{
HRESULT hResult = NOERROR;
LONG rv;
HKEY hkSLBKey;
HKEY hkTerminalsKey;
HKEY hkCardsKey;
bool bSLBKey = false, bTerminalsKey = false, bCardsKey = false;
try
{
rv = RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szSLBRegistryPath, NULL, KEY_ALL_ACCESS, &hkSLBKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bSLBKey = true;
RegOpenKeyEx(hkSLBKey, g_szTerminalsName, NULL, KEY_ALL_ACCESS, &hkTerminalsKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bTerminalsKey = true;
RegOpenKeyEx(hkTerminalsKey, g_szCardName, NULL, KEY_ALL_ACCESS, &hkCardsKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bCardsKey = true;
rv = RegDeleteKey(hkCardsKey, g_szCrypto4KName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szOldCrypto8KName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szNewCrypto8KName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szCrypto8KV2Name);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szAccessName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_sze_gateName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szCrypto16KName);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szAccessCampus);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegDeleteKey(hkCardsKey, g_szCryptoActivCard);
if(rv!=ERROR_SUCCESS) hResult = E_UNEXPECTED;
rv = RegCloseKey (hkCardsKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bCardsKey = false;
rv = RegDeleteKey(hkTerminalsKey, g_szCardName);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bCardsKey = false;
rv = RegCloseKey (hkTerminalsKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bTerminalsKey = false;
rv = RegDeleteKey(hkSLBKey, g_szTerminalsName);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bCardsKey = false;
rv = RegCloseKey(hkSLBKey);
if(rv!=ERROR_SUCCESS) throw scu::OsException(rv);
bSLBKey = false;
}
catch(...)
{
hResult = E_UNEXPECTED;
}
if(bCardsKey) RegCloseKey (hkCardsKey);
if(bTerminalsKey) RegCloseKey (hkTerminalsKey);
if(bSLBKey) RegCloseKey (hkSLBKey);
return hResult;
}