mirror of https://github.com/tongzx/nt5src
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.
1214 lines
35 KiB
1214 lines
35 KiB
// CSmartCard.cpp: implementation of the CSmartCard class.
|
|
//
|
|
// (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 <string>
|
|
#include <wtypes.h>
|
|
|
|
#include <scuOsExc.h>
|
|
#include <scuExcHelp.h>
|
|
#include <scuArrayP.h>
|
|
|
|
#include "iopExc.h"
|
|
#include "SmartCard.h"
|
|
#include "LockWrap.h"
|
|
|
|
using namespace std;
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Construction/Destruction
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////// Begin CSmartCard::Exception //////////////////////////
|
|
|
|
/////////////////////////// LOCAL/HELPER /////////////////////////////////
|
|
namespace
|
|
{
|
|
using namespace iop;
|
|
|
|
// Transfer data to/from the card. The data to read/write is
|
|
// broken up into cMaxBlock sized chunks when calling rsc's
|
|
// (CSmartCard) member procedure pbtm (pointer to member routine)
|
|
// to actually do the read/write.
|
|
template<class BlockType, class PBlockTransferMember>
|
|
void
|
|
TransferData(const WORD wOffset, // file offset to start
|
|
const WORD wDataLength, // amount to transfer
|
|
BlockType bData, // data buffer to read/write
|
|
CSmartCard &rsc, // card object to use
|
|
PBlockTransferMember pbtm, // member proc to read/write
|
|
WORD cMaxBlock) // max size transfer at once
|
|
{
|
|
WORD wBytesTransferred = 0;
|
|
WORD wFullRounds = wDataLength / cMaxBlock;
|
|
|
|
/////////////////////////////////////////////////
|
|
// Transfer maximum bytes at each full round //
|
|
/////////////////////////////////////////////////
|
|
for (WORD wCurrentRound = 0; wCurrentRound < wFullRounds; wCurrentRound++)
|
|
{
|
|
(rsc.*pbtm)(wOffset + wBytesTransferred,
|
|
bData + wBytesTransferred, cMaxBlock);
|
|
|
|
wBytesTransferred += cMaxBlock;
|
|
}
|
|
|
|
// Transfer any leftovers
|
|
BYTE bBytesLeft = wDataLength - wBytesTransferred;
|
|
if (bBytesLeft != 0)
|
|
{
|
|
(rsc.*pbtm)(wOffset + wBytesTransferred,
|
|
bData + wBytesTransferred, bBytesLeft);
|
|
}
|
|
}
|
|
|
|
scu::CauseCodeDescriptionTable<CSmartCard::CauseCode> ccdt[] =
|
|
{
|
|
{
|
|
CSmartCard::ccAccessConditionsNotMet,
|
|
TEXT("Access conditions not met.")
|
|
},
|
|
{
|
|
CSmartCard::ccAlgorithmIdNotSupported,
|
|
TEXT("The algorithm ID is not supported in the card.")
|
|
},
|
|
{
|
|
CSmartCard::ccAskRandomNotLastApdu,
|
|
TEXT("The random number is no longer available. The "
|
|
"AskRandom APDU must be sent immediately previous "
|
|
"to this one.")
|
|
},
|
|
{
|
|
CSmartCard::ccAuthenticationFailed,
|
|
TEXT("Authentication failed (i.e. CHV or key rejected, or "
|
|
"wrong cryptogram).")
|
|
},
|
|
{
|
|
CSmartCard::ccBadFilePath,
|
|
TEXT("The file path is invalid. Ensure that each file "
|
|
"name in the path is 4 characters long and is a valid "
|
|
"representation of the hexadecimal ID.")
|
|
},
|
|
{
|
|
CSmartCard::ccBadState,
|
|
TEXT("The Application is not in a state permitting this "
|
|
"operation.")
|
|
},
|
|
{
|
|
CSmartCard::ccCannotReadOutsideFileBoundaries,
|
|
TEXT("Could not read outside the file boundaries.")
|
|
},
|
|
{
|
|
CSmartCard::ccCannotWriteOutsideFileBoundaries,
|
|
TEXT("Could not write outside the file boundaries.")
|
|
},
|
|
{
|
|
CSmartCard::ccCardletNotInRegisteredState,
|
|
TEXT("Cardlet is not in a registered state. It may be "
|
|
"blocked or not completely installed.")
|
|
},
|
|
{
|
|
CSmartCard::ccChvNotInitialized,
|
|
TEXT("No CHV is initialzed.")
|
|
},
|
|
{
|
|
CSmartCard::ccChvVerificationFailedMoreAttempts,
|
|
TEXT("CHV verification was unsuccessful, at least one "
|
|
"attempt remains.")
|
|
},
|
|
{
|
|
CSmartCard::ccContradictionWithInvalidationStatus,
|
|
TEXT("Contradiction with invalidation status occured.")
|
|
},
|
|
{
|
|
CSmartCard::ccCurrentDirectoryIsNotSelected,
|
|
TEXT("The current directory is not selected.")
|
|
},
|
|
{
|
|
CSmartCard::ccDataPossiblyCorrupted,
|
|
TEXT("Data possibly corrupted.")
|
|
|
|
},
|
|
{
|
|
CSmartCard::ccDefaultLoaderNotSelected,
|
|
TEXT("Cardlet is currently selected and install cannot "
|
|
"run. Default loader application must be selected.")
|
|
},
|
|
{
|
|
CSmartCard::ccDirectoryNotEmpty,
|
|
TEXT("This directory still contains other files or "
|
|
"directories and may not be deleted.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileAlreadyInvalidated,
|
|
TEXT("File already invalidated.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileExists,
|
|
TEXT("The file ID requested is already in use.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileIdExistsOrTypeInconsistentOrRecordTooLong,
|
|
TEXT("Either the file ID already exists in the current "
|
|
"directory, the file type is inconsisent with the "
|
|
"command or the record length is too long.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileIndexDoesNotExist,
|
|
TEXT("The file index passed does not exist in the current "
|
|
"directory.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileInvalidated,
|
|
TEXT("The command attempted to operate on an invalidated "
|
|
"file.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileNotFound,
|
|
TEXT("The file requested for this operation was not "
|
|
"found.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileNotFoundOrNoMoreFilesInDf,
|
|
TEXT("The file specified was not found or no more files in "
|
|
"the current DF.")
|
|
},
|
|
{
|
|
CSmartCard::ccFileTypeInvalid,
|
|
TEXT("File type is invalid.")
|
|
},
|
|
{
|
|
CSmartCard::ccIncorrectP1P2,
|
|
TEXT("Incorrect parameter P1 or P2.")
|
|
},
|
|
{
|
|
CSmartCard::ccIncorrectP3,
|
|
TEXT("Incorrect P3.")
|
|
},
|
|
{
|
|
CSmartCard::ccInstallCannotRun,
|
|
TEXT("Cardlet is currently selected and install cannot "
|
|
"run. Default loader application must be selected.")
|
|
},
|
|
{
|
|
CSmartCard::ccInstanceIdInUse,
|
|
TEXT("Instance ID is being used by another file.")
|
|
},
|
|
{
|
|
CSmartCard::ccInsufficientSpace,
|
|
TEXT("Insufficient space available.")
|
|
},
|
|
{
|
|
CSmartCard::ccInvalidAnswerReceived,
|
|
TEXT("Invalid answer received from the card.")
|
|
},
|
|
{
|
|
CSmartCard::ccInvalidKey,
|
|
TEXT("CHV verification was unsuccessful; at least one "
|
|
"attempt remains.")
|
|
},
|
|
{
|
|
CSmartCard::ccInvalidSignature,
|
|
TEXT("Signature is invalid.")
|
|
},
|
|
{
|
|
CSmartCard::ccJava,
|
|
TEXT("Applet exception occured.")
|
|
},
|
|
{
|
|
CSmartCard::ccKeyBlocked,
|
|
TEXT("Key is blocked. No attempts remain.")
|
|
},
|
|
{
|
|
CSmartCard::ccLimitReached,
|
|
TEXT("Limit has been reached. Additional value would "
|
|
"exceed the record's limit.")
|
|
},
|
|
{
|
|
CSmartCard::ccMemoryProblem,
|
|
TEXT("Memory problem occured.")
|
|
},
|
|
{
|
|
CSmartCard::ccNoAccess,
|
|
TEXT("Access conditions not met.")
|
|
},
|
|
{
|
|
CSmartCard::ccNoEfSelected,
|
|
TEXT("No elementary file selected.")
|
|
},
|
|
{
|
|
CSmartCard::ccNoEfExistsOrNoChvKeyDefined,
|
|
TEXT("No EF exists, or no CHV or key defined.")
|
|
},
|
|
{
|
|
CSmartCard::ccNoFileSelected,
|
|
TEXT("No elementary file selected.")
|
|
},
|
|
{
|
|
CSmartCard::ccNoGetChallengeBefore,
|
|
TEXT("A Get Challenge was not performed before this "
|
|
"operation.")
|
|
},
|
|
{
|
|
CSmartCard::ccOperationNotActivatedForApdu,
|
|
TEXT("Algorithm is supported, but the operation is not "
|
|
"activated for this APDU.")
|
|
},
|
|
{
|
|
CSmartCard::ccOutOfRangeOrRecordNotFound,
|
|
TEXT("Out of range or record not found.")
|
|
},
|
|
{
|
|
CSmartCard::ccOutOfSpaceToCreateFile,
|
|
TEXT("Not enough space is available to create the file.")
|
|
},
|
|
{
|
|
CSmartCard::ccProgramFileInvalidated,
|
|
TEXT("Program file invalidated.")
|
|
},
|
|
{
|
|
CSmartCard::ccRecordInfoIncompatible,
|
|
TEXT("Record information is incompatible with the file "
|
|
"size.")
|
|
},
|
|
{
|
|
CSmartCard::ccRecordLengthTooLong,
|
|
TEXT("Record length is too long.")
|
|
},
|
|
{
|
|
CSmartCard::ccRequestedAlgIdMayNotMatchKeyUse,
|
|
TEXT("The requested algorithm ID may not match the key "
|
|
"used.")
|
|
},
|
|
{
|
|
CSmartCard::ccReturnedDataCorrupted,
|
|
TEXT("Data return from the card is corrupted.")
|
|
},
|
|
{
|
|
CSmartCard::ccRootDirectoryNotErasable,
|
|
TEXT("It is not valid to delete the root directory.")
|
|
},
|
|
{
|
|
CSmartCard::ccTimeOut,
|
|
TEXT("Time-out occured.")
|
|
},
|
|
{
|
|
CSmartCard::ccTooMuchDataForProMode,
|
|
TEXT("Too much data for PRO mode.")
|
|
},
|
|
{
|
|
CSmartCard::ccUnknownInstructionClass,
|
|
TEXT("Unknown instruction class.")
|
|
},
|
|
{
|
|
CSmartCard::ccUnknownInstructionCode,
|
|
TEXT("Unknown instruction code.")
|
|
},
|
|
{
|
|
CSmartCard::ccUnknownStatus,
|
|
TEXT("An unknown error status was returned from the "
|
|
"card.")
|
|
},
|
|
{
|
|
CSmartCard::ccUnidentifiedTechnicalProblem,
|
|
TEXT("Unidentified technical problem.")
|
|
},
|
|
{
|
|
CSmartCard::ccUpdateImpossible,
|
|
TEXT("Update is impossible.")
|
|
},
|
|
{
|
|
CSmartCard::ccVerificationFailed,
|
|
TEXT("Verification failed.")
|
|
},
|
|
};
|
|
|
|
} // namespace
|
|
|
|
namespace iop
|
|
{
|
|
|
|
|
|
/////////////////////////// PUBLIC /////////////////////////////////
|
|
|
|
// Types
|
|
// C'tors/D'tors
|
|
CSmartCard::Exception::Exception(CauseCode cc,
|
|
ClassByte cb,
|
|
Instruction ins,
|
|
StatusWord sw) throw()
|
|
: scu::Exception(scu::Exception::fcSmartCard),
|
|
m_cc(cc),
|
|
m_cb(cb),
|
|
m_ins(ins),
|
|
m_sw(sw)
|
|
{}
|
|
|
|
CSmartCard::Exception::~Exception()
|
|
{}
|
|
|
|
|
|
// Operators
|
|
// Operations
|
|
scu::Exception *
|
|
CSmartCard::Exception::Clone() const
|
|
{
|
|
return new CSmartCard::Exception(*this);
|
|
}
|
|
|
|
void
|
|
CSmartCard::Exception::Raise() const
|
|
{
|
|
throw *this;
|
|
}
|
|
|
|
// Access
|
|
CSmartCard::Exception::CauseCode
|
|
CSmartCard::Exception::Cause() const throw()
|
|
{
|
|
return m_cc;
|
|
}
|
|
|
|
CSmartCard::ClassByte
|
|
CSmartCard::Exception::Class() const throw()
|
|
{
|
|
return m_cb;
|
|
}
|
|
|
|
char const *
|
|
CSmartCard::Exception::Description() const
|
|
{
|
|
return scu::FindDescription(Cause(), ccdt, sizeof ccdt / sizeof *ccdt);
|
|
}
|
|
|
|
CSmartCard::Exception::ErrorCode
|
|
CSmartCard::Exception::Error() const throw()
|
|
{
|
|
return m_cc;
|
|
}
|
|
|
|
CSmartCard::Instruction
|
|
CSmartCard::Exception::Ins() const throw()
|
|
{
|
|
return m_ins;
|
|
}
|
|
|
|
CSmartCard::StatusWord
|
|
CSmartCard::Exception::Status() const throw()
|
|
{
|
|
return m_sw;
|
|
}
|
|
|
|
// Predicates
|
|
// Static Variables
|
|
|
|
/////////////////////////// PROTECTED /////////////////////////////////
|
|
|
|
// C'tors/D'tors
|
|
// Operators
|
|
// Operations
|
|
// Access
|
|
// Predicates
|
|
// Static Variables
|
|
|
|
|
|
/////////////////////////// PRIVATE /////////////////////////////////
|
|
|
|
// C'tors/D'tors
|
|
// Operators
|
|
// Operations
|
|
// Access
|
|
// Predicates
|
|
// Static Variables
|
|
|
|
///////////////////// End CSmartCard::Exception ///////////////////////////
|
|
|
|
CSmartCard::CSmartCard(const SCARDHANDLE hCardHandle,
|
|
const char* szReaderName,
|
|
const SCARDCONTEXT hContext,
|
|
const DWORD dwMode)
|
|
: m_hCard(hCardHandle),
|
|
m_hContext(hContext),
|
|
m_CurrentDirectory(),
|
|
m_CurrentFile(),
|
|
m_dwShareMode(dwMode),
|
|
m_fSupportLogout(false),
|
|
m_IOPLock(szReaderName),
|
|
m_apSharedMarker(new CSharedMarker(string(szReaderName))),
|
|
m_vecEvents(),
|
|
m_dwEventCounter(0),
|
|
m_cResponseAvailable(0),
|
|
m_sCardName()
|
|
{
|
|
m_IOPLock.Init(this);
|
|
ResetSelect();
|
|
}
|
|
|
|
CSmartCard::~CSmartCard()
|
|
{
|
|
try {
|
|
while(m_vecEvents.size())
|
|
{
|
|
delete *(m_vecEvents.begin());
|
|
m_vecEvents.erase(m_vecEvents.begin());
|
|
}
|
|
|
|
// Disconnect the card.
|
|
if (m_hCard)
|
|
SCardDisconnect(m_hCard, SCARD_LEAVE_CARD);
|
|
}
|
|
catch(...) {}
|
|
|
|
}
|
|
|
|
|
|
void CSmartCard::ReConnect()
|
|
{
|
|
|
|
DWORD dwProtocol;
|
|
|
|
HRESULT hr = SCardReconnect(m_hCard, m_dwShareMode, SCARD_PROTOCOL_T0, SCARD_LEAVE_CARD, &dwProtocol);
|
|
if (hr != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hr);
|
|
|
|
}
|
|
|
|
void CSmartCard::ResetCard()
|
|
{
|
|
//*
|
|
CLockWrap wrap(&m_IOPLock);
|
|
|
|
ResetSelect();
|
|
|
|
HRESULT hResult;
|
|
|
|
hResult = SCardBeginTransaction(m_hCard);
|
|
|
|
// We don't apply full logic to handle error detection since
|
|
// this was done in the CLockWrap constructor.
|
|
|
|
if (hResult != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hResult);
|
|
|
|
hResult = SCardEndTransaction(m_hCard,SCARD_RESET_CARD);
|
|
if (hResult != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hResult);
|
|
//*/
|
|
}
|
|
|
|
char const *
|
|
CSmartCard::getCardName() const
|
|
{
|
|
return m_sCardName.c_str();
|
|
}
|
|
|
|
void
|
|
CSmartCard::setCardName(char const *szName)
|
|
{
|
|
m_sCardName = string(szName);
|
|
}
|
|
|
|
void
|
|
CSmartCard::SendCardAPDU(const BYTE bCLA, const BYTE bINS,
|
|
const BYTE bP1, const BYTE bP2,
|
|
const BYTE bLengthIn, const BYTE* bDataIn,
|
|
const BYTE bLengthOut, BYTE* bDataOut)
|
|
{
|
|
CLockWrap wrap(&m_IOPLock);
|
|
HRESULT hResult;
|
|
DWORD dwRecvLength;
|
|
BYTE bSW[2];
|
|
StatusWord sw;
|
|
|
|
//////////////////
|
|
// Build APDU //
|
|
//////////////////
|
|
|
|
scu::AutoArrayPtr<BYTE> aabInput(new BYTE[5 + bLengthIn]);
|
|
scu::AutoArrayPtr<BYTE> aabOutput;
|
|
|
|
aabInput[0] = bCLA;
|
|
aabInput[1] = bINS;
|
|
aabInput[2] = bP1;
|
|
aabInput[3] = bP2;
|
|
aabInput[4] = bLengthIn;
|
|
|
|
///////////////////////////
|
|
// Is data being sent? //
|
|
///////////////////////////
|
|
if (bLengthIn)
|
|
{
|
|
memcpy(&aabInput[5], bDataIn, bLengthIn);
|
|
dwRecvLength = 2;
|
|
|
|
hResult = SCardTransmit(m_hCard, SCARD_PCI_T0,
|
|
aabInput.Get(), bLengthIn + 5,
|
|
NULL, bSW, &dwRecvLength);
|
|
|
|
if (hResult != SCARD_S_SUCCESS)
|
|
{
|
|
DWORD dwState;
|
|
DWORD dwProtocol;
|
|
BYTE bATR[cMaxAtrLength];
|
|
DWORD dwATRLen = sizeof bATR / sizeof *bATR;
|
|
DWORD dwReaderNameLen = 0;
|
|
|
|
HRESULT hr = SCardStatus(m_hCard, NULL,
|
|
&dwReaderNameLen, &dwState,
|
|
&dwProtocol, bATR,
|
|
&dwATRLen);
|
|
if (hr == SCARD_W_RESET_CARD)
|
|
{
|
|
ResetSelect();
|
|
ReConnect();
|
|
}
|
|
else
|
|
throw scu::OsException(hr);
|
|
|
|
hr=SCardTransmit(m_hCard, SCARD_PCI_T0,
|
|
aabInput.Get(), bLengthIn + 5, NULL,
|
|
bSW, &dwRecvLength);
|
|
if (hr != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hr);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// Rebuffer information so registered functions may not alter data in the IOP //
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
scu::AutoArrayPtr<BYTE> aabSendData(new BYTE[bLengthIn + 5]);
|
|
memcpy((void*)aabSendData.Get(), (void*)aabInput.Get(),
|
|
bLengthIn + 5);
|
|
BYTE bReceiveData[2];
|
|
memcpy((void*)bReceiveData, (void*)bSW, 2);
|
|
|
|
///////////////////////////////////////
|
|
// Fire event for information sent //
|
|
///////////////////////////////////////
|
|
|
|
FireEvents(0, bLengthIn + 5, aabSendData.Get());
|
|
FireEvents(1, 2, bReceiveData);
|
|
|
|
sw = (bSW[0] * 256) + bSW[1];
|
|
|
|
ProcessReturnStatus(bCLA, bINS, sw);
|
|
|
|
//////////////////////////////////
|
|
// Should we expect data back? //
|
|
//////////////////////////////////
|
|
if (bLengthOut)
|
|
{
|
|
if (ResponseLengthAvailable())
|
|
GetResponse(bCLA, bLengthOut, bDataOut);
|
|
else
|
|
throw iop::Exception(iop::ccNoResponseAvailable);
|
|
}
|
|
}
|
|
//////////////////////////////
|
|
// Data is NOT being sent //
|
|
//////////////////////////////
|
|
else
|
|
{
|
|
aabInput[4] = bLengthOut;
|
|
dwRecvLength = bLengthOut + 2;
|
|
aabOutput = scu::AutoArrayPtr<BYTE>(new BYTE[dwRecvLength]);
|
|
memset(aabOutput.Get(), 0, dwRecvLength);
|
|
|
|
hResult = SCardTransmit(m_hCard, SCARD_PCI_T0,
|
|
aabInput.Get(), 5, NULL,
|
|
aabOutput.Get(), &dwRecvLength);
|
|
|
|
/////////////////////////////////////////////
|
|
// if card has been reset, then reconnect //
|
|
/////////////////////////////////////////////
|
|
|
|
if (hResult != SCARD_S_SUCCESS)
|
|
{
|
|
DWORD dwState;
|
|
DWORD dwProtocol;
|
|
BYTE bATR[cMaxAtrLength];
|
|
DWORD dwATRLen = sizeof bATR / sizeof *bATR;
|
|
DWORD dwReaderNameLen = 0;
|
|
|
|
HRESULT hr = SCardStatus(m_hCard, NULL, &dwReaderNameLen, &dwState, &dwProtocol, bATR, &dwATRLen);
|
|
if (hr == SCARD_W_RESET_CARD)
|
|
{
|
|
ResetSelect();
|
|
ReConnect();
|
|
}
|
|
else
|
|
throw scu::OsException(hResult);
|
|
|
|
hr=SCardTransmit(m_hCard, SCARD_PCI_T0,
|
|
aabInput.Get(), 5, NULL,
|
|
aabOutput.Get(), &dwRecvLength);
|
|
if (hr != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hResult);
|
|
|
|
}
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// Rebuffer information so registered functions may not alter data in the IOP //
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
BYTE bSendData[5];
|
|
memcpy((void*)bSendData, (void*)aabInput.Get(), 5);
|
|
scu::AutoArrayPtr<BYTE> aabReceiveData(new BYTE[dwRecvLength/*bLengthOut + 2*/]);
|
|
memcpy((void*)aabReceiveData.Get(), (void*)aabOutput.Get(),
|
|
dwRecvLength/*bLengthOut + 2*/);
|
|
|
|
////////////////////////////////////////////////////
|
|
// Fire event for information sent and received //
|
|
////////////////////////////////////////////////////
|
|
|
|
FireEvents(0, 5, bSendData);
|
|
FireEvents(1, dwRecvLength/*bLengthOut + 2*/,
|
|
aabReceiveData.Get());
|
|
|
|
sw = (aabOutput[dwRecvLength - 2] * 256) +
|
|
aabOutput[dwRecvLength - 1];
|
|
|
|
ProcessReturnStatus(bCLA, bINS, sw);
|
|
|
|
memcpy(bDataOut, aabOutput.Get(), bLengthOut);
|
|
}
|
|
}
|
|
|
|
void CSmartCard::RequireSelect()
|
|
{
|
|
if ((m_CurrentDirectory.IsEmpty()) || (m_CurrentFile.IsEmpty()))
|
|
throw iop::Exception(iop::ccNoFileSelected);
|
|
}
|
|
|
|
void
|
|
CSmartCard::GetResponse(ClassByte cb,
|
|
BYTE bDataLength,
|
|
BYTE *bDataOut)
|
|
{
|
|
// TO DO: lock redundant??? Wouldn't GetResponse always be called
|
|
// by a routine that establishes a lock?
|
|
CLockWrap wrap(&m_IOPLock);
|
|
|
|
if (0 == ResponseLengthAvailable())
|
|
throw iop::Exception(iop::ccNoResponseAvailable);
|
|
|
|
struct Command // T=0 command bytes
|
|
{
|
|
ClassByte m_cb;
|
|
Instruction m_ins;
|
|
BYTE m_bP1;
|
|
BYTE m_bP2;
|
|
BYTE m_bP3;
|
|
} cmnd = { cb, insGetResponse, 0x00, 0x00, bDataLength };
|
|
DWORD dwRecvLength = bDataLength + sizeof StatusWord;
|
|
BYTE bOutput[cMaxGetResponseLength];
|
|
|
|
memset(bOutput, 0, dwRecvLength);
|
|
|
|
HRESULT hr = SCardTransmit(m_hCard, SCARD_PCI_T0,
|
|
reinterpret_cast<LPCBYTE>(&cmnd),
|
|
sizeof cmnd, NULL, bOutput,
|
|
&dwRecvLength);
|
|
|
|
if (hr != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hr);
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// Rebuffer information so registered functions may not alter data in the IOP //
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
BYTE bSendData[sizeof cmnd];
|
|
memcpy(static_cast<void *>(bSendData),
|
|
static_cast<void const *>(&cmnd), sizeof cmnd);
|
|
scu::AutoArrayPtr<BYTE> aabReceiveData(new BYTE[dwRecvLength/*bDataLength + 2*/]);
|
|
memcpy((void*)aabReceiveData.Get(), (void*)bOutput, dwRecvLength/*bDataLength + 2*/);
|
|
|
|
////////////////////////////////////////////////////
|
|
// Fire event for information sent and received //
|
|
////////////////////////////////////////////////////
|
|
|
|
FireEvents(0, sizeof bSendData, bSendData);
|
|
FireEvents(1, dwRecvLength/*bDataLength + 2*/, aabReceiveData.Get());
|
|
|
|
//////////////////////////////////////////////////////////
|
|
// Set card's status code and map to IOP status codes //
|
|
//////////////////////////////////////////////////////////
|
|
|
|
StatusWord sw = (bOutput[dwRecvLength - sizeof StatusWord]*256) +
|
|
bOutput[dwRecvLength - (sizeof StatusWord - 1)];
|
|
|
|
ProcessReturnStatus(cb, insGetResponse, sw);
|
|
|
|
memcpy(bDataOut, bOutput, bDataLength);
|
|
}
|
|
|
|
void
|
|
CSmartCard::ReadBinary(const WORD wOffset,
|
|
const WORD wDataLength,
|
|
BYTE* bData)
|
|
{
|
|
CLockWrap wrap(&m_IOPLock);
|
|
|
|
TransferData(wOffset, wDataLength, bData, *this,
|
|
&CSmartCard::DoReadBlock, cMaxRwDataBlock);
|
|
|
|
}
|
|
|
|
void
|
|
CSmartCard::WriteBinary(const WORD wOffset,
|
|
const WORD wDataLength,
|
|
const BYTE* bData)
|
|
{
|
|
CLockWrap wrap(&m_IOPLock);
|
|
|
|
m_apSharedMarker->UpdateMarker(CMarker::WriteMarker);
|
|
|
|
TransferData(wOffset, wDataLength, bData, *this,
|
|
&CSmartCard::DoWriteBlock, cMaxRwDataBlock);
|
|
|
|
}
|
|
|
|
void CSmartCard::ResetSelect()
|
|
{
|
|
m_CurrentDirectory.Clear();
|
|
m_CurrentFile.Clear();
|
|
}
|
|
|
|
void
|
|
CSmartCard::GetCurrentDir(char* CurrentDirectory)
|
|
{
|
|
if (!m_CurrentDirectory.IsEmpty())
|
|
strcpy(CurrentDirectory,m_CurrentDirectory.GetStringPath().c_str());
|
|
else
|
|
throw iop::Exception(ccInvalidParameter);
|
|
}
|
|
|
|
void
|
|
CSmartCard::GetCurrentFile(char* CurrentFile)
|
|
{
|
|
if (!m_CurrentFile.IsEmpty())
|
|
strcpy(CurrentFile,m_CurrentFile.GetStringPath().c_str());
|
|
else
|
|
throw iop::Exception(ccInvalidParameter);
|
|
}
|
|
|
|
DWORD CSmartCard::RegisterEvent(void (*FireEvent)(void *pToCard, int
|
|
iEventCode, DWORD
|
|
dwLen, BYTE* bData),
|
|
void *pToCard)
|
|
{
|
|
EventInfo *Event = new EventInfo;
|
|
Event->dwHandle = ++m_dwEventCounter;
|
|
Event->FireEvent = FireEvent;
|
|
Event->pToCard = pToCard;
|
|
|
|
m_vecEvents.push_back(Event);
|
|
|
|
return m_dwEventCounter;
|
|
}
|
|
|
|
bool CSmartCard::UnregisterEvent(DWORD dwHandle)
|
|
{
|
|
vector<EventInfo*>::iterator iter;
|
|
|
|
for(iter = m_vecEvents.begin(); iter != m_vecEvents.end(); iter++)
|
|
{
|
|
if ((*iter)->dwHandle == dwHandle)
|
|
break;
|
|
}
|
|
|
|
if (iter == m_vecEvents.end())
|
|
return false;
|
|
|
|
delete (*iter);
|
|
m_vecEvents.erase(iter);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool CSmartCard::HasProperty(WORD wPropNumber)
|
|
{
|
|
if (wPropNumber > 512)
|
|
return false;
|
|
|
|
////////////////////////////////////
|
|
// Open path to registered keys //
|
|
////////////////////////////////////
|
|
|
|
HKEY hkCardKey;
|
|
HKEY hkTestKey;
|
|
char szCardPath[] = "SOFTWARE\\Schlumberger\\Smart Cards and Terminals\\Smart Cards";
|
|
|
|
RegOpenKeyEx(HKEY_LOCAL_MACHINE, szCardPath, NULL, KEY_READ, &hkCardKey);
|
|
|
|
//////////////////////////////////////////////
|
|
// Enumerate subkeys to find an ATR match //
|
|
//////////////////////////////////////////////
|
|
|
|
FILETIME fileTime;
|
|
char szATR[] = "ATR";
|
|
char szMask[] = "ATR Mask";
|
|
char szProperties[] = "Properties";
|
|
char sBuffer[1024];
|
|
BYTE bATR[cMaxAtrLength];
|
|
BYTE bATRtest[cMaxAtrLength];
|
|
BYTE bMask[cMaxAtrLength];
|
|
BYTE bProperties[512];
|
|
BYTE bATRLength = sizeof bATR / sizeof *bATR;
|
|
DWORD dwBufferSize = sizeof(sBuffer);
|
|
DWORD dwATRSize = sizeof bATR / sizeof *bATR;
|
|
DWORD dwMaskSize = sizeof bMask / sizeof *bMask;
|
|
DWORD dwPropSize = sizeof(bProperties);
|
|
DWORD index = 0;
|
|
|
|
getATR(bATR, bATRLength);
|
|
|
|
LONG iRetVal = RegEnumKeyEx(hkCardKey, index, sBuffer, &dwBufferSize, NULL, NULL, NULL, &fileTime);
|
|
while (iRetVal == ERROR_SUCCESS)
|
|
{
|
|
RegOpenKeyEx(hkCardKey, sBuffer, NULL, KEY_READ, &hkTestKey);
|
|
|
|
RegQueryValueEx(hkTestKey, szATR, NULL, NULL, bATRtest, &dwATRSize);
|
|
RegQueryValueEx(hkTestKey, szMask, NULL, NULL, bMask, &dwMaskSize);
|
|
|
|
if (dwATRSize == bATRLength)
|
|
{
|
|
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 = cMaxAtrLength;
|
|
dwMaskSize = cMaxAtrLength;
|
|
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
|
|
if (iRetVal == ERROR_SUCCESS)
|
|
{
|
|
RegQueryValueEx(hkTestKey, szProperties, NULL, NULL, bProperties, &dwPropSize);
|
|
|
|
return (bProperties[(wPropNumber - 1) / 8] & (1 << ((wPropNumber - 1) % 8))) ? true : false;
|
|
}
|
|
// loop wasn't broken, i.e., ATR not found
|
|
else
|
|
return false;
|
|
}
|
|
|
|
void
|
|
CSmartCard::getATR(BYTE* bATR, BYTE& iATRLength)
|
|
{
|
|
DWORD dwProtocol;
|
|
LPDWORD pcchReaderLen = 0;
|
|
DWORD dwState;
|
|
BYTE bMyATR[cMaxAtrLength];
|
|
DWORD dwAtrLen = sizeof bMyATR / sizeof *bMyATR;
|
|
|
|
HRESULT hr = SCardStatus(m_hCard, NULL, pcchReaderLen, &dwState, &dwProtocol, bMyATR, &dwAtrLen);
|
|
|
|
if (hr == SCARD_W_RESET_CARD)
|
|
{
|
|
ResetSelect();
|
|
ReConnect();
|
|
hr = SCardStatus(m_hCard, NULL, pcchReaderLen, &dwState, &dwProtocol, bMyATR, &dwAtrLen);
|
|
}
|
|
if (hr != SCARD_S_SUCCESS)
|
|
throw scu::OsException(hr);
|
|
|
|
if ((BYTE)dwAtrLen > iATRLength)
|
|
throw iop::Exception(ccInvalidParameter);
|
|
|
|
memcpy(bATR, bMyATR, dwAtrLen);
|
|
iATRLength = (BYTE)dwAtrLen;
|
|
}
|
|
|
|
void CSmartCard::FireEvents(int iEventCode, DWORD dwLength, BYTE *bsData)
|
|
{
|
|
vector<EventInfo*>::iterator iter;
|
|
|
|
for (iter = m_vecEvents.begin(); iter != m_vecEvents.end(); iter++)
|
|
{
|
|
EventInfo* pEvent = *iter;
|
|
|
|
pEvent->FireEvent(pEvent->pToCard, iEventCode, dwLength, bsData);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
BYTE CSmartCard::FormatPath(char *szOutputPath, const char *szInputPath)
|
|
{
|
|
bool fResult = true;
|
|
BYTE bIndex = 0;
|
|
BYTE bFileCount = 1; // always allocate memory for at least 1 token
|
|
BYTE bFileIDLength = 0;
|
|
BYTE bPathLength = strlen(szInputPath);
|
|
WORD wFileHexID;
|
|
char szPad[] = "0";
|
|
char *cTestChar;
|
|
char *szHexID;
|
|
|
|
//////////////////////////////////////////////
|
|
// Count number of tokens in desired path //
|
|
//////////////////////////////////////////////
|
|
for (bIndex = 0; bIndex < bPathLength - 1; bIndex++)
|
|
{
|
|
if (szInputPath[bIndex] == '/' && szInputPath[bIndex + 1] != '/')
|
|
bFileCount++;
|
|
}
|
|
|
|
|
|
// Check path size
|
|
|
|
if (bFileCount * 5 + 1 > cMaxPathLength)
|
|
throw iop::Exception(iop::ccFilePathTooLong);
|
|
|
|
scu::AutoArrayPtr<char> aaszPathIn(new char[bPathLength + 1]);
|
|
memset((void*)aaszPathIn.Get(), 0, bPathLength + 1);
|
|
memset((void*)szOutputPath, 0, bFileCount * 5 + 1);
|
|
strcpy(aaszPathIn.Get(), szInputPath);
|
|
|
|
iop::CauseCode cc;
|
|
szHexID = strtok(aaszPathIn.Get(), "/");
|
|
for (bFileCount = 0; szHexID; bFileCount++, szHexID = strtok(NULL, "/"))
|
|
{
|
|
/////////////////////////////////////////////////////////
|
|
// File ID is too large -- greater than 4 characters //
|
|
/////////////////////////////////////////////////////////
|
|
if (strlen(szHexID) > 4)
|
|
{
|
|
fResult = false;
|
|
cc = iop::ccFileIdTooLarge;
|
|
break;
|
|
}
|
|
|
|
wFileHexID = (WORD)strtoul(szHexID, &cTestChar, 16);
|
|
/////////////////////////////////////////////////////
|
|
// File ID was not in hexadecimal representation //
|
|
/////////////////////////////////////////////////////
|
|
if (*cTestChar != NULL)
|
|
{
|
|
fResult = false;
|
|
cc = iop::ccFileIdNotHex;
|
|
break;
|
|
}
|
|
|
|
szOutputPath[bFileCount * 5] = '/';
|
|
/////////////////////////////////////////////////////////////////
|
|
// Pad file ID and put formatted file ID into formatted path //
|
|
/////////////////////////////////////////////////////////////////
|
|
for (bFileIDLength = strlen(szHexID); bFileIDLength < 4; bFileIDLength++)
|
|
strcat((szOutputPath + (bFileCount * 5) + (4 - bFileIDLength)), szPad);
|
|
|
|
strcpy((szOutputPath + (bFileCount * 5) + (5 - strlen(szHexID))), szHexID);
|
|
}
|
|
|
|
///////////////////////////////////////////
|
|
// If file ID formatting fails, throw //
|
|
///////////////////////////////////////////
|
|
if (!fResult)
|
|
{
|
|
// delete [] szOutputPath; // Can not mean to delete this???? HB.
|
|
throw iop::Exception(cc);
|
|
}
|
|
|
|
_strupr(szOutputPath);
|
|
|
|
return bFileCount;
|
|
}
|
|
|
|
CMarker CSmartCard::Marker(CMarker::MarkerType Type)
|
|
{
|
|
return m_apSharedMarker->Marker(Type);
|
|
}
|
|
|
|
void
|
|
CSmartCard::GetState(DWORD &rdwState,
|
|
DWORD &rdwProtocol)
|
|
{
|
|
BYTE bATR[cMaxAtrLength];
|
|
DWORD dwATRLen = sizeof bATR / sizeof *bATR;
|
|
DWORD dwReaderNameLen = 0;
|
|
|
|
HRESULT hr = SCardStatus(m_hCard, 0, &dwReaderNameLen, &rdwState,
|
|
&rdwProtocol, bATR, &dwATRLen);
|
|
|
|
if (SCARD_W_RESET_CARD == hr)
|
|
{
|
|
ResetSelect();
|
|
ReConnect();
|
|
|
|
// try again...
|
|
hr = SCardStatus(m_hCard, 0, &dwReaderNameLen, &rdwState,
|
|
&rdwProtocol, bATR, &dwATRLen);
|
|
}
|
|
|
|
if (SCARD_S_SUCCESS != hr)
|
|
throw scu::OsException(hr);
|
|
}
|
|
|
|
void
|
|
CSmartCard::DefaultDispatchError(ClassByte cb,
|
|
Instruction ins,
|
|
StatusWord sw) const
|
|
{
|
|
CauseCode cc;
|
|
bool fDoThrow = true;
|
|
|
|
switch (sw)
|
|
{
|
|
case 0x6283:
|
|
cc = ccFileInvalidated;
|
|
break;
|
|
|
|
case 0x6581:
|
|
cc = ccMemoryProblem;
|
|
break;
|
|
|
|
case 0x6982:
|
|
cc = ccNoAccess;
|
|
break;
|
|
|
|
case 0x6983:
|
|
cc = ccKeyBlocked;
|
|
break;
|
|
|
|
case 0x6A80:
|
|
cc = ccFileTypeInvalid;
|
|
break;
|
|
|
|
case 0x6A82:
|
|
cc = ccFileNotFound;
|
|
break;
|
|
|
|
case 0x6A84:
|
|
cc = ccInsufficientSpace;
|
|
break;
|
|
|
|
case 0x6B00:
|
|
cc = ccIncorrectP1P2;
|
|
break;
|
|
|
|
case 0x6D00:
|
|
cc = ccUnknownInstructionCode;
|
|
break;
|
|
|
|
case 0x6E00:
|
|
cc = ccUnknownInstructionClass;
|
|
break;
|
|
|
|
case 0x6F00:
|
|
cc = ccUnidentifiedTechnicalProblem;
|
|
break;
|
|
|
|
case 0x90FF:
|
|
cc = ccTimeOut;
|
|
break;
|
|
|
|
case 0x9002:
|
|
cc = ccInvalidAnswerReceived;
|
|
break;
|
|
|
|
default:
|
|
if (0x67 == HIBYTE(sw))
|
|
cc = ccIncorrectP3;
|
|
else
|
|
cc = ccUnknownStatus;
|
|
break;
|
|
}
|
|
|
|
if (fDoThrow)
|
|
throw Exception(cc, cb, ins, sw);
|
|
}
|
|
|
|
void
|
|
CSmartCard::DispatchError(ClassByte cb,
|
|
Instruction ins,
|
|
StatusWord sw) const
|
|
{
|
|
CauseCode cc;
|
|
bool fDoThrow = true;
|
|
|
|
switch (ins)
|
|
{
|
|
case insGetResponse:
|
|
switch (sw)
|
|
{
|
|
case 0x6281:
|
|
cc = ccReturnedDataCorrupted;
|
|
break;
|
|
|
|
case 0x6A86:
|
|
cc = ccIncorrectP1P2;
|
|
break;
|
|
|
|
default:
|
|
if (0x6C == HIBYTE(sw))
|
|
cc = ccIncorrectP3;
|
|
else
|
|
fDoThrow = false;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
fDoThrow = false;
|
|
break;
|
|
}
|
|
|
|
if (fDoThrow)
|
|
throw Exception(cc, cb, ins, sw);
|
|
|
|
DefaultDispatchError(cb, ins, sw);
|
|
}
|
|
|
|
BYTE
|
|
CSmartCard::ResponseLengthAvailable() const
|
|
{
|
|
return m_cResponseAvailable;
|
|
}
|
|
|
|
void
|
|
CSmartCard::ResponseLengthAvailable(BYTE cResponseLength)
|
|
{
|
|
m_cResponseAvailable = cResponseLength;
|
|
}
|
|
|
|
void
|
|
CSmartCard::WriteBlock(WORD wOffset,
|
|
BYTE const *pbBuffer,
|
|
BYTE cLength)
|
|
{
|
|
DoWriteBlock(wOffset, pbBuffer, cLength);
|
|
|
|
m_apSharedMarker->UpdateMarker(CMarker::WriteMarker);
|
|
}
|
|
|
|
void
|
|
CSmartCard::ProcessReturnStatus(ClassByte cb,
|
|
Instruction ins,
|
|
StatusWord sw)
|
|
{
|
|
ResponseLengthAvailable(0);
|
|
if (swSuccess != sw)
|
|
{
|
|
if (0x61 == HIBYTE(sw))
|
|
ResponseLengthAvailable(LOBYTE(sw));
|
|
else
|
|
DispatchError(cb, ins, sw);
|
|
}
|
|
}
|
|
|
|
} // namespace iop
|