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.
6695 lines
192 KiB
6695 lines
192 KiB
/*****************************************************************************
|
|
@doc INT EXT
|
|
******************************************************************************
|
|
* $ProjectName: $
|
|
* $ProjectRevision: $
|
|
*-----------------------------------------------------------------------------
|
|
* $Source: z:/pr/cmbs0/sw/sccmn50m.ms/rcs/sccmcb.c $
|
|
* $Revision: 1.7 $
|
|
*-----------------------------------------------------------------------------
|
|
* $Author: WFrischauf $
|
|
*-----------------------------------------------------------------------------
|
|
* History: see EOF
|
|
*-----------------------------------------------------------------------------
|
|
*
|
|
* Copyright © 2000 OMNIKEY AG
|
|
******************************************************************************/
|
|
|
|
#include <stdio.h>
|
|
#include "sccmn50m.h"
|
|
|
|
|
|
|
|
|
|
static ULONG dataRatesSupported[] = {9600,38400};
|
|
static ULONG CLKFrequenciesSupported[] = {4000,5000};
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_CardPower(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
NTSTATUS DebugStatus = STATUS_SUCCESS;
|
|
UCHAR pbAtrBuffer[MAXIMUM_ATR_LENGTH];
|
|
UCHAR abSyncAtrBuffer[MAXIMUM_ATR_LENGTH];
|
|
ULONG ulAtrLength;
|
|
#if DBG
|
|
ULONG i;
|
|
#endif;
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!CardPower: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
#if DBG
|
|
switch (pSmartcardExtension->MinorIoControlCode)
|
|
{
|
|
case SCARD_WARM_RESET:
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
( "%s!SCARD_WARM_RESTART\n",
|
|
DRIVER_NAME)
|
|
);
|
|
break;
|
|
case SCARD_COLD_RESET:
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
( "%s!SCARD_COLD_RESTART\n",
|
|
DRIVER_NAME)
|
|
);
|
|
break;
|
|
case SCARD_POWER_DOWN:
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
( "%s!SCARD_POWER_DOWN\n",
|
|
DRIVER_NAME)
|
|
);
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
//DbgBreakPoint();
|
|
|
|
switch (pSmartcardExtension->MinorIoControlCode)
|
|
{
|
|
case SCARD_WARM_RESET:
|
|
case SCARD_COLD_RESET:
|
|
status = SCCMN50M_PowerOn(pSmartcardExtension,
|
|
&ulAtrLength,
|
|
pbAtrBuffer,
|
|
sizeof(pbAtrBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitReaderPower;
|
|
}
|
|
|
|
pSmartcardExtension->CardCapabilities.ATR.Length = (UCHAR)ulAtrLength;
|
|
|
|
if (pSmartcardExtension->ReaderExtension->fRawModeNecessary == FALSE)
|
|
{
|
|
// copy ATR to smart card structure
|
|
// the lib needs the ATR for evaluation of the card parameters
|
|
|
|
MemCpy(pSmartcardExtension->CardCapabilities.ATR.Buffer,
|
|
sizeof(pSmartcardExtension->CardCapabilities.ATR.Buffer),
|
|
pbAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
pSmartcardExtension->CardCapabilities.ATR.Length = (UCHAR)ulAtrLength;
|
|
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_NEGOTIABLE;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
|
|
status = SmartcardUpdateCardCapabilities(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitReaderPower;
|
|
}
|
|
|
|
// add extra guard time value to card stop bits
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = (UCHAR)(pSmartcardExtension->CardCapabilities.N);
|
|
|
|
// copy ATR to user space
|
|
MemCpy(pSmartcardExtension->IoRequest.ReplyBuffer,
|
|
pSmartcardExtension->IoRequest.ReplyBufferLength,
|
|
pbAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
*pSmartcardExtension->IoRequest.Information = ulAtrLength;
|
|
#if DBG
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!ATR : ",
|
|
DRIVER_NAME));
|
|
for (i = 0;i < ulAtrLength;i++)
|
|
{
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%2.2x ",
|
|
pSmartcardExtension->CardCapabilities.ATR.Buffer[i]));
|
|
}
|
|
SmartcardDebug(DEBUG_ATR,("\n"));
|
|
|
|
#endif
|
|
|
|
}
|
|
else
|
|
{
|
|
abSyncAtrBuffer[0] = 0x3B;
|
|
abSyncAtrBuffer[1] = 0x04;
|
|
MemCpy(&abSyncAtrBuffer[2],
|
|
sizeof(abSyncAtrBuffer)-2,
|
|
pbAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
|
|
ulAtrLength += 2;
|
|
|
|
MemCpy(pSmartcardExtension->CardCapabilities.ATR.Buffer,
|
|
sizeof(pSmartcardExtension->CardCapabilities.ATR.Buffer),
|
|
abSyncAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
pSmartcardExtension->CardCapabilities.ATR.Length = (UCHAR)(ulAtrLength);
|
|
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_SPECIFIC;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_T0;
|
|
|
|
status = SmartcardUpdateCardCapabilities(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitReaderPower;
|
|
}
|
|
SmartcardDebug(DEBUG_ATR,("ATR of synchronous smart card : %2.2x %2.2x %2.2x %2.2x\n",
|
|
pbAtrBuffer[0],pbAtrBuffer[1],pbAtrBuffer[2],pbAtrBuffer[3]));
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.fCardResetRequested = TRUE;
|
|
|
|
// copy ATR to user space
|
|
MemCpy(pSmartcardExtension->IoRequest.ReplyBuffer,
|
|
pSmartcardExtension->IoRequest.ReplyBufferLength,
|
|
abSyncAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
*pSmartcardExtension->IoRequest.Information = ulAtrLength;
|
|
}
|
|
|
|
break;
|
|
|
|
case SCARD_POWER_DOWN:
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitReaderPower;
|
|
}
|
|
|
|
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_SWALLOWED;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
|
|
break;
|
|
}
|
|
|
|
|
|
|
|
ExitReaderPower:
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!CardPower: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_PowerOn (
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
OUT PULONG pulAtrLength,
|
|
OUT PUCHAR pbAtrBuffer,
|
|
IN ULONG ulAtrBufferSize
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
|
|
|
|
// We always use 0x80 for reset delay
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.ResetDelay = 0x80;
|
|
|
|
|
|
if (SCCMN50M_IsAsynchronousSmartCard(pSmartcardExtension) == TRUE)
|
|
{
|
|
if (pSmartcardExtension->MinorIoControlCode == SCARD_COLD_RESET)
|
|
{
|
|
status = SCCMN50M_UseColdWarmResetStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize,
|
|
FALSE);
|
|
// if cold reset was not succesfull ,it maybe a SAMOS card with the sensor bug
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
status = SCCMN50M_UseColdWarmResetStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize,
|
|
TRUE);
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
status = SCCMN50M_UseParsingStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
status = SCCMN50M_UseColdWarmResetStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize,
|
|
TRUE);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
status = SCCMN50M_UseParsingStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("check if synchronous smart card is inserted\n"));
|
|
// try to find a synchronous smart card
|
|
status = SCCMN50M_UseSyncStrategy(pSmartcardExtension,
|
|
pulAtrLength,
|
|
pbAtrBuffer,
|
|
ulAtrBufferSize);
|
|
}
|
|
|
|
|
|
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
// smart card not powered
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
*pulAtrLength = 0;
|
|
DebugStatus = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
return status;
|
|
}
|
|
else
|
|
{
|
|
// add extra guard time value to card stop bits
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = (UCHAR)(pSmartcardExtension->CardCapabilities.N);
|
|
return status;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
This routine inverts the buffer
|
|
Bit0 -> Bit 7
|
|
Bit1 -> Bit 6
|
|
Bit2 -> Bit 5
|
|
Bit3 -> Bit 4
|
|
Bit4 -> Bit 3
|
|
Bit5 -> Bit 2
|
|
Bit6 -> Bit 1
|
|
Bit7 -> Bit 0
|
|
|
|
|
|
Arguments: pbBuffer ... pointer to buffer
|
|
ulBufferSize ... size of buffer
|
|
|
|
|
|
Return Value: none
|
|
|
|
*****************************************************************************/
|
|
VOID SCCMN50M_InverseBuffer (
|
|
PUCHAR pbBuffer,
|
|
ULONG ulBufferSize
|
|
)
|
|
{
|
|
ULONG i;
|
|
ULONG j;
|
|
ULONG m;
|
|
ULONG n;
|
|
|
|
for (i=0; i<ulBufferSize; i++)
|
|
{
|
|
n = 0;
|
|
for (j=1; j<=8; j++)
|
|
{
|
|
m = (pbBuffer[i] << j);
|
|
m &= 0x00000100;
|
|
n |= (m >> (9-j));
|
|
}
|
|
pbBuffer[i] = (UCHAR)~n;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This function always permforms a cold reset of the smart card.
|
|
A SAMOS card with the sensor bug will not be powered by this function.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_UseParsingStrategy (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
OUT PULONG pulAtrLength,
|
|
OUT PUCHAR pbAtrBuffer,
|
|
IN ULONG ulAtrBufferSize
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
UCHAR ulCardType;
|
|
UCHAR ReadBuffer[SCARD_ATR_LENGTH];
|
|
UCHAR bAtrBytesRead[SCARD_ATR_LENGTH];
|
|
ULONG ulBytesRead;
|
|
BOOLEAN fInverseAtr = FALSE;
|
|
ULONG ulAtrBufferOffset = 0;
|
|
ULONG ulHistoricalBytes;
|
|
ULONG ulNextStepBytesToRead;
|
|
ULONG ulPrevStepBytesRead;
|
|
ULONG i;
|
|
BOOLEAN fTDxSent;
|
|
BOOLEAN fAtrParsed;
|
|
BOOLEAN fOnlyT0 = TRUE;
|
|
ULONG ulOldReadTotalTimeoutMultiplier;
|
|
|
|
|
|
// DBGBreakPoint();
|
|
|
|
// set ReadTotalTimeoutMultiplier to 250ms (9600 * 372/f = initial waiting time)
|
|
ulOldReadTotalTimeoutMultiplier = pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier = 250;
|
|
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0x02;
|
|
|
|
for (ulCardType = ASYNC3_CARD;ulCardType <= ASYNC5_CARD;ulCardType++)
|
|
{
|
|
// power off + resync
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitUseParsingStrategy;
|
|
}
|
|
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY);
|
|
|
|
if (ulCardType == ASYNC3_CARD)
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
else
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ);
|
|
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0,
|
|
0,
|
|
0,
|
|
2); // TS and T0 expected
|
|
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!ResetDelay = %d\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.ResetDelay));
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitUseParsingStrategy;
|
|
}
|
|
|
|
|
|
// read state and length + TS + T0
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,4,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
continue; // try next card
|
|
}
|
|
|
|
|
|
// contents of read buffer
|
|
// [0] ... state
|
|
// [1] ... length
|
|
// [2] ... TS
|
|
// [3] ... T0
|
|
|
|
// TS
|
|
if (ReadBuffer[2] == CHAR_INV)
|
|
{
|
|
fInverseAtr = TRUE;
|
|
}
|
|
|
|
if (fInverseAtr)
|
|
SCCMN50M_InverseBuffer(&ReadBuffer[3],1);
|
|
|
|
ulHistoricalBytes = ReadBuffer[3] & 0x0F;
|
|
ulPrevStepBytesRead = 2;
|
|
|
|
// T0 codes following TA1 - TD1
|
|
|
|
fAtrParsed = TRUE;
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!Step : Bytes to read = 2\n",
|
|
DRIVER_NAME));
|
|
|
|
|
|
do
|
|
{
|
|
ulNextStepBytesToRead = ulPrevStepBytesRead;
|
|
fTDxSent = FALSE;
|
|
if (ReadBuffer[ulBytesRead - 1 ] & 0x10)
|
|
ulNextStepBytesToRead++;
|
|
if (ReadBuffer[ulBytesRead - 1 ] & 0x20)
|
|
ulNextStepBytesToRead++;
|
|
if (ReadBuffer[ulBytesRead - 1 ] & 0x40)
|
|
ulNextStepBytesToRead++;
|
|
if (ReadBuffer[ulBytesRead - 1 ] & 0x80)
|
|
{
|
|
ulNextStepBytesToRead++;
|
|
fTDxSent = TRUE;
|
|
}
|
|
|
|
if (ulPrevStepBytesRead != 2 &&
|
|
ReadBuffer[ulBytesRead -1 ] & 0x0f)
|
|
{
|
|
fOnlyT0 = FALSE;
|
|
}
|
|
|
|
// -----------------------
|
|
// POWER OFF
|
|
// -----------------------
|
|
// turn power off and get state
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
fAtrParsed = FALSE;
|
|
goto ExitUseParsingStrategy; // try next card
|
|
}
|
|
|
|
// -----------------------
|
|
// POWER ON
|
|
// -----------------------
|
|
// turn on power flag
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY);
|
|
|
|
if (ulCardType == ASYNC3_CARD)
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
else
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ);
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0,
|
|
0,
|
|
0,
|
|
(UCHAR)ulNextStepBytesToRead);
|
|
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
fAtrParsed = FALSE;
|
|
goto ExitUseParsingStrategy; // try next card
|
|
}
|
|
|
|
|
|
|
|
// read state and length + TAx,TBx,TCx,TDx
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!Step : Bytes to read = %ld\n",
|
|
DRIVER_NAME,
|
|
ulNextStepBytesToRead));
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2 + ulNextStepBytesToRead,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
fAtrParsed = FALSE;
|
|
break; // try next card
|
|
}
|
|
if (fInverseAtr)
|
|
SCCMN50M_InverseBuffer(&ReadBuffer[2],ulBytesRead-2);
|
|
MemCpy(bAtrBytesRead,sizeof(bAtrBytesRead),&ReadBuffer[2],ulBytesRead -2);
|
|
|
|
#if DBG
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!read ATR bytes: ",
|
|
DRIVER_NAME));
|
|
for (i = 0;i < ulBytesRead-2;i++)
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%2.2x ",
|
|
bAtrBytesRead[i]));
|
|
SmartcardDebug(DEBUG_ATR,("\n"));
|
|
|
|
#endif
|
|
|
|
ulPrevStepBytesRead = ulBytesRead - 2;
|
|
|
|
} while (fTDxSent == TRUE);
|
|
|
|
|
|
// +++++++++++++++++++++++++++++++++++++++
|
|
// now we know how long the whole ATR is
|
|
// +++++++++++++++++++++++++++++++++++++++
|
|
|
|
// -----------------------
|
|
// POWER OFF
|
|
// -----------------------
|
|
// turn power off and get state
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitUseParsingStrategy;
|
|
}
|
|
|
|
|
|
// -----------------------
|
|
// POWER ON
|
|
// -----------------------
|
|
// turn on power flag
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY);
|
|
|
|
if (ulCardType == ASYNC3_CARD)
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
else
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ);
|
|
|
|
// bug fix : old SAMOS cards have a damaged ATR
|
|
if (bAtrBytesRead[0] == 0x3b &&
|
|
bAtrBytesRead[1] == 0xbf &&
|
|
bAtrBytesRead[2] == 0x11 &&
|
|
bAtrBytesRead[3] == 0x00 &&
|
|
bAtrBytesRead[4] == 0x81 &&
|
|
bAtrBytesRead[5] == 0x31 &&
|
|
bAtrBytesRead[6] == 0x90 &&
|
|
bAtrBytesRead[7] == 0x73 )
|
|
{
|
|
ulHistoricalBytes = 4;
|
|
}
|
|
|
|
|
|
ulNextStepBytesToRead = ulPrevStepBytesRead + ulHistoricalBytes;
|
|
if (fOnlyT0 == FALSE)
|
|
ulNextStepBytesToRead++; // TCK !
|
|
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0,
|
|
0,
|
|
0,
|
|
(UCHAR)ulNextStepBytesToRead);
|
|
|
|
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitUseParsingStrategy; // try next card
|
|
}
|
|
|
|
|
|
// read whole ATR
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!Step : Bytes to read = %ld\n",
|
|
DRIVER_NAME,
|
|
ulNextStepBytesToRead));
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2+ulNextStepBytesToRead,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!Reading of whole ATR failed\n !",
|
|
DRIVER_NAME));
|
|
continue; // try next card
|
|
}
|
|
|
|
// check ATR
|
|
if (ulBytesRead - 2 < MIN_ATR_LEN)
|
|
{
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,2+ulNextStepBytesToRead,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
goto ExitUseParsingStrategy;
|
|
}
|
|
|
|
if (ulBytesRead -2 > ulAtrBufferSize)
|
|
{
|
|
// the ATR is larger then 33 bytes !!!
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitUseParsingStrategy;
|
|
}
|
|
|
|
SCCMN50M_CheckAtrModified(pbAtrBuffer,*pulAtrLength);
|
|
|
|
// pass ATR and ATR length to calling function
|
|
MemCpy(pbAtrBuffer,ulAtrBufferSize,&ReadBuffer[2],ulBytesRead -2);
|
|
*pulAtrLength = ulBytesRead -2;
|
|
|
|
|
|
if (fInverseAtr)
|
|
{
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,INVERSE_DATA);
|
|
SCCMN50M_InverseBuffer(pbAtrBuffer,*pulAtrLength);
|
|
pSmartcardExtension->ReaderExtension->fInverseAtr = TRUE;
|
|
}
|
|
else
|
|
{
|
|
pSmartcardExtension->ReaderExtension->fInverseAtr = FALSE;
|
|
}
|
|
pSmartcardExtension->ReaderExtension->fRawModeNecessary = FALSE;
|
|
break;
|
|
}
|
|
|
|
|
|
ExitUseParsingStrategy:
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier
|
|
= ulOldReadTotalTimeoutMultiplier ;
|
|
SCCMN50M_ClearSCRControlFlags(pSmartcardExtension,IGNORE_PARITY);
|
|
SCCMN50M_ClearCardManHeader(pSmartcardExtension);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
This function performs either a cold or a warm reset depending on
|
|
the fWarmReset parameter .
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_UseColdWarmResetStrategy (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
OUT PULONG pulAtrLength,
|
|
OUT PUCHAR pbAtrBuffer,
|
|
IN ULONG ulAtrBufferSize,
|
|
IN BOOLEAN fWarmReset
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
ULONG ulCardType;
|
|
UCHAR bReadBuffer[SCARD_ATR_LENGTH];
|
|
ULONG ulBytesRead;
|
|
ULONG ulOldReadTotalTimeoutMultiplier;
|
|
|
|
|
|
//DBGBreakPoint();
|
|
// set ReadTotalTimeoutMultiplier to 250ms (9600 * 372/f = initial waiting time)
|
|
ulOldReadTotalTimeoutMultiplier = pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier = 250;
|
|
|
|
if (fWarmReset == FALSE)
|
|
{
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,COLD_RESET,ATR_LEN_ASYNC);
|
|
}
|
|
else
|
|
{
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,0,ATR_LEN_ASYNC);
|
|
}
|
|
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0x02;
|
|
|
|
|
|
for (ulCardType = ASYNC3_CARD;ulCardType <= ASYNC5_CARD;ulCardType++)
|
|
{
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY | CM2_GET_ATR);
|
|
|
|
|
|
if (ulCardType == ASYNC3_CARD)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
("%s!ASYNC_3\n",
|
|
DRIVER_NAME));
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
}
|
|
else
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
("%s!ASYN_5\n",
|
|
DRIVER_NAME));
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ);
|
|
}
|
|
|
|
|
|
status = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
// read state and length
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
continue; // try next card
|
|
}
|
|
|
|
|
|
if (bReadBuffer[1] < MIN_ATR_LEN)
|
|
{
|
|
// read all remaining bytes from the CardMan
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,bReadBuffer[1],&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
|
|
if (bReadBuffer[1] > ulAtrBufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
|
|
// read ATR
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,bReadBuffer[1],pulAtrLength,pbAtrBuffer,ulAtrBufferSize);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
switch (pbAtrBuffer[0])
|
|
{
|
|
case CHAR_INV:
|
|
pSmartcardExtension->ReaderExtension->fRawModeNecessary = FALSE;
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,INVERSE_DATA);
|
|
SCCMN50M_InverseBuffer(pbAtrBuffer,*pulAtrLength);
|
|
pSmartcardExtension->ReaderExtension->fInverseAtr = TRUE;
|
|
break;
|
|
|
|
case CHAR_NORM:
|
|
pSmartcardExtension->ReaderExtension->fRawModeNecessary = FALSE;
|
|
pSmartcardExtension->ReaderExtension->fInverseAtr = FALSE;
|
|
break;
|
|
|
|
default :
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
goto ExitUseColdWarmResetStrategy;
|
|
break;
|
|
|
|
}
|
|
|
|
// the smart card has been powered
|
|
SCCMN50M_CheckAtrModified(pbAtrBuffer,*pulAtrLength);
|
|
MemCpy(pSmartcardExtension->CardCapabilities.ATR.Buffer,
|
|
sizeof(pSmartcardExtension->CardCapabilities.ATR.Buffer),
|
|
pbAtrBuffer,
|
|
*pulAtrLength);
|
|
|
|
pSmartcardExtension->CardCapabilities.ATR.Length = (UCHAR)*pulAtrLength;
|
|
|
|
status = SmartcardUpdateCardCapabilities(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!Invalid ATR received\n",
|
|
DRIVER_NAME));
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
if (SCCMN50M_IsAtrValid(pbAtrBuffer,*pulAtrLength) == FALSE)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_ATR,
|
|
("%s!Invalid ATR received\n",
|
|
DRIVER_NAME));
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
goto ExitUseColdWarmResetStrategy;
|
|
}
|
|
break;
|
|
} // end for
|
|
|
|
ExitUseColdWarmResetStrategy:
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier
|
|
= ulOldReadTotalTimeoutMultiplier ;
|
|
SCCMN50M_ClearSCRControlFlags(pSmartcardExtension,CM2_GET_ATR | IGNORE_PARITY);
|
|
SCCMN50M_ClearCardManHeader(pSmartcardExtension);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This function checks if the received ATR is valid.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
BOOLEAN
|
|
SCCMN50M_IsAtrValid(
|
|
PUCHAR pbAtrBuffer,
|
|
ULONG ulAtrLength
|
|
)
|
|
{
|
|
BOOLEAN fAtrValid = TRUE;
|
|
ULONG ulTD1Offset = 0;
|
|
BOOLEAN fTD1Transmitted = FALSE;
|
|
BOOLEAN fOnlyT0 = FALSE;
|
|
BYTE bXor;
|
|
ULONG ulHistoricalBytes;
|
|
ULONG ulTx2Characters = 0;
|
|
ULONG i;
|
|
|
|
//DBGBreakPoint();
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IsAtrValid : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
// basic checks
|
|
if (ulAtrLength < 2 ||
|
|
(pbAtrBuffer[0] != 0x3F &&
|
|
pbAtrBuffer[0] != 0x3B ) ||
|
|
(pbAtrBuffer[1] & 0xF0) == 0x00 )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
if (pbAtrBuffer[1] & 0x10)
|
|
ulTD1Offset++;
|
|
if (pbAtrBuffer[1] & 0x20)
|
|
ulTD1Offset++;
|
|
if (pbAtrBuffer[1] & 0x40)
|
|
ulTD1Offset++;
|
|
|
|
ulHistoricalBytes = pbAtrBuffer[1] & 0x0F;
|
|
|
|
if (pbAtrBuffer[1] & 0x80) // TD1 in ATR ?
|
|
{
|
|
fTD1Transmitted = TRUE;
|
|
|
|
if ((pbAtrBuffer[2 + ulTD1Offset] & 0x0F) == 0x00) // T0 indicated ?
|
|
fOnlyT0 = TRUE;
|
|
}
|
|
else
|
|
{
|
|
fOnlyT0 = TRUE;
|
|
}
|
|
|
|
|
|
if (fOnlyT0 == FALSE)
|
|
{
|
|
bXor = pbAtrBuffer[1];
|
|
for (i=2;i<ulAtrLength;i++)
|
|
bXor ^= pbAtrBuffer[i];
|
|
|
|
if (bXor != 0x00)
|
|
fAtrValid = FALSE;
|
|
}
|
|
else
|
|
{
|
|
// only T0 protocol is indicated
|
|
if (fTD1Transmitted == TRUE)
|
|
{
|
|
if (pbAtrBuffer[2 + ulTD1Offset] & 0x10)
|
|
ulTx2Characters++;
|
|
if (pbAtrBuffer[2 + ulTD1Offset] & 0x20)
|
|
ulTx2Characters++;
|
|
if (pbAtrBuffer[2 + ulTD1Offset] & 0x40)
|
|
ulTx2Characters++;
|
|
if (ulAtrLength != 2 + ulTD1Offset + 1 + ulTx2Characters + ulHistoricalBytes)
|
|
fAtrValid = FALSE;
|
|
|
|
|
|
}
|
|
else
|
|
{
|
|
if (ulAtrLength != 2 + ulTD1Offset + ulHistoricalBytes)
|
|
fAtrValid = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IsAtrValid : Exit %d\n",
|
|
DRIVER_NAME,fAtrValid)
|
|
);
|
|
return fAtrValid;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This function checks if the received ATR is valid.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID SCCMN50M_CheckAtrModified (
|
|
PUCHAR pbBuffer,
|
|
ULONG ulBufferSize
|
|
)
|
|
{
|
|
UCHAR bNumberHistoricalBytes;
|
|
UCHAR bXorChecksum;
|
|
ULONG i;
|
|
|
|
if (ulBufferSize < 0x09) // mininmum length of a modified ATR
|
|
return ; // ATR is ok
|
|
|
|
|
|
// variant 2
|
|
if (pbBuffer[0] == 0x3b &&
|
|
pbBuffer[1] == 0xbf &&
|
|
pbBuffer[2] == 0x11 &&
|
|
pbBuffer[3] == 0x00 &&
|
|
pbBuffer[4] == 0x81 &&
|
|
pbBuffer[5] == 0x31 &&
|
|
pbBuffer[6] == 0x90 &&
|
|
pbBuffer[7] == 0x73 &&
|
|
ulBufferSize == 13 )
|
|
{
|
|
// correct number of historical bytes
|
|
bNumberHistoricalBytes = 4;
|
|
|
|
pbBuffer[1] &= 0xf0;
|
|
pbBuffer[1] |= bNumberHistoricalBytes;
|
|
|
|
// correct checksum byte
|
|
bXorChecksum = pbBuffer[1];
|
|
for (i=2;i<ulBufferSize-1;i++)
|
|
bXorChecksum ^= pbBuffer[i];
|
|
|
|
pbBuffer[ulBufferSize -1 ] = bXorChecksum;
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!correcting SAMOS ATR (variant 2)\n",
|
|
DRIVER_NAME));
|
|
}
|
|
|
|
|
|
|
|
|
|
// variant 1
|
|
if (pbBuffer[0] == 0x3b &&
|
|
pbBuffer[1] == 0xb4 &&
|
|
pbBuffer[2] == 0x11 &&
|
|
pbBuffer[3] == 0x00 &&
|
|
pbBuffer[4] == 0x81 &&
|
|
pbBuffer[5] == 0x31 &&
|
|
pbBuffer[6] == 0x90 &&
|
|
pbBuffer[7] == 0x73 &&
|
|
ulBufferSize == 13 )
|
|
{
|
|
// correct checksum byte
|
|
bXorChecksum = pbBuffer[1];
|
|
for (i=2;i<ulBufferSize-1;i++)
|
|
bXorChecksum ^= pbBuffer[i];
|
|
|
|
|
|
if (pbBuffer[ulBufferSize -1 ] != bXorChecksum )
|
|
{
|
|
pbBuffer[ulBufferSize -1 ] = bXorChecksum;
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!correcting SAMOS ATR (variant 1)\n",
|
|
DRIVER_NAME));
|
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// variant 3
|
|
if (pbBuffer[0] == 0x3b &&
|
|
pbBuffer[1] == 0xbf &&
|
|
pbBuffer[2] == 0x11 &&
|
|
pbBuffer[3] == 0x00 &&
|
|
pbBuffer[4] == 0x81 &&
|
|
pbBuffer[5] == 0x31 &&
|
|
pbBuffer[6] == 0x90 &&
|
|
pbBuffer[7] == 0x73 &&
|
|
ulBufferSize == 9 )
|
|
{
|
|
// correct number of historical bytes
|
|
bNumberHistoricalBytes = 0;
|
|
|
|
pbBuffer[1] &= 0xf0;
|
|
pbBuffer[1] |= bNumberHistoricalBytes;
|
|
|
|
// correct checksum byte
|
|
bXorChecksum = pbBuffer[1];
|
|
for (i=2;i<ulBufferSize-1;i++)
|
|
bXorChecksum ^= pbBuffer[i];
|
|
|
|
pbBuffer[ulBufferSize -1 ] = bXorChecksum;
|
|
SmartcardDebug(DEBUG_ATR,
|
|
("%s!correcting SAMOS ATR (variant 3)\n",
|
|
DRIVER_NAME));
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_PowerOff (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus = STATUS_SUCCESS;
|
|
UCHAR pReadBuffer[2];
|
|
ULONG ulBytesRead;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!PowerOff: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
// SCR control bytes
|
|
SCCMN50M_ClearSCRControlFlags(pSmartcardExtension,CARD_POWER);
|
|
// card control bytes
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
// header
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,0,1);
|
|
// rx length = 1 because we don't want to receive a status
|
|
|
|
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitSCCMN50M_PowerOff;
|
|
}
|
|
|
|
|
|
|
|
// CardMan echoes a BRK which is recevied in the read functions
|
|
|
|
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,pReadBuffer,sizeof(pReadBuffer));
|
|
|
|
#if 0
|
|
if (DebugStatus != STATUS_SUCCESS)
|
|
SmartcardDebug(
|
|
DEBUG_ERROR,
|
|
( "%s!PowerOffBRK received\n",
|
|
DRIVER_NAME)
|
|
);
|
|
#endif
|
|
|
|
|
|
ExitSCCMN50M_PowerOff:
|
|
if (pSmartcardExtension->ReaderExtension->ulOldCardState == POWERED)
|
|
pSmartcardExtension->ReaderExtension->ulOldCardState = INSERTED;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!PowerOff: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_Transmit(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
|
|
|
|
|
|
switch (pSmartcardExtension->CardCapabilities.Protocol.Selected)
|
|
{
|
|
case SCARD_PROTOCOL_RAW:
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
break;
|
|
|
|
case SCARD_PROTOCOL_T0:
|
|
status = SCCMN50M_TransmitT0(pSmartcardExtension);
|
|
break;
|
|
|
|
case SCARD_PROTOCOL_T1:
|
|
status = SCCMN50M_TransmitT1(pSmartcardExtension);
|
|
break;
|
|
|
|
default:
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
break;
|
|
|
|
}
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_TransmitT0(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
UCHAR bWriteBuffer[MIN_BUFFER_SIZE];
|
|
UCHAR bReadBuffer [MIN_BUFFER_SIZE];
|
|
ULONG ulWriteBufferOffset;
|
|
ULONG ulReadBufferOffset;
|
|
ULONG ulBytesToWrite;
|
|
ULONG ulBytesToRead;
|
|
ULONG ulBytesToWriteThisStep;
|
|
ULONG ulBytesToReadThisStep;
|
|
ULONG ulBytesStillToWrite;
|
|
ULONG ulBytesRead;
|
|
ULONG ulBytesStillToRead;
|
|
BOOLEAN fDataDirectionFromCard;
|
|
BYTE bProcedureByte;
|
|
BYTE bINS;
|
|
BOOLEAN fT0TransferToCard = FALSE;
|
|
BOOLEAN fT0TransferFromCard = FALSE;
|
|
BOOLEAN fSW1SW2Sent = FALSE;
|
|
ULONG ulReadTotalTimeoutMultiplier;
|
|
|
|
ULONG ulStatBytesRead;
|
|
BYTE abStatReadBuffer[2];
|
|
|
|
//SmartcardDebug(DEBUG_TRACE,("TransmitT0 : Enter\n"));
|
|
|
|
//
|
|
// Let the lib build a T=0 packet
|
|
//
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0; // no bytes additionally needed
|
|
status = SmartcardT0Request(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
//
|
|
// This lib detected an error in the data to send.
|
|
//
|
|
// ------------------------------------------
|
|
// ITSEC E2 requirements: clear write buffers
|
|
// ------------------------------------------
|
|
MemSet(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
'\0',
|
|
sizeof(bWriteBuffer));
|
|
MemSet(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
'\0',
|
|
pSmartcardExtension->SmartcardRequest.BufferSize);
|
|
return status;
|
|
}
|
|
|
|
|
|
// increase timeout for T0 Transmission
|
|
ulReadTotalTimeoutMultiplier = pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier =
|
|
pSmartcardExtension->CardCapabilities.T0.WT/1000 + 1500;
|
|
|
|
|
|
|
|
// ##################################
|
|
// TRANSPARENT MODE
|
|
// ##################################
|
|
|
|
ulBytesStillToWrite = ulBytesToWrite = T0_HEADER_LEN + pSmartcardExtension->T0.Lc;
|
|
ulBytesStillToRead = ulBytesToRead = pSmartcardExtension->T0.Le;
|
|
if (pSmartcardExtension->T0.Lc)
|
|
fT0TransferToCard = TRUE;
|
|
if (pSmartcardExtension->T0.Le)
|
|
fT0TransferFromCard = TRUE;
|
|
|
|
|
|
|
|
|
|
// copy data to the write buffer
|
|
/*
|
|
SmartcardDebug(DEBUG_TRACE,("CLA = %x INS = %x P1 = %x P2 = %X L = %x\n",
|
|
pSmartcardExtension->SmartcardRequest.Buffer[0],
|
|
pSmartcardExtension->SmartcardRequest.Buffer[1],
|
|
pSmartcardExtension->SmartcardRequest.Buffer[2],
|
|
pSmartcardExtension->SmartcardRequest.Buffer[3],
|
|
pSmartcardExtension->SmartcardRequest.Buffer[4]));
|
|
*/
|
|
|
|
MemCpy(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
pSmartcardExtension->SmartcardRequest.Buffer,
|
|
ulBytesToWrite);
|
|
|
|
bINS = bWriteBuffer[1];
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
SCCMN50M_InverseBuffer(bWriteBuffer,ulBytesToWrite);
|
|
}
|
|
|
|
status = SCCMN50M_EnterTransparentMode(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
|
|
// STEP 1 : write config + header to enter transparent mode
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0, // Tx control
|
|
0, // Tx length
|
|
0, // Rx control
|
|
0); // Rx length
|
|
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
0,
|
|
NULL);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->fTransparentMode = TRUE;
|
|
|
|
// if the inserted card uses inverse convention , we must now switch the COM port
|
|
// to odd parity
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = ODD_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = SERIAL_DATABITS_8;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
}
|
|
ulWriteBufferOffset = 0;
|
|
ulReadBufferOffset = 0;
|
|
|
|
|
|
// STEP 2 : write CLA INS P1 P2 Lc
|
|
|
|
ulBytesToWriteThisStep = 5;
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
ulBytesToWriteThisStep,
|
|
bWriteBuffer+ulWriteBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
ulWriteBufferOffset += ulBytesToWriteThisStep;
|
|
ulBytesStillToWrite -= ulBytesToWriteThisStep;
|
|
|
|
|
|
|
|
|
|
|
|
// STEP 2 : read procedure byte
|
|
do
|
|
{
|
|
do
|
|
{
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,1,&ulBytesRead,&bProcedureByte,sizeof(bProcedureByte));
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
SCCMN50M_InverseBuffer(&bProcedureByte,ulBytesRead);
|
|
}
|
|
|
|
//SmartcardDebug(DEBUG_TRACE,("Procedure byte = %x\n",bProcedureByte));
|
|
//SmartcardDebug(DEBUG_TRACE,("waiting time = %x\n",pSmartcardExtension->CardCapabilities.T0.WT));
|
|
if (bProcedureByte == 0x60)
|
|
{
|
|
// ISO 7816-3 :
|
|
// This byte is sent by the card to reset the work waiting time and to anticipate
|
|
// a subsequent procedure byte
|
|
// => we do nothing here
|
|
}
|
|
} while (bProcedureByte == 0x60);
|
|
|
|
|
|
// check for ACK
|
|
if ((bProcedureByte & 0xFE) == (bINS & 0xFE) )
|
|
{
|
|
if (fT0TransferToCard)
|
|
{
|
|
ulBytesToWriteThisStep = ulBytesStillToWrite;
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
ulBytesToWriteThisStep,
|
|
bWriteBuffer+ulWriteBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
ulWriteBufferOffset += ulBytesToWriteThisStep;
|
|
ulBytesStillToWrite -= ulBytesToWriteThisStep;
|
|
|
|
}
|
|
if (fT0TransferFromCard)
|
|
{
|
|
ulBytesToReadThisStep = ulBytesStillToRead;
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,ulBytesToReadThisStep,&ulBytesRead,bReadBuffer + ulReadBufferOffset,sizeof(bReadBuffer)-ulReadBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
SCCMN50M_InverseBuffer(bReadBuffer+ulReadBufferOffset,ulBytesRead);
|
|
}
|
|
|
|
ulReadBufferOffset += ulBytesRead;
|
|
ulBytesStillToRead -= ulBytesRead;
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
// check for NAK
|
|
else if ( (~bProcedureByte & 0xFE) == (bINS & 0xFE))
|
|
{
|
|
if (fT0TransferToCard)
|
|
{
|
|
ulBytesToWriteThisStep = 1;
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
ulBytesToWriteThisStep,
|
|
bWriteBuffer+ulWriteBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
ulWriteBufferOffset += ulBytesToWriteThisStep;
|
|
ulBytesStillToWrite -= ulBytesToWriteThisStep;
|
|
|
|
}
|
|
if (fT0TransferFromCard)
|
|
{
|
|
ulBytesToReadThisStep = 1;
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,ulBytesToReadThisStep,&ulBytesRead,bReadBuffer + ulReadBufferOffset,sizeof(bReadBuffer)-ulReadBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
SCCMN50M_InverseBuffer(bReadBuffer+ulReadBufferOffset,ulBytesRead);
|
|
}
|
|
|
|
ulReadBufferOffset += ulBytesRead;
|
|
ulBytesStillToRead -= ulBytesRead;
|
|
}
|
|
}
|
|
// check for SW1
|
|
else if ( (bProcedureByte > 0x60 && bProcedureByte <= 0x6F) ||
|
|
(bProcedureByte >= 0x90 && bProcedureByte <= 0x9F) )
|
|
{
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
bReadBuffer[ulReadBufferOffset] = bProcedureByte;
|
|
ulReadBufferOffset++;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,1,&ulBytesRead,bReadBuffer+ulReadBufferOffset,sizeof(bReadBuffer)-ulReadBufferOffset);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
SCCMN50M_InverseBuffer(bReadBuffer+ulReadBufferOffset,ulBytesRead);
|
|
}
|
|
ulReadBufferOffset += ulBytesRead;
|
|
fSW1SW2Sent = TRUE;
|
|
}
|
|
else
|
|
{
|
|
status = STATUS_UNSUCCESSFUL;
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
|
|
}while (!fSW1SW2Sent);
|
|
|
|
|
|
// copy received bytes
|
|
MemCpy(pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferSize,
|
|
bReadBuffer,
|
|
ulReadBufferOffset);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = ulReadBufferOffset;
|
|
|
|
|
|
// let the lib copy the received bytes to the user buffer
|
|
status = SmartcardT0Reply(pSmartcardExtension);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ExitTransparentTransmitT0:
|
|
// ------------------------------------------
|
|
// ITSEC E2 requirements: clear write buffers
|
|
// ------------------------------------------
|
|
MemSet(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
'\0',
|
|
sizeof(bWriteBuffer));
|
|
MemSet(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
'\0',
|
|
pSmartcardExtension->SmartcardRequest.BufferSize);
|
|
|
|
DebugStatus = SCCMN50M_ExitTransparentMode(pSmartcardExtension);
|
|
pSmartcardExtension->ReaderExtension->fTransparentMode = FALSE;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier = ulReadTotalTimeoutMultiplier;
|
|
|
|
// to be sure that the new settings take effect
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = 250;
|
|
DebugStatus = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
if (NT_SUCCESS(DebugStatus))
|
|
{
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulStatBytesRead,abStatReadBuffer,sizeof(abStatReadBuffer));
|
|
}
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_TransmitT1(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
ULONG ulBytesToWrite;
|
|
UCHAR bWriteBuffer [256 + T1_HEADER_LEN + MAX_EDC_LEN];
|
|
UCHAR bReadBuffer [256 + T1_HEADER_LEN + MAX_EDC_LEN];
|
|
ULONG ulBytesRead;
|
|
ULONG ulBytesStillToRead;
|
|
|
|
/*
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
("%s!CWT = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->CardCapabilities.T1.CWT/1000)
|
|
);
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
("%s!BWT = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->CardCapabilities.T1.BWT/1000);
|
|
);
|
|
*/
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant =
|
|
pSmartcardExtension->CardCapabilities.T1.BWT/1000;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier =
|
|
pSmartcardExtension->CardCapabilities.T1.CWT/1000;
|
|
/*
|
|
SmartcardDebug(DEBUG_TRACE,("%s!ReadTotalTimeoutConstant = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant));
|
|
|
|
SmartcardDebug(DEBUG_TRACE,("%s!ReadTotalTimeoutMultiplier = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier));
|
|
*/
|
|
|
|
// set T1 protocol flag for CardMan
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_T1);
|
|
|
|
if (pSmartcardExtension->CardCapabilities.T1.EDC == T1_CRC_CHECK)
|
|
{
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_CRC);
|
|
}
|
|
|
|
|
|
do
|
|
{
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0; // no bytes additionally needed
|
|
|
|
|
|
status = SmartcardT1Request(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitTransmitT1;
|
|
}
|
|
|
|
ulBytesToWrite = pSmartcardExtension->SmartcardRequest.BufferLength;
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0, // Tx conrol
|
|
(UCHAR)ulBytesToWrite, // Tx length
|
|
0, // Rx control
|
|
T1_HEADER_LEN); // Rx length
|
|
|
|
|
|
|
|
if (sizeof(bWriteBuffer) < ulBytesToWrite)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitTransmitT1;
|
|
}
|
|
|
|
// copy data to the write buffer
|
|
MemCpy(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
pSmartcardExtension->SmartcardRequest.Buffer,
|
|
ulBytesToWrite);
|
|
|
|
|
|
// write data to card
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
ulBytesToWrite,
|
|
bWriteBuffer);
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
|
|
// read CardMan Header
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
ulBytesStillToRead = bReadBuffer[1];
|
|
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
ulBytesStillToRead,
|
|
&ulBytesRead,
|
|
bReadBuffer,
|
|
sizeof(bReadBuffer));
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
if (bReadBuffer[1] == T1_WTX_REQUEST)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant =
|
|
(ULONG)(1000 +((pSmartcardExtension->CardCapabilities.T1.BWT*bReadBuffer[3])/1000));
|
|
SmartcardDebug(DEBUG_PROTOCOL,("%s!ReadTotalTimeoutConstant = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant));
|
|
}
|
|
else
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant =
|
|
pSmartcardExtension->CardCapabilities.T1.BWT/1000;
|
|
SmartcardDebug(DEBUG_PROTOCOL,("%s!ReadTotalTimeoutConstant = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant));
|
|
}
|
|
// copy received bytes
|
|
MemCpy(pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferSize,
|
|
bReadBuffer,
|
|
ulBytesRead);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = ulBytesRead;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
// reset serial timeout
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!reseting timeout constant\n",
|
|
DRIVER_NAME)
|
|
);
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant =
|
|
pSmartcardExtension->CardCapabilities.T1.BWT/1000;
|
|
SmartcardDebug(DEBUG_PROTOCOL,("%s!ReadTotalTimeoutConstant = %ld(ms)\n",
|
|
DRIVER_NAME,
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant));
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0L;
|
|
}
|
|
|
|
// bug fix for smclib
|
|
if (pSmartcardExtension->T1.State == T1_IFS_RESPONSE &&
|
|
pSmartcardExtension->T1.OriginalState == T1_I_BLOCK)
|
|
{
|
|
pSmartcardExtension->T1.State = T1_I_BLOCK;
|
|
}
|
|
|
|
status = SmartcardT1Reply(pSmartcardExtension);
|
|
}
|
|
while (status == STATUS_MORE_PROCESSING_REQUIRED);
|
|
|
|
|
|
|
|
ExitTransmitT1:
|
|
// ------------------------------------------
|
|
// ITSEC E2 requirements: clear write buffers
|
|
// ------------------------------------------
|
|
MemSet(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
'\0',
|
|
sizeof(bWriteBuffer));
|
|
MemSet(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
'\0',
|
|
pSmartcardExtension->SmartcardRequest.BufferSize);
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier =
|
|
DEFAULT_READ_TOTAL_TIMEOUT_MULTIPLIER;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant =
|
|
DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_InitializeSmartcardExtension(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulDeviceInstance
|
|
)
|
|
{
|
|
// ==================================
|
|
// Fill the Vendor_Attr structure
|
|
// ==================================
|
|
MemCpy(pSmartcardExtension->VendorAttr.VendorName.Buffer,
|
|
sizeof(pSmartcardExtension->VendorAttr.VendorName.Buffer),
|
|
ATTR_VENDOR_NAME,
|
|
sizeof(ATTR_VENDOR_NAME)
|
|
);
|
|
|
|
//
|
|
// Length of vendor name
|
|
//
|
|
pSmartcardExtension->VendorAttr.VendorName.Length = sizeof(ATTR_VENDOR_NAME);
|
|
|
|
|
|
//
|
|
// Version number
|
|
//
|
|
pSmartcardExtension->VendorAttr.IfdVersion.BuildNumber = IFD_NT_BUILDNUMBER_CARDMAN;
|
|
pSmartcardExtension->VendorAttr.IfdVersion.VersionMinor = IFD_NT_VERSIONMINOR_CARDMAN;
|
|
pSmartcardExtension->VendorAttr.IfdVersion.VersionMajor = IFD_NT_VERSIONMAJOR_CARDMAN;
|
|
|
|
|
|
MemCpy(pSmartcardExtension->VendorAttr.IfdType.Buffer,
|
|
sizeof(pSmartcardExtension->VendorAttr.IfdType.Buffer),
|
|
ATTR_IFD_TYPE_CM,
|
|
sizeof(ATTR_IFD_TYPE_CM));
|
|
|
|
//
|
|
// Length of reader name
|
|
//
|
|
pSmartcardExtension->VendorAttr.IfdType.Length = sizeof(ATTR_IFD_TYPE_CM);
|
|
|
|
|
|
|
|
//
|
|
// Unit number which is zero based
|
|
//
|
|
pSmartcardExtension->VendorAttr.UnitNo = ulDeviceInstance;
|
|
|
|
|
|
|
|
// ================================================
|
|
// Fill the SCARD_READER_CAPABILITIES structure
|
|
// ===============================================
|
|
//
|
|
// Supported protoclols by the reader
|
|
//
|
|
|
|
pSmartcardExtension->ReaderCapabilities.SupportedProtocols = SCARD_PROTOCOL_T1 | SCARD_PROTOCOL_T0;
|
|
|
|
|
|
|
|
|
|
//
|
|
// Reader type serial, keyboard, ....
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.ReaderType = SCARD_READER_TYPE_SERIAL;
|
|
|
|
//
|
|
// Mechanical characteristics like swallows etc.
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.MechProperties = 0;
|
|
|
|
|
|
//
|
|
// Current state of the reader
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_UNKNOWN;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
// Data Rate
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.DataRate.Default =
|
|
pSmartcardExtension->ReaderCapabilities.DataRate.Max =
|
|
dataRatesSupported[0];
|
|
|
|
|
|
// reader could support higher data rates
|
|
pSmartcardExtension->ReaderCapabilities.DataRatesSupported.List =
|
|
dataRatesSupported;
|
|
pSmartcardExtension->ReaderCapabilities.DataRatesSupported.Entries =
|
|
sizeof(dataRatesSupported) / sizeof(dataRatesSupported[0]);
|
|
|
|
|
|
|
|
|
|
//
|
|
// CLKFrequency
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.CLKFrequency.Default =
|
|
pSmartcardExtension->ReaderCapabilities.CLKFrequency.Max =
|
|
CLKFrequenciesSupported[0];
|
|
|
|
|
|
pSmartcardExtension->ReaderCapabilities.CLKFrequenciesSupported.List =
|
|
CLKFrequenciesSupported;
|
|
pSmartcardExtension->ReaderCapabilities.CLKFrequenciesSupported.Entries =
|
|
sizeof(CLKFrequenciesSupported) / sizeof(CLKFrequenciesSupported[0]);
|
|
|
|
|
|
//pSmartcardExtension->ReaderCapabilities.CLKFrequency.Default = 3571; //3.571 MHz
|
|
//pSmartcardExtension->ReaderCapabilities.CLKFrequency.Max = 3571; //3.571 MHz
|
|
|
|
//
|
|
// MaxIFSD
|
|
//
|
|
pSmartcardExtension->ReaderCapabilities.MaxIFSD = ATTR_MAX_IFSD_CARDMAN_II;
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
StrSet(PUCHAR Buffer,ULONG BufferSize,UCHAR Pattern)
|
|
{
|
|
ULONG i;
|
|
|
|
for (i=0;i < BufferSize;i++)
|
|
Buffer[i] = Pattern;
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
StrCpy(PUCHAR pszDestination,ULONG DestinationLen,PUCHAR pszSrc)
|
|
{
|
|
ULONG Len;
|
|
ULONG SrcLen;
|
|
ULONG i;
|
|
|
|
StrSet(pszDestination, DestinationLen,'\0');
|
|
SrcLen = StrLen(pszSrc);
|
|
|
|
if (DestinationLen - 1 < SrcLen)
|
|
{
|
|
Len = DestinationLen - 1;
|
|
}
|
|
else
|
|
{
|
|
Len = SrcLen;
|
|
}
|
|
|
|
|
|
for (i = 0; i < Len; i++)
|
|
{
|
|
pszDestination[i] = pszSrc[i];
|
|
if (pszSrc[i] == '\0')
|
|
break;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
ULONG
|
|
StrLen (PUCHAR pszString)
|
|
{
|
|
ULONG Len = 0;
|
|
|
|
while ( *(pszString+Len) != '\0')
|
|
Len++;
|
|
|
|
return(Len);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
StrCat(PUCHAR pszDestination,ULONG DestinationLen,PUCHAR pszSrc)
|
|
{
|
|
ULONG Len;
|
|
ULONG i;
|
|
ULONG SrcLen;
|
|
ULONG DestLen;
|
|
|
|
SrcLen = StrLen(pszSrc);
|
|
DestLen = StrLen(pszDestination);
|
|
|
|
|
|
if (StrLen(pszDestination)>=DestinationLen)
|
|
return;
|
|
|
|
|
|
if ((DestinationLen-DestLen-1) < SrcLen)
|
|
{
|
|
Len = DestinationLen-DestLen-1;
|
|
}
|
|
else
|
|
{
|
|
Len = SrcLen;
|
|
}
|
|
|
|
|
|
for (i=0; i < Len; i++)
|
|
{
|
|
pszDestination[DestLen+i] = pszSrc[i];
|
|
if (pszSrc[i] == '\0')
|
|
break;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
MemSet(PUCHAR pBuffer,
|
|
ULONG ulBufferSize,
|
|
UCHAR ucPattern,
|
|
ULONG ulCount)
|
|
{
|
|
ULONG i;
|
|
|
|
for (i=0; i<ulCount;i++)
|
|
{
|
|
if (i >= ulBufferSize)
|
|
break;
|
|
pBuffer[i] = ucPattern;
|
|
}
|
|
|
|
return ;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
MemCpy(PUCHAR pDestination,
|
|
ULONG ulDestinationLen,
|
|
PUCHAR pSource,
|
|
ULONG ulCount)
|
|
{
|
|
ULONG i = 0;
|
|
while ( ulCount-- && ulDestinationLen-- )
|
|
{
|
|
pDestination[i] = pSource[i];
|
|
i++;
|
|
}
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_UpdateCurrentStateThread(
|
|
IN PVOID Context
|
|
)
|
|
|
|
{
|
|
PDEVICE_EXTENSION deviceExtension = Context;
|
|
PSMARTCARD_EXTENSION smartcardExtension;
|
|
NTSTATUS status;
|
|
LONG lRetry;
|
|
KIRQL oldIrql;
|
|
LONG ulFailures;
|
|
BOOLEAN fPriorityIncreased;
|
|
LONG lOldPriority;
|
|
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!UpdateCurrentStateThread started\n",DRIVER_NAME));
|
|
|
|
ulFailures = 0;
|
|
smartcardExtension = &deviceExtension->SmartcardExtension;
|
|
|
|
//
|
|
// Increase priority for first loop,
|
|
// because state of card must be known for resource manager
|
|
//
|
|
fPriorityIncreased=TRUE;
|
|
lOldPriority=KeSetPriorityThread(KeGetCurrentThread(),HIGH_PRIORITY);
|
|
|
|
do
|
|
{
|
|
KeWaitForSingleObject(&smartcardExtension->ReaderExtension->CardManIOMutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
if ( smartcardExtension->ReaderExtension->TimeToTerminateThread )
|
|
{
|
|
KeReleaseMutex(&smartcardExtension->ReaderExtension->CardManIOMutex,FALSE);
|
|
smartcardExtension->ReaderExtension->TimeToTerminateThread = FALSE;
|
|
PsTerminateSystemThread( STATUS_SUCCESS );
|
|
}
|
|
|
|
lRetry = 1;
|
|
|
|
|
|
|
|
do
|
|
{
|
|
//SmartcardDebug(DEBUG_TRACE,( "%s!*.",DRIVER_NAME));
|
|
|
|
status=SCCMN50M_UpdateCurrentState(smartcardExtension);
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
lRetry--;
|
|
}
|
|
}
|
|
while (lRetry >= 0);
|
|
|
|
if (lRetry < 0)
|
|
{
|
|
ulFailures++;
|
|
if (ulFailures == 1)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
( "%s!CardMan removed\n",
|
|
DRIVER_NAME)
|
|
);
|
|
// issue a card removal event if reader has been removed
|
|
if (smartcardExtension->ReaderExtension->ulOldCardState == INSERTED ||
|
|
smartcardExtension->ReaderExtension->ulOldCardState == POWERED )
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
( "%s!issuing card removal event\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
SCCMN50M_CompleteCardTracking(smartcardExtension);
|
|
smartcardExtension->ReaderExtension->SyncParameters.fCardPowerRequested = TRUE;
|
|
|
|
smartcardExtension->ReaderExtension->ulNewCardState = REMOVED;
|
|
smartcardExtension->ReaderExtension->ulOldCardState = smartcardExtension->ReaderExtension->ulNewCardState;
|
|
smartcardExtension->ReaderCapabilities.CurrentState = SCARD_ABSENT;
|
|
smartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
smartcardExtension->CardCapabilities.ATR.Length = 0;
|
|
|
|
SCCMN50M_ClearCardControlFlags(smartcardExtension,ALL_FLAGS);
|
|
smartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0;
|
|
smartcardExtension->ReaderExtension->CardManConfig.ResetDelay = 0;
|
|
}
|
|
}
|
|
if (ulFailures == 3)
|
|
{
|
|
// remove the device and terminate this thread
|
|
if (KeReadStateEvent(&deviceExtension->SerialCloseDone) == 0l)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
( "%s!closing serial driver\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
SCCMN50M_CloseSerialDriver(smartcardExtension->OsData->DeviceObject);
|
|
|
|
|
|
KeReleaseMutex(&smartcardExtension->ReaderExtension->CardManIOMutex,FALSE);
|
|
smartcardExtension->ReaderExtension->TimeToTerminateThread = FALSE;
|
|
smartcardExtension->ReaderExtension->ThreadObjectPointer = NULL;
|
|
PsTerminateSystemThread( STATUS_SUCCESS );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ulFailures = 0;
|
|
}
|
|
|
|
KeReleaseMutex(&smartcardExtension->ReaderExtension->CardManIOMutex,FALSE);
|
|
|
|
if (fPriorityIncreased)
|
|
{
|
|
fPriorityIncreased=FALSE;
|
|
KeSetPriorityThread(KeGetCurrentThread(),lOldPriority);
|
|
|
|
//
|
|
// Lower ourselves down just at tad so that we compete a
|
|
// little less.
|
|
//
|
|
KeSetBasePriorityThread(KeGetCurrentThread(),-1);
|
|
}
|
|
|
|
//SmartcardDebug(DEBUG_TRACE,( "...#\n"));
|
|
|
|
Wait (smartcardExtension,500 * ms_);
|
|
}
|
|
while (TRUE);
|
|
}
|
|
|
|
|
|
|
|
|
|
NTSTATUS SCCMN50M_UpdateCurrentState(
|
|
IN PSMARTCARD_EXTENSION smartcardExtension
|
|
)
|
|
{
|
|
NTSTATUS NTStatus;
|
|
UCHAR pbReadBuffer[2];
|
|
ULONG ulBytesRead;
|
|
BOOLEAN fCardStateChanged;
|
|
|
|
fCardStateChanged = FALSE;
|
|
|
|
|
|
SCCMN50M_ClearCardManHeader(smartcardExtension);
|
|
|
|
smartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = 250;
|
|
NTStatus = SCCMN50M_WriteCardMan(smartcardExtension,0,NULL);
|
|
smartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
if (NT_SUCCESS(NTStatus))
|
|
{
|
|
NTStatus = SCCMN50M_ReadCardMan(smartcardExtension,2,&ulBytesRead,pbReadBuffer,sizeof(pbReadBuffer));
|
|
if (ulBytesRead == 0x02 && // two bytes must have benn received
|
|
(pbReadBuffer[0] & 0x0F) && // at least one version bit must be set
|
|
((pbReadBuffer[0] & 0x09) == 0x00)) // Bit 0 and Bit 3 must be 0
|
|
{
|
|
if ((pbReadBuffer[0] & 0x04) == 0x04 &&
|
|
(pbReadBuffer[0] & 0x02) == 0x02)
|
|
smartcardExtension->ReaderExtension->ulNewCardState = INSERTED;
|
|
|
|
if ((pbReadBuffer[0] & 0x04) == 0x00 &&
|
|
(pbReadBuffer[0] & 0x02) == 0x02)
|
|
smartcardExtension->ReaderExtension->ulNewCardState = REMOVED;
|
|
|
|
if ((pbReadBuffer[0] & 0x04) == 0x04 &&
|
|
(pbReadBuffer[0] & 0x02) == 0x00)
|
|
smartcardExtension->ReaderExtension->ulNewCardState = POWERED;
|
|
|
|
|
|
//SmartcardDebug(DEBUG_TRACE,("old %x ",smartcardExtension->ReaderExtension->ulOldCardState ));
|
|
//SmartcardDebug(DEBUG_TRACE,("new %x\n",smartcardExtension->ReaderExtension->ulNewCardState ));
|
|
|
|
|
|
|
|
if (smartcardExtension->ReaderExtension->ulNewCardState == INSERTED &&
|
|
smartcardExtension->ReaderExtension->ulOldCardState == POWERED )
|
|
{
|
|
// card has been removed and reinserted within 500ms
|
|
fCardStateChanged = TRUE;
|
|
SmartcardDebug(DEBUG_DRIVER,( "%s!Smartcard removed and reinserted\n",DRIVER_NAME));
|
|
smartcardExtension->ReaderExtension->ulOldCardState = REMOVED;
|
|
smartcardExtension->ReaderCapabilities.CurrentState = SCARD_ABSENT;
|
|
smartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
|
|
// clear any cardspecific data
|
|
smartcardExtension->CardCapabilities.ATR.Length = 0;
|
|
SCCMN50M_ClearCardControlFlags(smartcardExtension,ALL_FLAGS);
|
|
smartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0;
|
|
smartcardExtension->ReaderExtension->CardManConfig.ResetDelay = 0;
|
|
}
|
|
|
|
|
|
if (smartcardExtension->ReaderExtension->ulNewCardState == REMOVED &&
|
|
(smartcardExtension->ReaderExtension->ulOldCardState == UNKNOWN ||
|
|
smartcardExtension->ReaderExtension->ulOldCardState == INSERTED ||
|
|
smartcardExtension->ReaderExtension->ulOldCardState == POWERED ) )
|
|
{
|
|
// card has been removed
|
|
fCardStateChanged = TRUE;
|
|
SmartcardDebug(DEBUG_DRIVER,( "%s!Smartcard removed\n",DRIVER_NAME));
|
|
smartcardExtension->ReaderExtension->ulOldCardState = smartcardExtension->ReaderExtension->ulNewCardState;
|
|
smartcardExtension->ReaderCapabilities.CurrentState = SCARD_ABSENT;
|
|
smartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
|
|
// clear any cardspecific data
|
|
smartcardExtension->CardCapabilities.ATR.Length = 0;
|
|
SCCMN50M_ClearCardControlFlags(smartcardExtension,ALL_FLAGS);
|
|
smartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0;
|
|
smartcardExtension->ReaderExtension->CardManConfig.ResetDelay = 0;
|
|
}
|
|
|
|
|
|
|
|
if (smartcardExtension->ReaderExtension->ulNewCardState == INSERTED &&
|
|
(smartcardExtension->ReaderExtension->ulOldCardState == UNKNOWN ||
|
|
smartcardExtension->ReaderExtension->ulOldCardState == REMOVED ) )
|
|
{
|
|
// card has been inserted
|
|
fCardStateChanged = TRUE;
|
|
SmartcardDebug(DEBUG_DRIVER,( "%s!Smartcard inserted\n",DRIVER_NAME));
|
|
smartcardExtension->ReaderExtension->ulOldCardState = smartcardExtension->ReaderExtension->ulNewCardState;
|
|
smartcardExtension->ReaderCapabilities.CurrentState = SCARD_SWALLOWED;
|
|
smartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
}
|
|
|
|
|
|
// state after reset of the PC (only for CardMan Power+ possible)
|
|
if (smartcardExtension->ReaderExtension->ulNewCardState == POWERED &&
|
|
smartcardExtension->ReaderExtension->ulOldCardState == UNKNOWN )
|
|
{
|
|
// card has been inserted
|
|
fCardStateChanged = TRUE;
|
|
SmartcardDebug(DEBUG_DRIVER,( "%s!Smartcard inserted (and powered)\n",DRIVER_NAME));
|
|
smartcardExtension->ReaderExtension->ulOldCardState = smartcardExtension->ReaderExtension->ulNewCardState;
|
|
smartcardExtension->ReaderCapabilities.CurrentState = SCARD_SWALLOWED;
|
|
smartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
}
|
|
|
|
if (smartcardExtension->ReaderExtension->ulNewCardState == POWERED &&
|
|
smartcardExtension->ReaderExtension->ulOldCardState == INSERTED )
|
|
{
|
|
smartcardExtension->ReaderExtension->ulOldCardState = smartcardExtension->ReaderExtension->ulNewCardState;
|
|
}
|
|
|
|
|
|
// complete IOCTL_SMARTCARD_IS_ABSENT or IOCTL_SMARTCARD_IS_PRESENT
|
|
if (fCardStateChanged == TRUE &&
|
|
smartcardExtension->OsData->NotificationIrp )
|
|
{
|
|
SCCMN50M_CompleteCardTracking(smartcardExtension);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return NTStatus;
|
|
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS Wait (PSMARTCARD_EXTENSION pSmartcardExtension,ULONG ulMilliseconds)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
LARGE_INTEGER WaitTime;
|
|
|
|
|
|
WaitTime = RtlConvertLongToLargeInteger(ulMilliseconds * WAIT_MS);
|
|
KeDelayExecutionThread(KernelMode,FALSE,&WaitTime);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_SetSCRControlFlags(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN UCHAR Flags
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.SCRControl |= Flags;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_ClearSCRControlFlags(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN UCHAR Flags
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.SCRControl &= ~Flags;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_SetCardControlFlags(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN UCHAR Flags
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardControl |= Flags;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_ClearCardControlFlags(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN UCHAR Flags
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardControl &= ~Flags;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_ClearCardManHeader(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxControl = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxLength = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxControl = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxLength = 0x00;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_SetCardManHeader(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN UCHAR TxControl,
|
|
IN UCHAR TxLength,
|
|
IN UCHAR RxControl,
|
|
IN UCHAR RxLength
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxControl = TxControl;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxLength = TxLength;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxControl = RxControl;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxLength = RxLength;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_WriteCardMan (
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBytesToWrite,
|
|
IN PUCHAR pbWriteBuffer
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
PSERIAL_STATUS pSerialStatus;
|
|
|
|
|
|
// ===============================================
|
|
// Set up timeouts for following read operation
|
|
// ===============================================
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_TIMEOUTS;
|
|
|
|
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts,
|
|
sizeof(SERIAL_TIMEOUTS));
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_TIMEOUTS);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
|
|
/*
|
|
SmartcardDebug(DEBUG_TRACE,("ReadTotalTimeoutMultiplier = %ld\n",
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier));
|
|
SmartcardDebug(DEBUG_TRACE,("ReadTotalTimeoutConstant = %ld\n",
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant));
|
|
*/
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// ===============================================
|
|
// write to the CardMan
|
|
// ===============================================
|
|
DebugStatus = SCCMN50M_SetWrite(pSmartcardExtension,ulBytesToWrite,pbWriteBuffer);
|
|
|
|
|
|
// add pseudoboost (0x00) to write buffer for CardManII
|
|
if (pSmartcardExtension->ReaderExtension->fTransparentMode == FALSE )
|
|
{
|
|
pSmartcardExtension->SmartcardRequest.Buffer[pSmartcardExtension->SmartcardRequest.BufferLength] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength++;
|
|
}
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
|
|
|
|
// overwrite write buffer with '@'
|
|
RtlFillMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferLength,
|
|
'@');
|
|
|
|
|
|
|
|
// ===============================================
|
|
// error checking
|
|
// ===============================================
|
|
DebugStatus = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
|
|
pSerialStatus = (PSERIAL_STATUS) pSmartcardExtension->SmartcardReply.Buffer;
|
|
if (pSerialStatus->Errors || NT_ERROR(status))
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialErrors = pSerialStatus->Errors;
|
|
if (!pSmartcardExtension->ReaderExtension->fTransparentMode )
|
|
DebugStatus = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
goto ExitSCCMN50M_WriteCardMan;
|
|
}
|
|
|
|
|
|
|
|
|
|
ExitSCCMN50M_WriteCardMan:
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!WriteCardMan: Failed, exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS SCCMN50M_ResyncCardManI (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
// SmartcardDebug(DEBUG_TRACE,("%s!ResyncCardManI: Enter\n",DRIVER_NAME))
|
|
|
|
// clear error flags
|
|
pSmartcardExtension->ReaderExtension->SerialErrors = 0;
|
|
|
|
|
|
// clear any pending errors
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
|
|
// clear COM buffers
|
|
status = SCCMN50M_PurgeComm(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_PurgeComm failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
|
|
|
|
|
|
// ####################################################################
|
|
// set break
|
|
if (!pSmartcardExtension->ReaderExtension->fTransparentMode)
|
|
{
|
|
status = SCCMN50M_SetBRK(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SetBreak failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
}
|
|
|
|
|
|
// wait 1ms
|
|
Wait(pSmartcardExtension,1 * ms_);
|
|
|
|
// clear RTS
|
|
status = SCCMN50M_ClearRTS(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_ClearRTS failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
// wait 2ms
|
|
Wait(pSmartcardExtension,2 * ms_);
|
|
|
|
|
|
// set RTS
|
|
status = SCCMN50M_SetRTS(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_SetRTS failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
// wait 1ms
|
|
Wait(pSmartcardExtension,1 * ms_);
|
|
|
|
|
|
|
|
// clear break
|
|
|
|
if (!pSmartcardExtension->ReaderExtension->fTransparentMode)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->BreakSet = FALSE;
|
|
status = SCCMN50M_ClearBRK(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("ClearBreak failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
}
|
|
|
|
// ####################################################################
|
|
|
|
// next write operation must send config data
|
|
pSmartcardExtension->ReaderExtension->NoConfig = FALSE;
|
|
|
|
// clear COM buffers
|
|
status = SCCMN50M_PurgeComm(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
// clear any pending errors
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitSCCMN50M_ResyncCardManI;
|
|
}
|
|
|
|
|
|
ExitSCCMN50M_ResyncCardManI:
|
|
//SmartcardDebug(DEBUG_TRACE,("%s!ResyncCardManI: Exit %lx\n",DRIVER_NAME,status))
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NTSTATUS SCCMN50M_ResyncCardManII (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ResyncCardManII: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
// clear error flags
|
|
pSmartcardExtension->ReaderExtension->SerialErrors = 0;
|
|
|
|
|
|
// clear any pending errors
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
|
|
// clear COM buffers
|
|
status = SCCMN50M_PurgeComm(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_PurgeComm failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
|
|
|
|
// 150 * 0xFE
|
|
RtlFillMemory(pSmartcardExtension->SmartcardRequest.Buffer,150,0xFE);
|
|
pSmartcardExtension->SmartcardRequest.Buffer[150] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 151;
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = SMARTCARD_WRITE;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
Wait(pSmartcardExtension,2 * ms_);
|
|
// try resync once more
|
|
|
|
// clear error flags
|
|
pSmartcardExtension->ReaderExtension->SerialErrors = 0;
|
|
|
|
// clear any pending errors
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
// clear COM buffers
|
|
status = SCCMN50M_PurgeComm(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_PurgeComm failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
|
|
// 150 * 0xFE
|
|
RtlFillMemory(pSmartcardExtension->SmartcardRequest.Buffer,150,0xFE);
|
|
pSmartcardExtension->SmartcardRequest.Buffer[150] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 151;
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = SMARTCARD_WRITE;
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_SerialIo failed ! status = %x\n",status))
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
// normally the second resync command is always successful
|
|
|
|
}
|
|
|
|
|
|
// clear COM buffers
|
|
status = SCCMN50M_PurgeComm(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
// clear any pending errors
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitSCCMN50M_ResyncCardManII;
|
|
}
|
|
|
|
|
|
ExitSCCMN50M_ResyncCardManII:
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ResyncCardManII: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SerialIo(IN PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
IO_STATUS_BLOCK ioStatus;
|
|
KEVENT event;
|
|
PIRP irp;
|
|
PIO_STACK_LOCATION irpNextStack;
|
|
PUCHAR pbRequestBuffer;
|
|
PUCHAR pbReplyBuffer;
|
|
ULONG ulRequestBufferLength;
|
|
ULONG ulReplyBufferLength ;
|
|
|
|
//
|
|
// Check if the buffers are large enough
|
|
//
|
|
ASSERT(pSmartcardExtension->SmartcardReply.BufferLength <=
|
|
pSmartcardExtension->SmartcardReply.BufferSize);
|
|
|
|
ASSERT(pSmartcardExtension->SmartcardRequest.BufferLength <=
|
|
pSmartcardExtension->SmartcardRequest.BufferSize);
|
|
|
|
if (pSmartcardExtension->SmartcardReply.BufferLength >
|
|
pSmartcardExtension->SmartcardReply.BufferSize ||
|
|
pSmartcardExtension->SmartcardRequest.BufferLength >
|
|
pSmartcardExtension->SmartcardRequest.BufferSize)
|
|
{
|
|
SmartcardLogError(pSmartcardExtension->OsData->DeviceObject,
|
|
SCCMN50M_BUFFER_TOO_SMALL,
|
|
NULL,
|
|
0);
|
|
return STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
|
|
|
|
|
|
|
|
// set pointer and length of request and reply buffer
|
|
ulRequestBufferLength = pSmartcardExtension->SmartcardRequest.BufferLength;
|
|
pbRequestBuffer = (ulRequestBufferLength ? pSmartcardExtension->SmartcardRequest.Buffer : NULL);
|
|
|
|
pbReplyBuffer = pSmartcardExtension->SmartcardReply.Buffer;
|
|
ulReplyBufferLength = pSmartcardExtension->SmartcardReply.BufferLength;
|
|
|
|
|
|
|
|
KeInitializeEvent(&event,
|
|
NotificationEvent,
|
|
FALSE);
|
|
|
|
|
|
//
|
|
// Build irp to be sent to serial driver
|
|
//
|
|
irp = IoBuildDeviceIoControlRequest(pSmartcardExtension->ReaderExtension->SerialIoControlCode,
|
|
pSmartcardExtension->ReaderExtension->AttachedDeviceObject,
|
|
pbRequestBuffer,
|
|
ulRequestBufferLength,
|
|
pbReplyBuffer,
|
|
ulReplyBufferLength,
|
|
FALSE,
|
|
&event,
|
|
&ioStatus);
|
|
|
|
|
|
ASSERT(irp != NULL);
|
|
if (irp == NULL)
|
|
{
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
|
|
irpNextStack = IoGetNextIrpStackLocation(irp);
|
|
|
|
switch (pSmartcardExtension->ReaderExtension->SerialIoControlCode)
|
|
{
|
|
//
|
|
// The serial driver transfers data from/to irp->AssociatedIrp.SystemBuffer
|
|
//
|
|
case SMARTCARD_WRITE:
|
|
irpNextStack->MajorFunction = IRP_MJ_WRITE;
|
|
irpNextStack->Parameters.Write.Length = pSmartcardExtension->SmartcardRequest.BufferLength;
|
|
break;
|
|
|
|
|
|
case SMARTCARD_READ:
|
|
irpNextStack->MajorFunction = IRP_MJ_READ;
|
|
irpNextStack->Parameters.Read.Length = pSmartcardExtension->SmartcardReply.BufferLength;
|
|
break;
|
|
}
|
|
|
|
|
|
status = IoCallDriver(pSmartcardExtension->ReaderExtension->AttachedDeviceObject,irp);
|
|
|
|
|
|
if (status == STATUS_PENDING)
|
|
{
|
|
KeWaitForSingleObject(&event,
|
|
Suspended,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
status = ioStatus.Status;
|
|
}
|
|
|
|
switch (pSmartcardExtension->ReaderExtension->SerialIoControlCode)
|
|
{
|
|
case SMARTCARD_READ:
|
|
if (status == STATUS_TIMEOUT)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,
|
|
("%s!Timeout while reading from CardMan\n",
|
|
DRIVER_NAME));
|
|
//
|
|
// STATUS_TIMEOUT isn't correctly mapped
|
|
// to a WIN32 error, that's why we change it here
|
|
// to STATUS_IO_TIMEOUT
|
|
//
|
|
status = STATUS_IO_TIMEOUT;
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
}
|
|
break;
|
|
}
|
|
|
|
#if 0
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_DRIVER,
|
|
("%s!SerialIo = %lx\n",
|
|
DRIVER_NAME,
|
|
status));
|
|
}
|
|
#endif
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ReadCardMan (
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG BytesToRead,
|
|
OUT PULONG pBytesRead,
|
|
IN PUCHAR pReadBuffer,
|
|
IN ULONG ReadBufferSize
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
BOOLEAN fRc;
|
|
|
|
// check if read buffer is large enough
|
|
ASSERT(BytesToRead <= ReadBufferSize);
|
|
|
|
|
|
*pBytesRead = 0; // default setting
|
|
|
|
|
|
DebugStatus = SCCMN50M_SetRead(pSmartcardExtension,BytesToRead);
|
|
|
|
//
|
|
// read operation
|
|
//
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
*pBytesRead = pSmartcardExtension->SmartcardReply.BufferLength;
|
|
|
|
MemCpy(pReadBuffer,
|
|
ReadBufferSize,
|
|
pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferLength);
|
|
|
|
// overwrite read buffer with '@'
|
|
MemSet(pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferSize,
|
|
'@',
|
|
pSmartcardExtension->SmartcardReply.BufferLength);
|
|
}
|
|
|
|
if (status != STATUS_SUCCESS || SCCMN50M_IOOperationFailed(pSmartcardExtension))
|
|
{
|
|
if (!pSmartcardExtension->ReaderExtension->fTransparentMode)
|
|
{
|
|
DebugStatus = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
}
|
|
goto ExitSCCMN50M_ReadCardMan;
|
|
}
|
|
|
|
|
|
// *****************************************
|
|
// set CardManII to state RH Config
|
|
// *****************************************
|
|
// don't set CardMan to RH config if there are still bytes to be read
|
|
if (pSmartcardExtension->ReaderExtension->ToRHConfig == TRUE)
|
|
{
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
pSmartcardExtension->SmartcardRequest.Buffer [0] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.Buffer [1] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.Buffer [2] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.Buffer [3] = 0x00;
|
|
pSmartcardExtension->SmartcardRequest.Buffer [4] = 0x89;
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 5;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = SMARTCARD_WRITE;
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS || SCCMN50M_IOOperationFailed(pSmartcardExtension))
|
|
{
|
|
DebugStatus = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
goto ExitSCCMN50M_ReadCardMan;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
ExitSCCMN50M_ReadCardMan:
|
|
// set default value;
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = TRUE;
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ReadCardMan: Failed, exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_GetCommStatus (
|
|
IN PSMARTCARD_EXTENSION SmartcardExtension
|
|
)
|
|
{
|
|
PSERIAL_READER_CONFIG configData = &SmartcardExtension->ReaderExtension->SerialConfigData;
|
|
NTSTATUS status;
|
|
PUCHAR request = SmartcardExtension->SmartcardRequest.Buffer;
|
|
|
|
|
|
SmartcardExtension->SmartcardReply.BufferLength = SmartcardExtension->SmartcardReply.BufferSize;
|
|
|
|
SmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_GET_COMMSTATUS;
|
|
|
|
SmartcardExtension->SmartcardRequest.Buffer = (PUCHAR) &configData->SerialStatus;
|
|
|
|
SmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_STATUS);
|
|
|
|
status = SCCMN50M_SerialIo(SmartcardExtension);
|
|
|
|
//
|
|
// restore pointer to original request buffer
|
|
//
|
|
SmartcardExtension->SmartcardRequest.Buffer = request;
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
BOOLEAN
|
|
SCCMN50M_IOOperationFailed(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS DebugStatus;
|
|
PSERIAL_STATUS pSerialStatus;
|
|
|
|
DebugStatus = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
|
|
pSerialStatus = (PSERIAL_STATUS)pSmartcardExtension->SmartcardReply.Buffer;
|
|
if (pSerialStatus->Errors)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS SCCMN50M_PurgeComm (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
PSERIAL_READER_CONFIG configData = &pSmartcardExtension->ReaderExtension->SerialConfigData;
|
|
NTSTATUS status;
|
|
PUCHAR request = pSmartcardExtension->SmartcardRequest.Buffer;
|
|
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = pSmartcardExtension->SmartcardReply.BufferSize;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_PURGE;
|
|
|
|
pSmartcardExtension->SmartcardRequest.Buffer = (PUCHAR) &configData->PurgeMask;
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(ULONG);
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
//
|
|
// restore pointer to original request buffer
|
|
//
|
|
pSmartcardExtension->SmartcardRequest.Buffer = request;
|
|
|
|
// under W2000 & CardMan P+ STATUS_CANCELLED may be returned
|
|
|
|
if (status == STATUS_CANCELLED)
|
|
status = STATUS_SUCCESS;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetRead(IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBytesToRead
|
|
)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = SMARTCARD_READ;
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = ulBytesToRead;
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetWrite(IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG BytesToWrite,
|
|
IN PUCHAR WriteBuffer
|
|
)
|
|
{
|
|
ULONG Offset = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = SMARTCARD_WRITE;
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
|
|
if (pSmartcardExtension->ReaderExtension->fTransparentMode == FALSE)
|
|
{
|
|
// send always config string for CardManII, expect we set it manualy
|
|
// to NoConfig = TRUE. (note: only one time)
|
|
if (pSmartcardExtension->ReaderExtension->NoConfig == FALSE)
|
|
{
|
|
MemCpy(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
(PUCHAR)&pSmartcardExtension->ReaderExtension->CardManConfig,
|
|
sizeof(CARDMAN_CONFIG));
|
|
Offset = 4;
|
|
}
|
|
else
|
|
{
|
|
pSmartcardExtension->ReaderExtension->NoConfig = FALSE;
|
|
}
|
|
|
|
|
|
|
|
MemCpy(pSmartcardExtension->SmartcardRequest.Buffer + Offset,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
(PUCHAR)&pSmartcardExtension->ReaderExtension->CardManHeader,
|
|
sizeof(CARDMAN_HEADER));
|
|
Offset+=4;
|
|
}
|
|
|
|
|
|
if (BytesToWrite != 0)
|
|
{
|
|
MemCpy(pSmartcardExtension->SmartcardRequest.Buffer + Offset,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
WriteBuffer,
|
|
BytesToWrite);
|
|
}
|
|
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = Offset + BytesToWrite;
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_StartCardTracking(
|
|
PDEVICE_EXTENSION pDeviceExtension
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
HANDLE hThread;
|
|
PSMARTCARD_EXTENSION pSmartcardExtension = &pDeviceExtension->SmartcardExtension;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartCardTracking: Enter\n",DRIVER_NAME));
|
|
|
|
KeWaitForSingleObject(&pSmartcardExtension->ReaderExtension->CardManIOMutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
|
|
|
|
// create thread for updating current state
|
|
status = PsCreateSystemThread(&hThread,
|
|
THREAD_ALL_ACCESS,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
SCCMN50M_UpdateCurrentStateThread,
|
|
pDeviceExtension);
|
|
if (!NT_ERROR(status))
|
|
{
|
|
//
|
|
// We've got the thread. Now get a pointer to it.
|
|
//
|
|
status = ObReferenceObjectByHandle(hThread,
|
|
THREAD_ALL_ACCESS,
|
|
NULL,
|
|
KernelMode,
|
|
&pSmartcardExtension->ReaderExtension->ThreadObjectPointer,
|
|
NULL);
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
pSmartcardExtension->ReaderExtension->TimeToTerminateThread = TRUE;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Now that we have a reference to the thread
|
|
// we can simply close the handle.
|
|
//
|
|
ZwClose(hThread);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
}
|
|
|
|
// Release the mutex
|
|
KeReleaseMutex(&pSmartcardExtension->ReaderExtension->CardManIOMutex,
|
|
FALSE);
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!StartCardTracking: Exit %lx\n",DRIVER_NAME,status));
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_InitCommPort (PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!InitCommPort: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
// ===============================
|
|
// clear any pending errors
|
|
// ===============================
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %ld\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
|
|
// ==============================
|
|
// set baudrate for CardMan
|
|
// ==============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate = 38400;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BAUD_RATE;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate,
|
|
sizeof(SERIAL_BAUD_RATE));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_BAUD_RATE);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_BAUDRATE failed ! status = %ld\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
|
|
|
|
// ===============================
|
|
// set comm timeouts
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadIntervalTimeout = DEFAULT_READ_INTERVAL_TIMEOUT;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier = DEFAULT_READ_TOTAL_TIMEOUT_MULTIPLIER;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.WriteTotalTimeoutConstant = DEFAULT_WRITE_TOTAL_TIMEOUT_CONSTANT;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.WriteTotalTimeoutMultiplier = DEFAULT_WRITE_TOTAL_TIMEOUT_MULTIPLIER;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_TIMEOUTS;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts,
|
|
sizeof(SERIAL_TIMEOUTS));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_TIMEOUTS);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_TIMEOUTS failed ! status = %x\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
// ===============================
|
|
// set line control
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = EVEN_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = 8;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_LINE_CONTROL failed ! status = %x\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
|
|
|
|
// ===============================
|
|
// Set handflow
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.XonLimit = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.XoffLimit = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.FlowReplace = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.ControlHandShake = SERIAL_ERROR_ABORT | SERIAL_DTR_CONTROL;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_HANDFLOW;
|
|
|
|
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow,
|
|
sizeof(SERIAL_HANDFLOW));
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_HANDFLOW);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_HANDFLOW failed ! status = %x\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
// ===============================
|
|
// set purge mask
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.PurgeMask =
|
|
SERIAL_PURGE_TXABORT | SERIAL_PURGE_RXABORT |
|
|
SERIAL_PURGE_TXCLEAR | SERIAL_PURGE_RXCLEAR;
|
|
|
|
|
|
|
|
// ===============================
|
|
// set DTR
|
|
// ===============================
|
|
status = SCCMN50M_SetDTR(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_DRT failed ! status = %x\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
// ===============================
|
|
// set RTS
|
|
// ===============================
|
|
status = SCCMN50M_SetRTS(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_RTS failed ! status = %x\n",status))
|
|
goto ExitInitCommPort;
|
|
}
|
|
|
|
|
|
|
|
|
|
ExitInitCommPort:
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!InitCommPort: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetDTR(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_DTR;
|
|
|
|
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
// under W2000 & CardMan P+ STATUS_CANCELLED may be returned
|
|
|
|
if (status == STATUS_CANCELLED)
|
|
status = STATUS_SUCCESS;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetRTS(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = pSmartcardExtension->SmartcardReply.BufferSize;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_RTS;
|
|
|
|
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
// under W2000 & CardMan P+ STATUS_CANCELLED may be returned
|
|
|
|
if (status == STATUS_CANCELLED)
|
|
status = STATUS_SUCCESS;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_InitializeCardMan(IN PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
UCHAR pReadBuffer[2];
|
|
ULONG ulBytesRead;
|
|
BOOLEAN fCardManFound = FALSE;
|
|
PREADER_EXTENSION readerExtension = pSmartcardExtension->ReaderExtension;
|
|
ULONG ulRetries;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!InitializeCardMan: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->ulOldCardState = UNKNOWN;
|
|
|
|
|
|
// ==============================================
|
|
// CardManII
|
|
// ==============================================
|
|
pSmartcardExtension->ReaderExtension->NoConfig = FALSE;
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = TRUE;
|
|
|
|
|
|
// This waiting time if necessary for CardMan Power+, because
|
|
// the pnP string may be dumped
|
|
Wait(pSmartcardExtension,200);
|
|
|
|
status = SCCMN50M_InitCommPort(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
goto ExitInitializeCardMan;
|
|
//
|
|
// init CommPort was O.K.
|
|
// now try to find a reader
|
|
//
|
|
|
|
// To be sure wait make an additional wait
|
|
Wait(pSmartcardExtension,100);
|
|
|
|
status = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
status = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
|
|
|
|
// no data except config + header
|
|
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.SCRControl = XMIT_HANDSHAKE_OFF;
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardControl = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.ResetDelay = 0x00;
|
|
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxControl = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.TxLength = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxControl = 0x00;
|
|
pSmartcardExtension->ReaderExtension->CardManHeader.RxLength = 0x00;
|
|
|
|
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,pReadBuffer,sizeof(pReadBuffer));
|
|
|
|
if (status == STATUS_SUCCESS &&
|
|
ulBytesRead == 0x02 && // two bytes received
|
|
pReadBuffer[0] >= 0x40 && // at least one version bit must be set
|
|
pReadBuffer[1] == 0x00 &&
|
|
((pReadBuffer[0] & 0x09) == 0) ) // bit 0 and 3 must be cleared
|
|
{
|
|
pSmartcardExtension->ReaderExtension->ulFWVersion = (pReadBuffer[0] >> 4) * 30 + 120;
|
|
pSmartcardExtension->ReaderExtension->fSPESupported = FALSE;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
( "%s!CardMan (FW %ld) found\n",
|
|
DRIVER_NAME,pSmartcardExtension->ReaderExtension->ulFWVersion)
|
|
);
|
|
fCardManFound = TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
ExitInitializeCardMan:
|
|
|
|
if (fCardManFound == TRUE)
|
|
status = STATUS_SUCCESS;
|
|
else
|
|
status = STATUS_UNSUCCESSFUL;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!InitializeCardMan: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_EnterTransparentMode (IN PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,("EnterTransparentMode : enter\n"));
|
|
|
|
// Step 1 : Resync CardMan by RTS usage
|
|
status = SCCMN50M_ResyncCardManI(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_ResyncCardManI failed ! status = %ld\n",status))
|
|
goto ExitEnterTransparentMode;
|
|
}
|
|
|
|
// Step 2 : set baud rate to 9600
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate = 9600;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BAUD_RATE;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate,
|
|
sizeof(SERIAL_BAUD_RATE));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_BAUD_RATE);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_BAUDRATE failed ! status = %ld\n",status))
|
|
goto ExitEnterTransparentMode;
|
|
}
|
|
|
|
|
|
|
|
ExitEnterTransparentMode:
|
|
// Step 3 : set ATR and DUMP_BUFFER flags
|
|
// During normal operation these two flags can never be set at the same time
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CM2_GET_ATR | TO_STATE_XH);
|
|
|
|
|
|
|
|
|
|
|
|
SmartcardDebug(DEBUG_TRACE,("EnterTransparentMode : exit\n"));
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ExitTransparentMode (IN PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
|
|
SmartcardDebug(DEBUG_TRACE,("ExitTransparentMode : enter\n"));
|
|
|
|
|
|
// ===============================
|
|
// clear any pending errors
|
|
// ===============================
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %ld\n",status))
|
|
goto ExitExitTransparentMode;
|
|
}
|
|
|
|
|
|
// Step 1 : Resync CardMan by RTS usage
|
|
status = SCCMN50M_ResyncCardManI(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_ResyncCardManI failed ! status = %ld\n",status))
|
|
goto ExitExitTransparentMode;
|
|
}
|
|
|
|
// Step 2 : set baud rate to 38400
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate = 38400;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BAUD_RATE;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate,
|
|
sizeof(SERIAL_BAUD_RATE));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_BAUD_RATE);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_BAUDRATE failed ! status = %ld\n",status))
|
|
goto ExitExitTransparentMode;
|
|
}
|
|
|
|
|
|
// if the inserted card uses inverse convention , we must now switch the COM port
|
|
// back to even parity
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = EVEN_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = SERIAL_DATABITS_8;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_LINE_CONTROL failed ! status = %x\n",status))
|
|
goto ExitExitTransparentMode;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
ExitExitTransparentMode:
|
|
|
|
// Step 3 : set ATR and DUMP_BUFFER flags
|
|
// During normal operation these two flags can never be set at the same time
|
|
SCCMN50M_ClearSCRControlFlags(pSmartcardExtension,CM2_GET_ATR | TO_STATE_XH);
|
|
|
|
|
|
|
|
status = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_ResyncCardManII failed ! status = %x\n",status))
|
|
goto ExitExitTransparentMode;
|
|
}
|
|
|
|
|
|
SmartcardDebug(DEBUG_TRACE,("ExitTransparentMode : exit\n"));
|
|
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ClearRTS(IN PSMARTCARD_EXTENSION SmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
SmartcardExtension->SmartcardReply.BufferLength = SmartcardExtension->SmartcardReply.BufferSize;
|
|
SmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
SmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_CLR_RTS;
|
|
|
|
|
|
|
|
status = SCCMN50M_SerialIo(SmartcardExtension);
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_IoCtlVendor(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
NTSTATUS DebugStatus;
|
|
UCHAR pbAttrBuffer[MAXIMUM_ATR_LENGTH];
|
|
ULONG ulAtrLength;
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IoCtlVendor : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
|
|
switch (pSmartcardExtension->MajorIoControlCode)
|
|
{
|
|
case CM_IOCTL_SET_READER_9600_BAUD:
|
|
status = SCCMN50M_SetFl_1Dl_1(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_SET_READER_38400_BAUD:
|
|
status = SCCMN50M_SetFl_1Dl_3(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_CR80S_SAMOS_SET_HIGH_SPEED:
|
|
status = SCCMN50M_SetHighSpeed_CR80S_SAMOS(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_GET_FW_VERSION:
|
|
status = SCCMN50M_GetFWVersion(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_READ_DEVICE_DESCRIPTION:
|
|
status = SCCMN50M_ReadDeviceDescription(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_SET_SYNC_PARAMETERS :
|
|
status = SCCMN50M_SetSyncParameters(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_3WBP_TRANSFER : // for SLE4428
|
|
status = SCCMN50M_Transmit3WBP(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_2WBP_TRANSFER : // for SLE4442
|
|
status = SCCMN50M_Transmit2WBP(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_2WBP_RESET_CARD: // SLE4442 Reset Card
|
|
status = SCCMN50M_ResetCard2WBP(pSmartcardExtension);
|
|
break;
|
|
|
|
case CM_IOCTL_SYNC_CARD_POWERON:
|
|
status = SCCMN50M_SyncCardPowerOn(pSmartcardExtension);
|
|
break;
|
|
default:
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
break;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IoCtlVendor : Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetFl_1Dl_3(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
NTSTATUS DebugStatus;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!SetFl_1Dl_3 Enter\n",
|
|
DRIVER_NAME));
|
|
|
|
// check if T=1 active
|
|
if (pSmartcardExtension->CardCapabilities.Protocol.Selected !=
|
|
SCARD_PROTOCOL_T1)
|
|
{
|
|
status = STATUS_CTL_FILE_NOT_SUPPORTED;
|
|
goto ExitSetFl_1Dl_3;
|
|
}
|
|
|
|
// Fl=1
|
|
// Dl=3
|
|
// => 38400 Baud for 3.72 MHz
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_3MHZ | ENABLE_5MHZ |
|
|
ENABLE_3MHZ_FAST | ENABLE_5MHZ_FAST );
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ_FAST);
|
|
|
|
|
|
ExitSetFl_1Dl_3:
|
|
*pSmartcardExtension->IoRequest.Information = 0L;
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!SetFl_1Dl_3 Exit\n",
|
|
DRIVER_NAME));
|
|
return status;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetFl_1Dl_1(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!SetFl_1Dl_1 Enter\n",
|
|
DRIVER_NAME));
|
|
// Fl=1
|
|
// Dl=1
|
|
// => 9600 for 3.72 MHz
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_3MHZ | ENABLE_5MHZ |
|
|
ENABLE_3MHZ_FAST | ENABLE_5MHZ_FAST );
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
|
|
|
|
*pSmartcardExtension->IoRequest.Information = 0L;
|
|
SmartcardDebug(DEBUG_TRACE,
|
|
("%s!SetFl_1Dl_1 Exit\n",
|
|
DRIVER_NAME));
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_GetFWVersion (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!GetFWVersion : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
if (pSmartcardExtension->IoRequest.ReplyBufferLength < sizeof (ULONG))
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitGetFWVersion;
|
|
}
|
|
else
|
|
{
|
|
*(PULONG)(pSmartcardExtension->IoRequest.ReplyBuffer) =
|
|
pSmartcardExtension->ReaderExtension->ulFWVersion;
|
|
}
|
|
|
|
|
|
ExitGetFWVersion:
|
|
*pSmartcardExtension->IoRequest.Information = sizeof(ULONG);
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!GetFWVersion : Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ReadDeviceDescription(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ReadDeviceDescription : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
if (pSmartcardExtension->IoRequest.ReplyBufferLength < sizeof(pSmartcardExtension->ReaderExtension->abDeviceDescription))
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
*pSmartcardExtension->IoRequest.Information = 0L;
|
|
goto ExitReadDeviceDescription;
|
|
}
|
|
else
|
|
{
|
|
if (pSmartcardExtension->ReaderExtension->abDeviceDescription[0] == 0x00 &&
|
|
pSmartcardExtension->ReaderExtension->abDeviceDescription[1] == 0x00 )
|
|
{
|
|
status = SCCMN50M_GetDeviceDescription(pSmartcardExtension);
|
|
}
|
|
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
StrCpy(pSmartcardExtension->IoRequest.ReplyBuffer,
|
|
pSmartcardExtension->IoRequest.ReplyBufferLength,
|
|
pSmartcardExtension->ReaderExtension->abDeviceDescription);
|
|
*pSmartcardExtension->IoRequest.Information = StrLen(pSmartcardExtension->ReaderExtension->abDeviceDescription) +1;
|
|
}
|
|
else
|
|
{
|
|
MemSet(pSmartcardExtension->ReaderExtension->abDeviceDescription,
|
|
sizeof(pSmartcardExtension->ReaderExtension->abDeviceDescription),
|
|
0x00,
|
|
sizeof(pSmartcardExtension->ReaderExtension->abDeviceDescription));
|
|
|
|
*pSmartcardExtension->IoRequest.Information = 0;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
ExitReadDeviceDescription:
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ReadDeviceDescription : Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetHighSpeed_CR80S_SAMOS (IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
UCHAR bReadBuffer[16];
|
|
ULONG ulBytesRead;
|
|
BYTE bCR80S_SAMOS_SET_HIGH_SPEED[4] = {0xFF,0x11,0x94,0x7A};
|
|
ULONG ulAtrLength;
|
|
BYTE bAtr[MAXIMUM_ATR_LENGTH];
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetHighSpeed_CR80S_SAMOS : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_SYN | ENABLE_T0 |
|
|
ENABLE_T1 );
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0, // Tx control
|
|
sizeof(bCR80S_SAMOS_SET_HIGH_SPEED), // Tx length
|
|
0, // Rx control
|
|
sizeof(bCR80S_SAMOS_SET_HIGH_SPEED)); // Rx length
|
|
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,
|
|
sizeof(bCR80S_SAMOS_SET_HIGH_SPEED),
|
|
bCR80S_SAMOS_SET_HIGH_SPEED);
|
|
if (status != STATUS_SUCCESS)
|
|
goto ExitSetHighSpeed;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
goto ExitSetHighSpeed;
|
|
|
|
if (bReadBuffer[1] > sizeof(bReadBuffer))
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitSetHighSpeed;
|
|
}
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,bReadBuffer[1],&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
goto ExitSetHighSpeed;
|
|
|
|
// if the card has accepted this string , the string is echoed
|
|
if (bReadBuffer[0] == bCR80S_SAMOS_SET_HIGH_SPEED[0] &&
|
|
bReadBuffer[1] == bCR80S_SAMOS_SET_HIGH_SPEED[1] &&
|
|
bReadBuffer[2] == bCR80S_SAMOS_SET_HIGH_SPEED[2] &&
|
|
bReadBuffer[3] == bCR80S_SAMOS_SET_HIGH_SPEED[3] )
|
|
{
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_3MHZ | ENABLE_5MHZ |
|
|
ENABLE_3MHZ_FAST | ENABLE_5MHZ_FAST );
|
|
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ_FAST);
|
|
}
|
|
else
|
|
{
|
|
DebugStatus = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
|
|
DebugStatus = SCCMN50M_PowerOn(pSmartcardExtension,&ulAtrLength,bAtr,sizeof(bAtr));
|
|
status = STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ExitSetHighSpeed:
|
|
*pSmartcardExtension->IoRequest.Information = 0L;
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetHighSpeed_CR80S_SAMOS : Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetBRK(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = pSmartcardExtension->SmartcardReply.BufferSize;
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BREAK_ON;
|
|
|
|
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ClearBRK(IN PSMARTCARD_EXTENSION SmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
SmartcardExtension->SmartcardReply.BufferLength = SmartcardExtension->SmartcardReply.BufferSize;
|
|
SmartcardExtension->SmartcardRequest.BufferLength = 0;
|
|
|
|
SmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BREAK_OFF;
|
|
|
|
|
|
|
|
status = SCCMN50M_SerialIo(SmartcardExtension);
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetProtocol(PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
ULONG ulNewProtocol;
|
|
UCHAR abPTSRequest[4];
|
|
UCHAR abReadBuffer[6];
|
|
UCHAR abPTSReply [4];
|
|
ULONG ulBytesRead;
|
|
UCHAR bTemp;
|
|
ULONG ulPtsType;
|
|
ULONG ulPTSReplyLength=0;
|
|
ULONG ulStatBytesRead;
|
|
BYTE abStatReadBuffer[2];
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetProtocol : Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
//
|
|
// Check if the card is already in specific state
|
|
// and if the caller wants to have the already selected protocol.
|
|
// We return success if this is the case.
|
|
//
|
|
if ((pSmartcardExtension->CardCapabilities.Protocol.Selected & pSmartcardExtension->MinorIoControlCode))
|
|
{
|
|
status = STATUS_SUCCESS;
|
|
goto ExitSetProtocol;
|
|
}
|
|
|
|
ulNewProtocol = pSmartcardExtension->MinorIoControlCode;
|
|
|
|
|
|
|
|
ulPtsType = PTS_TYPE_OPTIMAL;
|
|
|
|
// we are not sure if we need this at all
|
|
pSmartcardExtension->CardCapabilities.PtsData.Type = PTS_TYPE_OPTIMAL;
|
|
while (TRUE)
|
|
{
|
|
|
|
// set initial character of PTS
|
|
abPTSRequest[0] = 0xFF;
|
|
|
|
// set the format character
|
|
if (pSmartcardExtension->CardCapabilities.Protocol.Supported &
|
|
ulNewProtocol &
|
|
SCARD_PROTOCOL_T1)
|
|
{
|
|
// select T=1 and indicate that PTS1 follows
|
|
abPTSRequest[1] = 0x11;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected =
|
|
SCARD_PROTOCOL_T1;
|
|
}
|
|
else if (pSmartcardExtension->CardCapabilities.Protocol.Supported &
|
|
ulNewProtocol &
|
|
SCARD_PROTOCOL_T0)
|
|
{
|
|
// select T=1 and indicate that PTS1 follows
|
|
abPTSRequest[1] = 0x10;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected =
|
|
SCARD_PROTOCOL_T0;
|
|
}
|
|
else
|
|
{
|
|
status = STATUS_INVALID_DEVICE_REQUEST;
|
|
goto ExitSetProtocol;
|
|
}
|
|
|
|
// bug fix :
|
|
// don 't use the suggestion from smclib
|
|
pSmartcardExtension->CardCapabilities.PtsData.Fl =
|
|
pSmartcardExtension->CardCapabilities.Fl;
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl =
|
|
pSmartcardExtension->CardCapabilities.Dl;
|
|
|
|
|
|
// CardMan support higher baudrates only for T=1
|
|
// ==> Dl=1
|
|
if (abPTSRequest[1] == 0x10)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!overwriting PTS1 for T=0\n",
|
|
DRIVER_NAME)
|
|
);
|
|
pSmartcardExtension->CardCapabilities.PtsData.Fl = 0x01;
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl = 0x01;
|
|
}
|
|
|
|
|
|
if (ulPtsType == PTS_TYPE_DEFAULT)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!overwriting PTS1 with default values\n",
|
|
DRIVER_NAME)
|
|
);
|
|
pSmartcardExtension->CardCapabilities.PtsData.Fl = 0x01;
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl = 0x01;
|
|
}
|
|
|
|
|
|
// set pts1 which codes Fl and Dl
|
|
bTemp = (BYTE) (pSmartcardExtension->CardCapabilities.PtsData.Fl << 4 |
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl);
|
|
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!PTS1 = %x (suggestion)\n",
|
|
DRIVER_NAME,bTemp)
|
|
);
|
|
|
|
|
|
switch (bTemp)
|
|
{
|
|
case 0x11:
|
|
// do nothing
|
|
// we support these Fl/Dl parameters
|
|
break;
|
|
|
|
case 0x13:
|
|
case 0x94:
|
|
break ;
|
|
|
|
|
|
case 0x14:
|
|
// let's try it with 38400 baud
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!trying 57600 baud\n",DRIVER_NAME)
|
|
);
|
|
// we must correct Fl/Dl
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl = 0x03;
|
|
pSmartcardExtension->CardCapabilities.PtsData.Fl = 0x01;
|
|
bTemp = (BYTE) (pSmartcardExtension->CardCapabilities.PtsData.Fl << 4 |
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl);
|
|
break;
|
|
|
|
default:
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!overwriting PTS1(0x%x)\n",
|
|
DRIVER_NAME,bTemp)
|
|
);
|
|
// we must correct Fl/Dl
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl = 0x01;
|
|
pSmartcardExtension->CardCapabilities.PtsData.Fl = 0x01;
|
|
bTemp = (BYTE) (pSmartcardExtension->CardCapabilities.PtsData.Fl << 4 |
|
|
pSmartcardExtension->CardCapabilities.PtsData.Dl);
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
abPTSRequest[2] = bTemp;
|
|
|
|
// set pck (check character)
|
|
abPTSRequest[3] = (BYTE)(abPTSRequest[0] ^ abPTSRequest[1] ^ abPTSRequest[2]);
|
|
|
|
SmartcardDebug(DEBUG_PROTOCOL,("%s!PTS request: 0x%x 0x%x 0x%x 0x%x\n",
|
|
DRIVER_NAME,
|
|
abPTSRequest[0],
|
|
abPTSRequest[1],
|
|
abPTSRequest[2],
|
|
abPTSRequest[3]));
|
|
|
|
|
|
MemSet(abPTSReply,sizeof(abPTSReply),0x00,sizeof(abPTSReply));
|
|
|
|
|
|
|
|
DebugStatus = SCCMN50M_EnterTransparentMode(pSmartcardExtension);
|
|
|
|
// STEP 1 : write config + header to enter transparent mode
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0, // Tx control
|
|
0, // Tx length
|
|
0, // Rx control
|
|
0); // Rx length
|
|
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
0,
|
|
NULL);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitSetProtocol;
|
|
}
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->fTransparentMode = TRUE;
|
|
|
|
// if the inserted card uses inverse convention , we must now switch the COM port
|
|
// to odd parity
|
|
/*
|
|
if (pSmartcardExtension->ReaderExtension->fInverseAtr == TRUE)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = ODD_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = SERIAL_DATABITS_8;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
goto ExitTransparentTransmitT0;
|
|
}
|
|
}
|
|
*/
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!writing PTS request\n",
|
|
DRIVER_NAME)
|
|
);
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,
|
|
4,
|
|
abPTSRequest);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!writing PTS request failed\n",
|
|
DRIVER_NAME)
|
|
);
|
|
goto ExitSetProtocol;
|
|
}
|
|
|
|
|
|
|
|
// read back pts data
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!trying to read PTS reply\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
// first read CardMan header
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,3,&ulBytesRead,abReadBuffer,sizeof(abReadBuffer));
|
|
if (status != STATUS_SUCCESS &&
|
|
status != STATUS_IO_TIMEOUT )
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!reading status failed\n",
|
|
DRIVER_NAME)
|
|
);
|
|
goto ExitSetProtocol;
|
|
}
|
|
ulPTSReplyLength = 3;
|
|
MemCpy(abPTSReply,sizeof(abPTSReply),abReadBuffer,3);
|
|
|
|
|
|
|
|
// check if bit 5 is set
|
|
if (abPTSReply[1] & 0x10)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig= FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,1,&ulBytesRead,abReadBuffer,sizeof(abReadBuffer));
|
|
if (status != STATUS_SUCCESS &&
|
|
status != STATUS_IO_TIMEOUT )
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!reading status failed\n",
|
|
DRIVER_NAME)
|
|
);
|
|
goto ExitSetProtocol;
|
|
}
|
|
ulPTSReplyLength += 1;
|
|
MemCpy(&abPTSReply[3],sizeof(abPTSReply)-3,abReadBuffer,1);
|
|
}
|
|
|
|
DebugStatus = SCCMN50M_ExitTransparentMode(pSmartcardExtension);
|
|
pSmartcardExtension->ReaderExtension->fTransparentMode = FALSE;
|
|
|
|
// to be sure that the new settings take effect
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = 250;
|
|
DebugStatus = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
if (NT_SUCCESS(DebugStatus))
|
|
{
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulStatBytesRead,abStatReadBuffer,sizeof(abStatReadBuffer));
|
|
}
|
|
|
|
|
|
#if DBG
|
|
if (ulPTSReplyLength == 3)
|
|
{
|
|
SmartcardDebug(DEBUG_PROTOCOL,("PTS reply: 0x%x 0x%x 0x%x\n",
|
|
abPTSReply[0],
|
|
abPTSReply[1],
|
|
abPTSReply[2]));
|
|
}
|
|
|
|
if (ulPTSReplyLength == 4)
|
|
{
|
|
SmartcardDebug(DEBUG_PROTOCOL,("PTS reply: 0x%x 0x%x 0x%x 0x%x\n",
|
|
abPTSReply[0],
|
|
abPTSReply[1],
|
|
abPTSReply[2],
|
|
abPTSReply[3]));
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
if (ulPTSReplyLength == 3 &&
|
|
abPTSReply[0] == abPTSRequest[0] &&
|
|
(abPTSReply[1] & 0x7F) == (abPTSRequest[1] & 0x0F) &&
|
|
abPTSReply[2] == (BYTE)(abPTSReply[0] ^ abPTSReply[1]) )
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!short PTS reply received\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
break;
|
|
}
|
|
|
|
if (ulPTSReplyLength == 4 &&
|
|
abPTSReply[0] == abPTSRequest[0] &&
|
|
abPTSReply[1] == abPTSRequest[1] &&
|
|
abPTSReply[2] == abPTSRequest[2] &&
|
|
abPTSReply[3] == abPTSRequest[3])
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!PTS request and reply match\n",
|
|
DRIVER_NAME)
|
|
);
|
|
switch (bTemp)
|
|
{
|
|
case 0x11:
|
|
break;
|
|
|
|
case 0x13:
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_3MHZ | ENABLE_5MHZ |
|
|
ENABLE_3MHZ_FAST | ENABLE_5MHZ_FAST );
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ_FAST);
|
|
break ;
|
|
|
|
case 0x94:
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ENABLE_3MHZ | ENABLE_5MHZ |
|
|
ENABLE_3MHZ_FAST | ENABLE_5MHZ_FAST );
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ_FAST);
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (pSmartcardExtension->CardCapabilities.PtsData.Type !=
|
|
PTS_TYPE_DEFAULT)
|
|
{
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!PTS failed : Trying default parameters\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
// the card did either not reply or it replied incorrectly
|
|
// so try default valies
|
|
ulPtsType = pSmartcardExtension->CardCapabilities.PtsData.Type = PTS_TYPE_DEFAULT;
|
|
pSmartcardExtension->MinorIoControlCode = SCARD_COLD_RESET;
|
|
status = SCCMN50M_CardPower(pSmartcardExtension);
|
|
continue;
|
|
}
|
|
|
|
// the card failed the pts request
|
|
status = STATUS_DEVICE_PROTOCOL_ERROR;
|
|
goto ExitSetProtocol;
|
|
|
|
}
|
|
|
|
|
|
|
|
ExitSetProtocol:
|
|
switch (status)
|
|
{
|
|
case STATUS_IO_TIMEOUT:
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
*pSmartcardExtension->IoRequest.Information = 0;
|
|
break;
|
|
|
|
|
|
case STATUS_SUCCESS:
|
|
|
|
// now indicate that we're in specific mode
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_SPECIFIC;
|
|
|
|
// return the selected protocol to the caller
|
|
*(PULONG) pSmartcardExtension->IoRequest.ReplyBuffer =
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected;
|
|
|
|
*pSmartcardExtension->IoRequest.Information =
|
|
sizeof(pSmartcardExtension->CardCapabilities.Protocol.Selected);
|
|
SmartcardDebug(
|
|
DEBUG_PROTOCOL,
|
|
( "%s!Selected protocol: T=%ld\n",
|
|
DRIVER_NAME,pSmartcardExtension->CardCapabilities.Protocol.Selected-1)
|
|
);
|
|
break;
|
|
|
|
default :
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_UNDEFINED;
|
|
*pSmartcardExtension->IoRequest.Information = 0;
|
|
break;
|
|
}
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetProtocol : Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
The smart card lib requires to have this function. It is called
|
|
to setup event tracking for card insertion and removal events.
|
|
|
|
Arguments:
|
|
|
|
pSmartcardExtension - pointer to the smart card data struct.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_CardTracking(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
KIRQL oldIrql;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!CardTracking: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
//
|
|
// Set cancel routine for the notification irp
|
|
//
|
|
IoAcquireCancelSpinLock(&oldIrql);
|
|
|
|
IoSetCancelRoutine(pSmartcardExtension->OsData->NotificationIrp,SCCMN50M_Cancel);
|
|
|
|
IoReleaseCancelSpinLock(oldIrql);
|
|
|
|
//
|
|
// Mark notification irp pending
|
|
//
|
|
IoMarkIrpPending(pSmartcardExtension->OsData->NotificationIrp);
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!CardTracking: Exit\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
return STATUS_PENDING;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_StopCardTracking(
|
|
IN PDEVICE_EXTENSION pDeviceExtension
|
|
)
|
|
{
|
|
PSMARTCARD_EXTENSION pSmartcardExtension = &pDeviceExtension->SmartcardExtension;
|
|
NTSTATUS status;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!StopCardTracking: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
if (pSmartcardExtension->ReaderExtension->ThreadObjectPointer != NULL)
|
|
{
|
|
|
|
// kill thread
|
|
KeWaitForSingleObject(&pSmartcardExtension->ReaderExtension->CardManIOMutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL
|
|
);
|
|
pSmartcardExtension->ReaderExtension->TimeToTerminateThread = TRUE;
|
|
KeReleaseMutex(&pSmartcardExtension->ReaderExtension->CardManIOMutex,FALSE);
|
|
|
|
|
|
//
|
|
// Wait on the thread handle, when the wait is satisfied, the
|
|
// thread has gone away.
|
|
//
|
|
status = KeWaitForSingleObject(
|
|
pSmartcardExtension->ReaderExtension->ThreadObjectPointer,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL
|
|
);
|
|
|
|
pSmartcardExtension->ReaderExtension->ThreadObjectPointer = NULL;
|
|
}
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!StopCardTracking: Exit %lx\n",
|
|
DRIVER_NAME,
|
|
status)
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_GetDeviceDescription (PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
ULONG ulBytesRead;
|
|
BYTE bByteRead;
|
|
ULONG i,j;
|
|
BYTE abReadBuffer[256];
|
|
ULONG ulPnPStringLength = 0;
|
|
ULONG ulExtend;
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!GetDeviceDescriptiong: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
|
|
// ===============================
|
|
// clear any pending errors
|
|
// ===============================
|
|
status = SCCMN50M_GetCommStatus(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("SCCMN50M_GetCommStatus failed ! status = %ld\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
|
|
// =================================
|
|
// set baudrate for CardMan to 1200
|
|
// =================================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate = 1200;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BAUD_RATE;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate,
|
|
sizeof(SERIAL_BAUD_RATE));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_BAUD_RATE);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_BAUDRATE failed ! status = %ld\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
// ===============================
|
|
// set comm timeouts
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadIntervalTimeout = DEFAULT_READ_INTERVAL_TIMEOUT;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT + 5000;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutMultiplier = DEFAULT_READ_TOTAL_TIMEOUT_MULTIPLIER;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.WriteTotalTimeoutConstant = DEFAULT_WRITE_TOTAL_TIMEOUT_CONSTANT;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.WriteTotalTimeoutMultiplier = DEFAULT_WRITE_TOTAL_TIMEOUT_MULTIPLIER;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_TIMEOUTS;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts,
|
|
sizeof(SERIAL_TIMEOUTS));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_TIMEOUTS);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_TIMEOUTS failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ===============================
|
|
// set line control
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = NO_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = 7;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_LINE_CONTROL failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
|
|
|
|
// ===============================
|
|
// Set handflow
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.XonLimit = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.XoffLimit = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.FlowReplace = 0;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow.ControlHandShake = SERIAL_ERROR_ABORT | SERIAL_DTR_CONTROL;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_HANDFLOW;
|
|
|
|
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.HandFlow,
|
|
sizeof(SERIAL_HANDFLOW));
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_HANDFLOW);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_HANDFLOW failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
// ===============================
|
|
// set purge mask
|
|
// ===============================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.PurgeMask =
|
|
SERIAL_PURGE_TXABORT | SERIAL_PURGE_RXABORT |
|
|
SERIAL_PURGE_TXCLEAR | SERIAL_PURGE_RXCLEAR;
|
|
|
|
|
|
// ===============================
|
|
// clear RTS
|
|
// ===============================
|
|
status = SCCMN50M_ClearRTS(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_RTS failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
Wait(pSmartcardExtension,1);
|
|
|
|
// ===============================
|
|
// set DTR
|
|
// ===============================
|
|
|
|
status = SCCMN50M_SetDTR(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_DRT failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
Wait(pSmartcardExtension,1);
|
|
|
|
|
|
// ===============================
|
|
// set RTS
|
|
// ===============================
|
|
status = SCCMN50M_SetRTS(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_RTS failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
i=0;
|
|
while (1)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,1,&ulBytesRead,&bByteRead,sizeof(bByteRead));
|
|
if (status == STATUS_SUCCESS)
|
|
{
|
|
abReadBuffer[i++] = bByteRead;
|
|
if (bByteRead == 0x29)
|
|
{
|
|
ulPnPStringLength = i;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
if (ulPnPStringLength > 11 )
|
|
{
|
|
ulExtend = 0;
|
|
for (i=0;i<ulPnPStringLength;i++)
|
|
{
|
|
if (abReadBuffer[i] == 0x5C)
|
|
ulExtend++;
|
|
if (ulExtend == 4)
|
|
{
|
|
j = 0;
|
|
i++;
|
|
while (i < ulPnPStringLength - 3)
|
|
{
|
|
pSmartcardExtension->ReaderExtension->abDeviceDescription[j] = abReadBuffer[i];
|
|
i++;
|
|
j++;
|
|
}
|
|
SmartcardDebug(
|
|
DEBUG_DRIVER,
|
|
( "%s!Device=%s\n",
|
|
pSmartcardExtension->ReaderExtension->abDeviceDescription)
|
|
);
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ===================
|
|
// restore baud rate
|
|
// ===================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate = 38400;
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_BAUD_RATE;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.BaudRate.BaudRate,
|
|
sizeof(SERIAL_BAUD_RATE));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_BAUD_RATE);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_BAUDRATE failed ! status = %ld\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
// ====================
|
|
// retore line control
|
|
// ====================
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.StopBits = STOP_BITS_2;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.Parity = EVEN_PARITY;
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl.WordLength = 8;
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialIoControlCode = IOCTL_SERIAL_SET_LINE_CONTROL;
|
|
RtlCopyMemory(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
&pSmartcardExtension->ReaderExtension->SerialConfigData.LineControl,
|
|
sizeof(SERIAL_LINE_CONTROL));
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = sizeof(SERIAL_LINE_CONTROL);
|
|
pSmartcardExtension->SmartcardReply.BufferLength = 0;
|
|
|
|
status = SCCMN50M_SerialIo(pSmartcardExtension);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
SmartcardDebug(DEBUG_ERROR,("IOCTL_SERIAL_SET_LINE_CONTROL failed ! status = %x\n",status))
|
|
goto ExitGetDeviceDescription;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ExitGetDeviceDescription:
|
|
DebugStatus = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
|
|
if (status != STATUS_SUCCESS)
|
|
{ // map all errors to STATUS_UNSUCCESSFULL;
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!GetDeviceDescriptiong: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SetSyncParameters(IN PSMARTCARD_EXTENSION pSmartcardExtension )
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetSyncParameters: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
//DBGBreakPoint();
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.ulProtocol =
|
|
((PSYNC_PARAMETERS)pSmartcardExtension->IoRequest.RequestBuffer)->ulProtocol;
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.ulStateResetLineWhileReading =
|
|
((PSYNC_PARAMETERS)pSmartcardExtension->IoRequest.RequestBuffer)->ulStateResetLineWhileReading;
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.ulStateResetLineWhileWriting =
|
|
((PSYNC_PARAMETERS)pSmartcardExtension->IoRequest.RequestBuffer)->ulStateResetLineWhileWriting;
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.ulWriteDummyClocks =
|
|
((PSYNC_PARAMETERS)pSmartcardExtension->IoRequest.RequestBuffer)->ulWriteDummyClocks;
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.ulHeaderLen =
|
|
((PSYNC_PARAMETERS)pSmartcardExtension->IoRequest.RequestBuffer)->ulHeaderLen;
|
|
|
|
// Used for the 2 Wire Protocol. We must make a Card reset after Power On
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.fCardResetRequested = TRUE;
|
|
|
|
|
|
// return length of reply
|
|
*pSmartcardExtension->IoRequest.Information = 0L;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SetSyncParameters: Exit\n",
|
|
DRIVER_NAME)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
UCHAR
|
|
SCCMN50M_CalcTxControlByte (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBitsToWrite )
|
|
{
|
|
UCHAR bTxControlByte = 0;
|
|
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.ulProtocol == SCARD_PROTOCOL_2WBP)
|
|
{
|
|
bTxControlByte = CLOCK_FORCED_2WBP;
|
|
}
|
|
else
|
|
{
|
|
if (ulBitsToWrite >= 255 * 8)
|
|
bTxControlByte |= TRANSMIT_A8;
|
|
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.ulStateResetLineWhileWriting ==
|
|
SCARD_RESET_LINE_HIGH)
|
|
bTxControlByte |= SYNC_RESET_LINE_HIGH;
|
|
}
|
|
|
|
bTxControlByte |= (BYTE)((ulBitsToWrite-1) & 0x00000007);
|
|
|
|
return bTxControlByte;
|
|
|
|
}
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
UCHAR
|
|
SCCMN50M_CalcTxLengthByte (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBitsToWrite )
|
|
{
|
|
UCHAR bTxLengthByte = 0;
|
|
|
|
bTxLengthByte = (BYTE)( ((ulBitsToWrite - 1) >> 3) + 1);
|
|
|
|
return bTxLengthByte;
|
|
}
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
UCHAR
|
|
SCCMN50M_CalcRxControlByte (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBitsToRead )
|
|
{
|
|
UCHAR bRxControlByte = 0;
|
|
|
|
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.ulStateResetLineWhileReading ==
|
|
SCARD_RESET_LINE_HIGH)
|
|
bRxControlByte |= SYNC_RESET_LINE_HIGH;
|
|
|
|
if (ulBitsToRead == 0)
|
|
{
|
|
ulBitsToRead = pSmartcardExtension->ReaderExtension->SyncParameters.ulWriteDummyClocks;
|
|
bRxControlByte |= SYNC_DUMMY_RECEIVE;
|
|
}
|
|
|
|
if (ulBitsToRead > 255 * 8)
|
|
bRxControlByte |= RECEIVE_A8;
|
|
|
|
bRxControlByte |= (BYTE)( (ulBitsToRead-1) & 0x00000007);
|
|
|
|
return bRxControlByte;
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
UCHAR
|
|
SCCMN50M_CalcRxLengthByte (IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
IN ULONG ulBitsToRead )
|
|
{
|
|
UCHAR bRxLengthByte = 0;
|
|
|
|
// if (pSmartcardExtension->ReaderExtension->SyncParameters.ulProtocol == SCARD_PROTOCOL_3WBP)
|
|
// {
|
|
if (ulBitsToRead == 0)
|
|
ulBitsToRead = pSmartcardExtension->ReaderExtension->SyncParameters.ulWriteDummyClocks;
|
|
|
|
bRxLengthByte = (BYTE)( ((ulBitsToRead - 1) >> 3) + 1);
|
|
// }
|
|
|
|
return bRxLengthByte;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_SyncCardPowerOn (
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
UCHAR pbAtrBuffer[MAXIMUM_ATR_LENGTH];
|
|
UCHAR abSyncAtrBuffer[MAXIMUM_ATR_LENGTH];
|
|
ULONG ulAtrLength = 0;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SyncCardPowerOn: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
status = SCCMN50M_UseSyncStrategy(pSmartcardExtension,
|
|
&ulAtrLength,
|
|
pbAtrBuffer,
|
|
sizeof(pbAtrBuffer));
|
|
|
|
|
|
abSyncAtrBuffer[0] = 0x3B;
|
|
abSyncAtrBuffer[1] = 0x04;
|
|
MemCpy(&abSyncAtrBuffer[2],
|
|
sizeof(abSyncAtrBuffer)-2,
|
|
pbAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
|
|
ulAtrLength += 2;
|
|
|
|
MemCpy(pSmartcardExtension->CardCapabilities.ATR.Buffer,
|
|
sizeof(pSmartcardExtension->CardCapabilities.ATR.Buffer),
|
|
abSyncAtrBuffer,
|
|
ulAtrLength);
|
|
|
|
pSmartcardExtension->CardCapabilities.ATR.Length = (UCHAR)(ulAtrLength);
|
|
|
|
pSmartcardExtension->ReaderCapabilities.CurrentState = SCARD_SPECIFIC;
|
|
pSmartcardExtension->CardCapabilities.Protocol.Selected = SCARD_PROTOCOL_T0;
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.fCardResetRequested = TRUE;
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.fCardPowerRequested = FALSE;
|
|
|
|
|
|
SmartcardDebug(DEBUG_ATR,("ATR of synchronous smart card : %2.2x %2.2x %2.2x %2.2x\n",
|
|
pbAtrBuffer[0],pbAtrBuffer[1],pbAtrBuffer[2],pbAtrBuffer[3]));
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!SyncCardPowerOn: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
|
|
return status;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_Transmit2WBP(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
UCHAR bWriteBuffer [128];
|
|
UCHAR bReadBuffer [128];
|
|
UCHAR bTxControlByte;
|
|
UCHAR bTxLengthByte;
|
|
UCHAR bRxControlByte;
|
|
UCHAR bRxLengthByte;
|
|
ULONG ulBytesToWrite;
|
|
ULONG ulBytesToRead;
|
|
ULONG ulBitsToWrite;
|
|
ULONG ulBitsToRead;
|
|
ULONG ulBytesRead;
|
|
// ULONG ulBitsRead;
|
|
ULONG ulBytesStillToRead;
|
|
ULONG ulMaxIFSD;
|
|
PCHAR pbInData;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!Transmit2WBP: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** Power smartcard - if smartcard was removed and reinserted **/
|
|
/*-----------------------------------------------------------------------*/
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.fCardPowerRequested == TRUE)
|
|
{
|
|
status = SCCMN50M_SyncCardPowerOn (pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
}
|
|
|
|
|
|
pbInData = pSmartcardExtension->IoRequest.RequestBuffer + sizeof(SYNC_TRANSFER);
|
|
ulBitsToWrite = ((PSYNC_TRANSFER)(pSmartcardExtension->IoRequest.RequestBuffer))->ulSyncBitsToWrite;
|
|
ulBitsToRead = ((PSYNC_TRANSFER)(pSmartcardExtension->IoRequest.RequestBuffer))->ulSyncBitsToRead;
|
|
ulBytesToWrite = ulBitsToWrite/8;
|
|
ulBytesToRead = ulBitsToRead/8 + (ulBitsToRead % 8 ? 1 : 0);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// check buffer sizes
|
|
/*-----------------------------------------------------------------------*/
|
|
ulMaxIFSD = ATTR_MAX_IFSD_CARDMAN_II;
|
|
|
|
|
|
if (ulBytesToRead > ulMaxIFSD ||
|
|
ulBytesToRead > pSmartcardExtension->SmartcardReply.BufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
if (ulBytesToWrite > pSmartcardExtension->SmartcardRequest.BufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = ulBytesToWrite+1;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy data to the Smartcard Request Buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
(pSmartcardExtension->SmartcardRequest.Buffer)[0] = '\x0F';
|
|
MemCpy((pSmartcardExtension->SmartcardRequest.Buffer+1),
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
pbInData,
|
|
ulBytesToWrite);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy data to the write buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy((bWriteBuffer),
|
|
sizeof(bWriteBuffer),
|
|
pSmartcardExtension->SmartcardRequest.Buffer,
|
|
(ulBytesToWrite+1));
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// set SYNC protocol flag for CardMan
|
|
/*-----------------------------------------------------------------------*/
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_SYN);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// Header
|
|
/*-----------------------------------------------------------------------*/
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.fCardResetRequested == TRUE)
|
|
{
|
|
status = SCCMN50M_ResetCard2WBP(pSmartcardExtension);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
pSmartcardExtension->ReaderExtension->SyncParameters.fCardResetRequested = FALSE;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// 1. Send Carman-Header 4-Byte
|
|
// 2. Send 0x0F, that builds a HIGH-LOW Edge for 4432 CC
|
|
// 3. Send the Data (CC command = 3 Byte)
|
|
/*-----------------------------------------------------------------------*/
|
|
bTxControlByte = SCCMN50M_CalcTxControlByte(pSmartcardExtension,ulBitsToWrite);
|
|
bTxLengthByte = (BYTE)(SCCMN50M_CalcTxLengthByte(pSmartcardExtension,ulBitsToWrite)+1);
|
|
bRxControlByte = 0;
|
|
bRxLengthByte = 0;
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
bTxControlByte,
|
|
bTxLengthByte,
|
|
bRxControlByte,
|
|
bRxLengthByte);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// write data to card
|
|
/*-----------------------------------------------------------------------*/
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
(ulBytesToWrite+1),
|
|
bWriteBuffer);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// read CardMan Header
|
|
// no Data from CC received
|
|
/*-----------------------------------------------------------------------*/
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
2,
|
|
&ulBytesRead,
|
|
bReadBuffer,
|
|
sizeof(bReadBuffer));
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// 1. Send Carman-Header 4-Byte
|
|
// 2. Send 0xF0, that builds a LOW-HIGH Edge for 4432 CC
|
|
// 3. Now the receiviing of card-data begins
|
|
/*-----------------------------------------------------------------------*/
|
|
bTxControlByte = SCCMN50M_CalcTxControlByte(pSmartcardExtension, 8);
|
|
bTxLengthByte = SCCMN50M_CalcTxLengthByte(pSmartcardExtension, 8);
|
|
bRxControlByte = SCCMN50M_CalcRxControlByte(pSmartcardExtension,ulBitsToRead);
|
|
bRxLengthByte = SCCMN50M_CalcRxLengthByte(pSmartcardExtension,ulBitsToRead);
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
bTxControlByte,
|
|
bTxLengthByte,
|
|
bRxControlByte,
|
|
bRxLengthByte);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// in this sequnce SCCMN50M_WriteCardMan must not send the Config string.
|
|
// write 0xF0 -> is the trigger to read data from card or start the
|
|
// processing.
|
|
/*-----------------------------------------------------------------------*/
|
|
pSmartcardExtension->ReaderExtension->NoConfig = TRUE;
|
|
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
1, // one byte to write
|
|
"\xF0"); // LOW-HIGH - Edge
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// read CardMan Header
|
|
// Data from CC received will be received
|
|
/*-----------------------------------------------------------------------*/
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
2,
|
|
&ulBytesRead,
|
|
bReadBuffer,
|
|
sizeof(bReadBuffer));
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// Read the data string
|
|
/*-----------------------------------------------------------------------*/
|
|
ulBytesStillToRead = (ULONG)bReadBuffer[1];
|
|
|
|
if (bReadBuffer[0] & RECEIVE_A8)
|
|
ulBytesStillToRead += 256;
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
ulBytesStillToRead,
|
|
&ulBytesRead,
|
|
bReadBuffer,
|
|
sizeof(bReadBuffer));
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** calculate data length in bits - this value is not used **/
|
|
/*-----------------------------------------------------------------------*/
|
|
// ulBitsRead = ((ulBytesRead-1) * 8) + ((ulBitsToRead-1 ) & 0x00000007) + 1;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** shift the bits in the last byte to the correct position **/
|
|
/*-----------------------------------------------------------------------*/
|
|
bReadBuffer[ulBytesRead-1] >>= (7 - ((ulBitsToRead-1) & 0x00000007));
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// the first bit of the returned string is lost
|
|
// so we must shift the whole data string one bit left
|
|
// the first bit of the first data byte is lost while reading
|
|
// this bit maybe incorrect
|
|
/*-----------------------------------------------------------------------*/
|
|
SCCMN50M_Shift_Msg(bReadBuffer, ulBytesRead);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy received bytes to Smartcard Reply Buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferSize,
|
|
bReadBuffer,
|
|
ulBytesRead);
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = ulBytesRead;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy received bytes to IoReply Buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(pSmartcardExtension->IoRequest.ReplyBuffer,
|
|
pSmartcardExtension->IoRequest.ReplyBufferLength,
|
|
pSmartcardExtension->SmartcardReply.Buffer,
|
|
ulBytesRead);
|
|
|
|
*(pSmartcardExtension->IoRequest.Information) = ulBytesRead;
|
|
|
|
|
|
ExitTransmit2WBP:
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!Transmit2WBP: Exit\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
VOID
|
|
SCCMN50M_Shift_Msg (PUCHAR pbBuffer,
|
|
ULONG ulMsgLen)
|
|
{
|
|
UCHAR bTmp1, bTmp2;
|
|
int i;
|
|
|
|
for (i=(int)ulMsgLen-1; i>=0; i--)
|
|
{
|
|
bTmp1=(BYTE)((pbBuffer[i] >> 7) & 0x01); /* bTmp1 = bit 7 naechstes byte */
|
|
if (i+1 != (int)ulMsgLen)
|
|
{
|
|
bTmp2=(BYTE)((pbBuffer[i+1] << 1) | bTmp1);
|
|
pbBuffer[i+1] = bTmp2;
|
|
}
|
|
}
|
|
|
|
pbBuffer[0] = (BYTE)(pbBuffer[0] << 1);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
Reset Card 4442
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_ResetCard2WBP(IN PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
BYTE bBuffer[10];
|
|
ULONG ulBytesRead;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// Enter Card Reset
|
|
/*-----------------------------------------------------------------------*/
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ResetCard2WBP: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// bTxControlByte = 0;
|
|
// bTxLengthByte = 0;
|
|
// bRxControlByte = RESET_CARD;
|
|
// bRxLengthByte = 5;
|
|
/*-----------------------------------------------------------------------*/
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
0, // bTxControlByte
|
|
0, // bTxLengthByte
|
|
COLD_RESET, // bRxControlByte
|
|
5); // bRxLengthByte
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// write data to card
|
|
/*-----------------------------------------------------------------------*/
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
0,
|
|
NULL);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitResetCard2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// read CardMan Header
|
|
/*-----------------------------------------------------------------------*/
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
2, &ulBytesRead, bBuffer, sizeof(bBuffer));
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitResetCard2WBP;
|
|
}
|
|
|
|
if (bBuffer[1] != 5)
|
|
{
|
|
status = !STATUS_SUCCESS;
|
|
goto ExitResetCard2WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// read ATR
|
|
/*-----------------------------------------------------------------------*/
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
5, &ulBytesRead, bBuffer, sizeof(bBuffer));
|
|
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitResetCard2WBP;
|
|
}
|
|
|
|
SmartcardDebug(DEBUG_ATR,("%s!Card Reset ATR : %02x %02x %02x %02x\n",
|
|
DRIVER_NAME,bBuffer[0],bBuffer[1],bBuffer[2],bBuffer[3]));
|
|
|
|
ExitResetCard2WBP:
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!ResetCard2WBP: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_Transmit3WBP(PSMARTCARD_EXTENSION pSmartcardExtension)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
UCHAR bWriteBuffer [128];
|
|
UCHAR bReadBuffer [128];
|
|
UCHAR bTxControlByte;
|
|
UCHAR bTxLengthByte;
|
|
UCHAR bRxControlByte;
|
|
UCHAR bRxLengthByte;
|
|
ULONG ulBytesToWrite;
|
|
ULONG ulBytesToRead;
|
|
ULONG ulBitsToWrite;
|
|
ULONG ulBitsToRead;
|
|
ULONG ulBytesRead;
|
|
// ULONG ulBitsRead;
|
|
ULONG ulBytesStillToRead;
|
|
ULONG ulMaxIFSD;
|
|
PCHAR pbInData;
|
|
|
|
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!Transmit3WBP: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
// DBGBreakPoint();
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** Power smartcard - if smartcard was removed and reinserted **/
|
|
/*-----------------------------------------------------------------------*/
|
|
if (pSmartcardExtension->ReaderExtension->SyncParameters.fCardPowerRequested == TRUE)
|
|
{
|
|
status = SCCMN50M_SyncCardPowerOn (pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
}
|
|
|
|
|
|
pbInData = pSmartcardExtension->IoRequest.RequestBuffer + sizeof(SYNC_TRANSFER);
|
|
ulBitsToWrite = ((PSYNC_TRANSFER)(pSmartcardExtension->IoRequest.RequestBuffer))->ulSyncBitsToWrite;
|
|
ulBitsToRead = ((PSYNC_TRANSFER)(pSmartcardExtension->IoRequest.RequestBuffer))->ulSyncBitsToRead;
|
|
ulBytesToWrite = ulBitsToWrite/8;
|
|
ulBytesToRead = ulBitsToRead/8 + (ulBitsToRead % 8 ? 1 : 0);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// check buffer sizes
|
|
/*-----------------------------------------------------------------------*/
|
|
ulMaxIFSD = ATTR_MAX_IFSD_CARDMAN_II;
|
|
|
|
|
|
if (ulBytesToRead > ulMaxIFSD ||
|
|
ulBytesToRead > pSmartcardExtension->SmartcardReply.BufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
|
|
if (ulBytesToWrite > pSmartcardExtension->SmartcardRequest.BufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
|
|
pSmartcardExtension->SmartcardRequest.BufferLength = ulBytesToWrite;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy data to the Smartcard Request Buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(pSmartcardExtension->SmartcardRequest.Buffer,
|
|
pSmartcardExtension->SmartcardRequest.BufferSize,
|
|
pbInData,
|
|
ulBytesToWrite);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy data to the write buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(bWriteBuffer,
|
|
sizeof(bWriteBuffer),
|
|
pSmartcardExtension->SmartcardRequest.Buffer,
|
|
ulBytesToWrite);
|
|
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// set SYNC protocol flag for CardMan
|
|
/*-----------------------------------------------------------------------*/
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_SYN);
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// build cardman header
|
|
/*-----------------------------------------------------------------------*/
|
|
bTxControlByte = SCCMN50M_CalcTxControlByte(pSmartcardExtension,ulBitsToWrite);
|
|
bTxLengthByte = SCCMN50M_CalcTxLengthByte(pSmartcardExtension,ulBitsToWrite);
|
|
bRxControlByte = SCCMN50M_CalcRxControlByte(pSmartcardExtension,ulBitsToRead);
|
|
bRxLengthByte = SCCMN50M_CalcRxLengthByte(pSmartcardExtension,ulBitsToRead);
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,
|
|
bTxControlByte,
|
|
bTxLengthByte,
|
|
bRxControlByte,
|
|
bRxLengthByte);
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** write data to card **/
|
|
/*-----------------------------------------------------------------------*/
|
|
status = SCCMN50M_WriteCardMan (pSmartcardExtension,
|
|
ulBytesToWrite,
|
|
bWriteBuffer);
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
/** read CardMan Header **/
|
|
/*-----------------------------------------------------------------------*/
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// calc data length to receive
|
|
/*-----------------------------------------------------------------------*/
|
|
ulBytesStillToRead = (ULONG)(bReadBuffer[1]);
|
|
if (bReadBuffer[0] & RECEIVE_A8)
|
|
ulBytesStillToRead += 256;
|
|
|
|
|
|
// read data from card
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,
|
|
ulBytesStillToRead,
|
|
&ulBytesRead,
|
|
bReadBuffer,
|
|
sizeof(bReadBuffer));
|
|
if (NT_ERROR(status))
|
|
{
|
|
goto ExitTransmit3WBP;
|
|
}
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// calculate data length in bits - this value is not used
|
|
/*-----------------------------------------------------------------------*/
|
|
// ulBitsRead = ((ulBytesRead-1) * 8) + ((ulBitsToRead-1 ) & 0x00000007) + 1;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// shift the bits in the last byte to the correct position
|
|
/*-----------------------------------------------------------------------*/
|
|
bReadBuffer[ulBytesRead-1] >>= (7 - ((ulBitsToRead-1) & 0x00000007));
|
|
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy received bytes to Smartcard Reply Buffer
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(pSmartcardExtension->SmartcardReply.Buffer,
|
|
pSmartcardExtension->SmartcardReply.BufferSize,
|
|
bReadBuffer,
|
|
ulBytesRead);
|
|
|
|
pSmartcardExtension->SmartcardReply.BufferLength = ulBytesRead;
|
|
|
|
/*-----------------------------------------------------------------------*/
|
|
// copy received bytes to IoReply Buffer
|
|
// this Memcpy should respond to SmartcardRawReply function
|
|
/*-----------------------------------------------------------------------*/
|
|
MemCpy(pSmartcardExtension->IoRequest.ReplyBuffer,
|
|
pSmartcardExtension->IoRequest.ReplyBufferLength,
|
|
pSmartcardExtension->SmartcardReply.Buffer,
|
|
ulBytesRead);
|
|
|
|
*pSmartcardExtension->IoRequest.Information = ulBytesRead;
|
|
|
|
|
|
ExitTransmit3WBP:
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!Transmit3WBP: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This function powers a synchronous smart card.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
NTSTATUS
|
|
SCCMN50M_UseSyncStrategy (
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension,
|
|
OUT PULONG pulAtrLength,
|
|
OUT PUCHAR pbAtrBuffer,
|
|
IN ULONG ulAtrBufferSize
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
NTSTATUS DebugStatus;
|
|
UCHAR bReadBuffer[SCARD_ATR_LENGTH];
|
|
ULONG ulBytesRead;
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!UseSyncStrategy: Enter\n",
|
|
DRIVER_NAME)
|
|
);
|
|
//DBGBreakPoint();
|
|
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,SYNC_ATR_RX_CONTROL,ATR_LEN_SYNC);
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.ResetDelay = SYNC_RESET_DELAY;
|
|
pSmartcardExtension->ReaderExtension->CardManConfig.CardStopBits = 0x02;
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY );
|
|
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_SYN);
|
|
|
|
status = SCCMN50M_ResyncCardManII(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
// read state and length
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
if (bReadBuffer[1] < MIN_ATR_LEN )
|
|
{
|
|
// read all remaining bytes from the CardMan
|
|
DebugStatus = SCCMN50M_ReadCardMan(pSmartcardExtension,bReadBuffer[1],&ulBytesRead,bReadBuffer,sizeof(bReadBuffer));
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
if (bReadBuffer[1] > ulAtrBufferSize)
|
|
{
|
|
status = STATUS_BUFFER_OVERFLOW;
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
// read ATR
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,bReadBuffer[1],pulAtrLength,pbAtrBuffer,ulAtrBufferSize);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
|
|
if (pbAtrBuffer[0] == 0x00 ||
|
|
pbAtrBuffer[0] == 0xff )
|
|
{
|
|
status = STATUS_UNRECOGNIZED_MEDIA;
|
|
goto ExitPowerSynchronousCard;
|
|
}
|
|
pSmartcardExtension->ReaderExtension->fRawModeNecessary = TRUE;
|
|
|
|
|
|
ExitPowerSynchronousCard:
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!UseSyncStrategy: Exit %lx\n",
|
|
DRIVER_NAME,status)
|
|
);
|
|
SCCMN50M_ClearSCRControlFlags(pSmartcardExtension,IGNORE_PARITY | CM2_GET_ATR);
|
|
SCCMN50M_ClearCardManHeader(pSmartcardExtension);
|
|
return status;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
Routine Description:
|
|
|
|
This function checks if the inserted card is a synchronous one
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
|
|
Return Value:
|
|
|
|
*****************************************************************************/
|
|
BOOLEAN
|
|
SCCMN50M_IsAsynchronousSmartCard(
|
|
IN PSMARTCARD_EXTENSION pSmartcardExtension
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
UCHAR ReadBuffer[3];
|
|
ULONG ulBytesRead;
|
|
BOOLEAN fIsAsynchronousSmartCard = TRUE;
|
|
UCHAR abATR[33];
|
|
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IsAsynchronousSmartcard: Enter \n",
|
|
DRIVER_NAME)
|
|
);
|
|
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = 200;
|
|
|
|
// 3MHz smart card ?
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,0,1);
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY | CM2_GET_ATR);
|
|
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_3MHZ);
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
// read state and length
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
if (status == STATUS_SUCCESS &&
|
|
ReadBuffer[1] == 0x01 )
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------
|
|
// power off card
|
|
// ---------------------------------------
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
|
|
// 5MHz smart card ?
|
|
SCCMN50M_SetCardManHeader(pSmartcardExtension,0,0,0,1);
|
|
|
|
SCCMN50M_ClearCardControlFlags(pSmartcardExtension,ALL_FLAGS);
|
|
|
|
SCCMN50M_SetSCRControlFlags(pSmartcardExtension,CARD_POWER| IGNORE_PARITY | CM2_GET_ATR);
|
|
|
|
SCCMN50M_SetCardControlFlags(pSmartcardExtension,ENABLE_5MHZ);
|
|
|
|
// write config + header
|
|
status = SCCMN50M_WriteCardMan(pSmartcardExtension,0,NULL);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
pSmartcardExtension->ReaderExtension->ToRHConfig = FALSE;
|
|
|
|
// read state and length
|
|
status = SCCMN50M_ReadCardMan(pSmartcardExtension,2,&ulBytesRead,ReadBuffer,sizeof(ReadBuffer));
|
|
if (status == STATUS_SUCCESS &&
|
|
ReadBuffer[1] == 0x01 )
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
// now we assume that it is a synchronous smart card
|
|
fIsAsynchronousSmartCard = FALSE;
|
|
// ---------------------------------------
|
|
// power off card
|
|
// ---------------------------------------
|
|
status = SCCMN50M_PowerOff(pSmartcardExtension);
|
|
if (status != STATUS_SUCCESS)
|
|
{
|
|
goto ExitIsAsynchronousSmartCard;
|
|
}
|
|
|
|
|
|
|
|
ExitIsAsynchronousSmartCard:
|
|
pSmartcardExtension->ReaderExtension->SerialConfigData.Timeouts.ReadTotalTimeoutConstant = DEFAULT_READ_TOTAL_TIMEOUT_CONSTANT;
|
|
SmartcardDebug(
|
|
DEBUG_TRACE,
|
|
( "%s!IsAsynchronousSmartcard: Exit \n",
|
|
DRIVER_NAME)
|
|
);
|
|
return fIsAsynchronousSmartCard;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
* History:
|
|
* $Log: sccmcb.c $
|
|
* Revision 1.7 2001/01/22 08:39:41 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.6 2000/09/25 10:46:22 WFrischauf
|
|
* No comment given
|
|
*
|
|
* Revision 1.5 2000/08/24 09:05:44 TBruendl
|
|
* No comment given
|
|
*
|
|
* Revision 1.4 2000/08/16 08:24:04 TBruendl
|
|
* warning :uninitialized memory removed
|
|
*
|
|
* Revision 1.3 2000/07/28 09:24:12 TBruendl
|
|
* Changes for OMNIKEY on Whistler CD
|
|
*
|
|
* Revision 1.16 2000/06/27 11:56:28 TBruendl
|
|
* workaraound for SAMOR smart cards with invalid ATR (ITSEC)
|
|
*
|
|
* Revision 1.15 2000/06/08 10:08:47 TBruendl
|
|
* bug fix : warm reset for ScfW
|
|
*
|
|
* Revision 1.14 2000/05/23 09:58:26 TBruendl
|
|
* OMNIKEY 3.0.0.1
|
|
*
|
|
* Revision 1.13 2000/04/13 08:07:22 TBruendl
|
|
* PPS bug fix for SCfW
|
|
*
|
|
* Revision 1.12 2000/04/04 07:52:18 TBruendl
|
|
* problem with the new WfsC fixed
|
|
*
|
|
* Revision 1.11 2000/03/03 09:50:50 TBruendl
|
|
* No comment given
|
|
*
|
|
* Revision 1.10 2000/03/01 09:32:04 TBruendl
|
|
* R02.20.0
|
|
*
|
|
* Revision 1.9 2000/01/04 10:40:33 TBruendl
|
|
* bug fix: status instead of DebugStatus used
|
|
*
|
|
* Revision 1.8 1999/12/16 14:10:16 TBruendl
|
|
* After transparent mode has been left, the status is read from the CardMan to be sure that
|
|
* the new settings are effective.
|
|
*
|
|
* Revision 1.6 1999/12/13 07:55:38 TBruendl
|
|
* Bug fix for P+ druing initialization
|
|
* PTS for 4.9 mhz smartcards added
|
|
*
|
|
* Revision 1.5 1999/11/04 07:53:21 WFrischauf
|
|
* bug fixes due to error reports 2 - 7
|
|
*
|
|
* Revision 1.4 1999/07/12 12:49:04 TBruendl
|
|
* Bug fix: Resync after GET_DEVICE_DESCRIPTION
|
|
* Power On SLE4428
|
|
*
|
|
* Revision 1.3 1999/06/10 09:03:57 TBruendl
|
|
* No comment given
|
|
*
|
|
* Revision 1.2 1999/02/25 10:12:22 TBruendl
|
|
* No comment given
|
|
*
|
|
* Revision 1.1 1999/02/02 13:34:37 TBruendl
|
|
* This is the first release (R01.00) of the IFD handler for CardMan running under NT5.0.
|
|
*
|
|
*
|
|
*****************************************************************************/
|
|
|
|
|