windows-nt/Source/XPSP1/NT/ds/security/csps/cryptoflex/slbcci/v1contrec.cpp
2020-09-26 16:20:57 +08:00

790 lines
22 KiB
C++

// V1ContRec.cpp -- definition of CV1ContainerRecord
// (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 "NoWarning.h"
#include <scuArrayP.h>
#include <slbCrc32.h>
#include <iopPubBlob.h>
#include <SmartCard.h>
#include "TransactionWrap.h"
#include "V1Paths.h"
#include "V1Card.h"
#include "V1ContRec.h"
using namespace std;
using namespace scu;
using namespace cci;
using namespace iop;
/////////////////////////// LOCAL/HELPER /////////////////////////////////
namespace
{
enum // KeyId in card
{
kidExchange = 0x00,
kidSignature = 0x01,
kidNone = 0xFF
};
TCHAR const
szCachedCertSignature[] = TEXT("CERTSI");
TCHAR const
szCachedCertExchange[] = TEXT("CERTEX");
TCHAR const
szCachedPublicKeySignature[] = TEXT("PUBKSI");
TCHAR const
szCachedPublicKeyExchange[] = TEXT("PUBKEX");
BYTE
AsKeyId(KeySpec ks)
{
BYTE kid;
switch (ks)
{
case ksExchange:
kid = kidExchange;
break;
case ksSignature:
kid = kidSignature;
break;
default:
throw cci::Exception(cci::ccBadKeySpec);
}
return kid;
}
} // namespace
/////////////////////////// PUBLIC /////////////////////////////////
// Types
// C'tors/D'tors
CV1ContainerRecord::CV1ContainerRecord(CV1Card const &rv1card,
string const &rsCntrType,
CreateMode mode)
: m_rcard(rv1card),
m_sCntrType(rsCntrType),
m_szKeyPath(0)
{
m_szKeyPath = IsDefault()
? CV1Paths::DefaultKey()
: CV1Paths::DefaultContainer();
switch (mode)
{
case cmNever:
if (!Exists())
throw Exception(ccInvalidParameter);
break;
case cmConditionally:
if (!Exists())
Create();
break;
case cmAlways:
if (!Exists())
Create();
else
throw Exception(ccOutOfSymbolTableEntries);
break;
case cmNoCheck:
break;
default:
throw Exception(ccInvalidParameter);
break;
}
}
CV1ContainerRecord::~CV1ContainerRecord()
{}
// Operators
// Operations
string
CV1ContainerRecord::ComputeSignature(KeySpec ks,
string const &rsCipher) const
{
CTransactionWrap wrap(m_rcard);
m_rcard.SmartCard().Select(m_szKeyPath);
AutoArrayPtr<BYTE> aabBuffer(new BYTE[rsCipher.length()]);
m_rcard.SmartCard().InternalAuth(ktRSA1024,
AsKeyId(ks),
static_cast<BYTE>(rsCipher.length()),
reinterpret_cast<BYTE const *>(rsCipher.data()),
aabBuffer.Get());
return string(reinterpret_cast<char *>(aabBuffer.Get()),
rsCipher.length());
}
void
CV1ContainerRecord::Delete() const
{
CTransactionWrap wrap(m_rcard);
// if (IsEmpty())
// throw scu::OsException(NTE_BAD_KEYSET_PARAM);
// Open the container file and find the offset of the container
DWORD dwFileSize = OpenContainer();
DWORD dwOffset = 0x00;
DWORD dwLen = FindOffset(dwOffset);
// Actually check the existence of key container
if (sizeof ContainerBuffer > dwLen)
throw scu::OsException(NTE_BAD_KEYSET);
// Intialize search variables
DWORD dwNext = dwOffset + dwLen;
// Get following ContainerBuffer
ContainerBuffer container;
GetContainer(dwNext, container);
dwLen = container.Size;
// Move all following blocks up to deleted block position
while (sizeof container <= dwLen)
{
basic_string<BYTE> bsBuffer(reinterpret_cast<BYTE *>(&container),
sizeof container);
WORD cPublicKeysLength = dwLen - sizeof container;
AutoArrayPtr<BYTE> aabPublicKeys(new BYTE[cPublicKeysLength]);
if (cPublicKeysLength > 0)
{
m_rcard.SmartCard().ReadBinary(dwNext + sizeof container,
cPublicKeysLength,
aabPublicKeys.Get());
}
bsBuffer.append(aabPublicKeys.Get(), cPublicKeysLength);
m_rcard.SmartCard().WriteBinary(dwOffset,
static_cast<WORD>(bsBuffer.length()),
bsBuffer.data());
dwOffset += dwLen;
dwNext += dwLen;
GetContainer(dwNext, container);
dwLen = container.Size;
}; // end while loop
// NO MORE CONTAINERS TO MOVE UP
// if there is still room put 2 null bytes of termination
const BYTE NullSize[]= {0x00, 0x00};
if ((dwOffset + 2) <= dwFileSize)
m_rcard.SmartCard().WriteBinary(dwOffset, 2, NullSize);
}
void
CV1ContainerRecord::Name(string const &rsNewName)
{
m_rcard.CardId(rsNewName);
}
void
CV1ContainerRecord::Read(KeySpec ks,
CPublicKeyBlob &rKeyBlob) const
{
CTransactionWrap wrap(m_rcard);
if ((ksSignature != ks) && (ksExchange != ks))
throw Exception(ccBadKeySpec);
string sBuffer;
DWORD dwExponent;
if (GetContainerContent(ks, sBuffer, dwExponent))
{
CopyMemory(rKeyBlob.bModulus, sBuffer.data(),
sBuffer.length());
rKeyBlob.bModulusLength = static_cast<BYTE>(sBuffer.length());
*reinterpret_cast<DWORD *>(rKeyBlob.bExponent) =
dwExponent;
}
else
rKeyBlob.bModulusLength = 0;
}
void
CV1ContainerRecord::Read(KeySpec ks,
string &rsBlob) const
{
CTransactionWrap wrap(m_rcard);
DWORD dwOriginalCrc = 0;
// Get blob from the container
if (!GetContainerContent(ks,
rsBlob,
dwOriginalCrc))
throw Exception(ccNoCertificate);
// If a non-zero CRC exists, then verify integrity of the
// compressed certificate by comparing the CRC read
// (original) against a test one generated using the
// compressed certificate read. If the CRCs aren't equal,
// then the certificate is corrupted and it shouldn't be
// decompressed because the decompress routine may go into
// an infinite loop or otherwise fail badly without
// notification. If the original CRC is zero, then a CRC
// wasn't performed so for backward compatibility with
// earlier versions the decompression is taken with the
// inherent risk.
if (0 != dwOriginalCrc)
{
DWORD dwTestCrc = Crc32(rsBlob.data(), rsBlob.length());
if (dwTestCrc != dwOriginalCrc)
throw Exception(ccSymbolDataCorrupted);
}
}
void
CV1ContainerRecord::Write(KeySpec ks,
CPrivateKeyBlob const &rKeyBlob)
{
CTransactionWrap wrap(m_rcard);
m_rcard.SmartCard().Select(CV1Paths::PrivateKeys());
// Make sure that previous key blocks exists in Secret Key file
// or at least the header of the block exists
// in order for the Card OS to be able to retrieve the key
// that is added in this procedure:
// Write the header of previous keys
WORD const wPrivateKeyBlockSize = 323;
// inversion necessary for PRIVATE KEY BLOC SIZE
WORD wBSize = (wPrivateKeyBlockSize >> 8) & 0x00FF;
wBSize += (wPrivateKeyBlockSize << 8) & 0x00FF00;
BYTE bId;
DWORD dwOffset;
BYTE bKeyId = AsKeyId(ks);
for (dwOffset = 0x00, bId = 0;
bId < bKeyId;
bId++, dwOffset += wPrivateKeyBlockSize)
{
BYTE Header[3];
CopyMemory(Header, &wBSize, sizeof WORD);
Header[2] = bId + 1;
m_rcard.SmartCard().WriteBinary(dwOffset, 3, Header);
}
m_rcard.SmartCard().WritePrivateKey(rKeyBlob, bKeyId);
}
void
CV1ContainerRecord::Write(KeySpec ks,
CPublicKeyBlob const &rKeyBlob)
{
CTransactionWrap wrap(m_rcard);
DWORD dwExponent = *(reinterpret_cast<DWORD const *>(rKeyBlob.bExponent));
Write(ks, reinterpret_cast<BYTE const *>(rKeyBlob.bModulus),
rKeyBlob.bModulusLength, dwExponent);
}
void
CV1ContainerRecord::Write(KeySpec ks,
string const &rsBlob) const
{
CTransactionWrap wrap(m_rcard);
// Calculate the CRC to verify when reading reading the
// blob back.
DWORD dwCrc = 0;
if (rsBlob.length())
dwCrc = Crc32(rsBlob.data(), rsBlob.length());
Write(ks, reinterpret_cast<BYTE const *>(rsBlob.data()),
static_cast<WORD>(rsBlob.length()), dwCrc);
}
// Access
string
CV1ContainerRecord::CertName()
{
static string const sCertContainerName("CERT");
return sCertContainerName;
}
string
CV1ContainerRecord::DefaultName()
{
static string const sDefaultName("USER");
return sDefaultName;
}
string
CV1ContainerRecord::Name() const
{
return m_rcard.CardId();
}
// Predicates
bool
CV1ContainerRecord::Exists() const
{
CTransactionWrap wrap(m_rcard);
DWORD dwLen = 0;
try
{
if (m_rcard.CardId() == m_sCntrType)
return true;
DWORD dwOffset = 0x00;
dwLen = FindOffset(dwOffset);
}
catch (iop::CSmartCard::Exception &)
{
}
return (sizeof ContainerBuffer <= dwLen);
}
bool
CV1ContainerRecord::KeyExists(KeySpec ks) const
{
CTransactionWrap wrap(m_rcard);
bool fExists = false;
//
// Does a key of this type exist in this container?
// Note: assumes that m_KeyPath is set to correct container path?
//
// Open the container file
DWORD dwFileSize = OpenContainer();
DWORD dwOffset = 0x00;
DWORD const dwLen = FindOffset(dwOffset);
//
// Actually check the existence of key container
// by seeing if we have a record of the right size
//
ContainerBuffer container;
if (sizeof container <= dwLen)
{
GetContainer(dwOffset, container);
//
// Check which key exists by checking lengths
//
switch (ks)
{
case ksExchange:
if (0x00 < container.XK_wLen)
fExists = true;
break;
case ksSignature:
if (0x00 < container.SK_wLen)
fExists = true;
break;
}
}
return fExists;
}
// Static Variables
/////////////////////////// PROTECTED /////////////////////////////////
// C'tors/D'tors
// Operators
// Operations
// Access
// Predicates
// Static Variables
/////////////////////////// PRIVATE /////////////////////////////////
// C'tors/D'tors
// Operators
// Operations
void
CV1ContainerRecord::Create() const
{
// Open the file and find the offset to the container
DWORD dwFileSize = OpenContainer();
DWORD dwOffset = 0x00;
DWORD dwLen = FindOffset(dwOffset);
// Actually check the existence of key container
if (sizeof ContainerBuffer <= dwLen)
throw scu::OsException(NTE_EXISTS);
// Set the new the container management data
dwLen = SetContainer(dwOffset);
// if there is still room put 2 null bytes of termination
if ((dwOffset + dwLen + 2) <= dwFileSize)
{
const BYTE NullSize[] = { 0x00, 0x00 };
m_rcard.SmartCard().WriteBinary(dwOffset + dwLen,
sizeof NullSize, NullSize);
}
}
DWORD
CV1ContainerRecord::FindOffset(DWORD &rdwOffset) const
{
DWORD dwFileSize = OpenContainer();
if ((rdwOffset + sizeof ContainerBuffer) > dwFileSize)
return 0x00;
bool fFound = false;
DWORD dwLen = sizeof ContainerBuffer; // arbitrary value to start
size_t const cBufferSize =
sizeof WORD + (sizeof BYTE *
ContainerBuffer::cMaxContainerNameLength) + 1;
// +1 allows null terminator
AutoArrayPtr<BYTE> aabBuffer(new BYTE[cBufferSize]);
while (!fFound &&
(0x00 < dwLen) &&
((rdwOffset + sizeof ContainerBuffer) <= dwFileSize))
{
m_rcard.SmartCard().ReadBinary(rdwOffset,
cBufferSize - 1,
aabBuffer.Get());
WORD const *pwLen = reinterpret_cast<WORD *>(aabBuffer.Get());
dwLen = *pwLen;
aabBuffer[cBufferSize - 1] = 0x00; // ensure null terminate string
string sName(reinterpret_cast<char *>(&aabBuffer[sizeof WORD]));
if ((m_sCntrType == sName) && (0x00 < dwLen))
fFound = true;
else
rdwOffset += dwLen;
}
if (fFound)
return (dwLen & 0x00FFFF);
else
return 0x00;
}
void
CV1ContainerRecord::GetContainer(DWORD dwOffset,
ContainerBuffer &rcontainer) const
{
bool fClearContainer = true;
try
{
DWORD dwFileSize = OpenContainer();
if ((dwOffset + sizeof rcontainer) <= dwFileSize)
{
m_rcard.SmartCard().ReadBinary(dwOffset, sizeof rcontainer,
reinterpret_cast<BYTE *>(&rcontainer));
fClearContainer = false;
}
}
catch (...)
{
}
if (fClearContainer)
{
rcontainer.Size = 0x00;
rcontainer.Name[0] = '\0';
}
}
bool
CV1ContainerRecord::GetContainerContent(KeySpec ks,
string &rsBuffer,
DWORD &rdwExponent) const
{
bool fExists = false;
OpenContainer();
DWORD dwOffset = 0x00;
if (0x00 != FindOffset(dwOffset))
{
fExists = true;
ContainerBuffer container;
GetContainer(dwOffset, container);
DWORD dwKeyLength = 0;
AutoArrayPtr<BYTE> aabKey;
if (ksExchange == ks)
{
if (0x00 < container.XK_wLen)
{
rdwExponent = container.XK_dwExp;
dwKeyLength = container.XK_wLen;
aabKey = AutoArrayPtr<BYTE>(new BYTE[container.XK_wLen]);
m_rcard.SmartCard().ReadBinary(dwOffset + sizeof container,
container.XK_wLen,
aabKey.Get());
}
}
else
{
if (0x00 < container.SK_wLen)
{
rdwExponent = container.SK_dwExp;
dwKeyLength = container.SK_wLen;
aabKey = AutoArrayPtr<BYTE>(new BYTE[container.SK_wLen]);
m_rcard.SmartCard().ReadBinary(dwOffset +
sizeof container +
container.XK_wLen,
container.SK_wLen,
aabKey.Get());
}
}
if (aabKey.Get())
rsBuffer.assign(reinterpret_cast<char *>(aabKey.Get()),
dwKeyLength);
}
return fExists;
}
DWORD
CV1ContainerRecord::OpenContainer() const
{
DWORD dwFileSize;
string sPath(m_szKeyPath);
sPath.append("/");
sPath.append(CV1Paths::RelativeContainers());
dwFileSize = m_rcard.OpenFile(sPath.c_str());
return dwFileSize;
}
DWORD
CV1ContainerRecord::SetContainer(DWORD dwOffset) const
{
DWORD dwFileSize;
dwFileSize = OpenContainer();
if ((dwOffset + sizeof ContainerBuffer) > dwFileSize)
throw Exception(ccOutOfSymbolTableSpace);
// Create the container buffer
ContainerBuffer container;
ZeroMemory(&container, sizeof container);
container.Size = sizeof container;
CopyMemory(container.Name, m_sCntrType.data(), m_sCntrType.length());
m_rcard.SmartCard().WriteBinary(dwOffset, sizeof container,
reinterpret_cast<BYTE *>(&container));
return container.Size;
}
void
CV1ContainerRecord::Write(KeySpec ks,
BYTE const *pbModulus,
WORD wModulusLength,
DWORD dwExponent) const
{
// Open container, get the data
DWORD dwFileSize = OpenContainer();
DWORD dwOffset = 0x00;
DWORD dwLen = FindOffset(dwOffset);
ContainerBuffer container;
GetContainer(dwOffset, container);
// Check which key exists
AutoArrayPtr<BYTE> aabXKey(new BYTE[container.XK_wLen]);
if (0x00 < container.XK_wLen)
m_rcard.SmartCard().ReadBinary(dwOffset + sizeof container,
container.XK_wLen,
aabXKey.Get());
AutoArrayPtr<BYTE> aabSKey(new BYTE[container.SK_wLen]);
if (0x00 < container.SK_wLen)
m_rcard.SmartCard().ReadBinary(dwOffset + sizeof container +
container.XK_wLen,
container.SK_wLen,
aabSKey.Get());
// Give an arbitrary value if key spec not specified
if ((ksSignature != ks) && (ksExchange != ks))
{
if (0x00 == container.XK_wLen)
ks = ksExchange;
else
{
if (0x00 == container.SK_wLen)
ks = ksSignature;
else
throw Exception(ccBadKeySpec);
}
}
// Is it the last container of Container file?
ContainerBuffer NextContainer;
GetContainer(dwOffset + dwLen, NextContainer);
bool fDeleted = false;
if (sizeof NextContainer <= NextContainer.Size)
{
// Delete the existing container
Delete();
fDeleted = true;
// No need to recreate it now
}
// Now the container is at the end of the Container file
// Find the "NEW" offset of the container which may not exist anymore
dwOffset = 0x00;
FindOffset(dwOffset); // keep the INITIAL dwLen of the container
// Check that there is enough room to put the new key
bool fEnoughMemory = false;
switch (ks)
{
case ksExchange:
if ((dwOffset + dwLen - container.XK_wLen +
wModulusLength) <= dwFileSize)
{
aabXKey = AutoArrayPtr<BYTE>(new BYTE[wModulusLength]);
CopyMemory(aabXKey.Get(), pbModulus, wModulusLength);
container.XK_dwExp = dwExponent;
container.XK_wLen = wModulusLength;
fEnoughMemory = true;
}
break;
case ksSignature:
if ((dwOffset + dwLen - container.SK_wLen +
wModulusLength) <= dwFileSize)
{
aabSKey = AutoArrayPtr<BYTE>(new BYTE[wModulusLength]);
CopyMemory(aabSKey.Get(), pbModulus, wModulusLength);
container.SK_dwExp = dwExponent;
container.SK_wLen = wModulusLength;
fEnoughMemory = true;
}
break;
}
// Recreate the container buffer accounting for "card tearing"
// where the card could be pulled during the write operation.
// This is done using a type of transact and commit phases.
// The container size is initially set to zero, then the container
// contents are written (transaction), followed by resetting the
// container size to the actual length of the container to
// "commit" the changes to the card.
container.Size = 0;
DWORD const dwTrueSize = sizeof container + container.XK_wLen +
container.SK_wLen;
size_t cBufferSize = dwTrueSize;
BYTE const abNull[] = {0x00,0x00};
bool fAppendNull = (dwTrueSize + sizeof abNull) <= dwFileSize;
if (fAppendNull)
cBufferSize += sizeof abNull;
AutoArrayPtr<BYTE> aabBuffer(new BYTE[cBufferSize]);
BYTE *pbBuffer = aabBuffer.Get();
CopyMemory(pbBuffer, &container, sizeof container);
pbBuffer += sizeof container;
CopyMemory(pbBuffer, aabXKey.Get(), container.XK_wLen);
pbBuffer += container.XK_wLen;
CopyMemory(pbBuffer, aabSKey.Get(), container.SK_wLen);
pbBuffer += container.SK_wLen;
if (fAppendNull)
{
CopyMemory(pbBuffer, abNull, sizeof abNull);
pbBuffer += sizeof abNull;
}
// Rewrite the container even if there is not enough to write the
// NEW public key, then there should be enough room to write the
// existing key.
m_rcard.SmartCard().WriteBinary(dwOffset,
pbBuffer - aabBuffer.Get(),
aabBuffer.Get());
// Now commit these changes with the actual size.
container.Size = dwTrueSize;
m_rcard.SmartCard().WriteBinary(dwOffset, sizeof container.Size,
reinterpret_cast<BYTE *>(&container));
if (!fEnoughMemory)
throw Exception(ccOutOfSymbolTableSpace);
}
// Access
// Predicates
bool
CV1ContainerRecord::IsDefault() const
{
return (DefaultName() == m_sCntrType);
}
// Static Variables