2012-08-15 01:09:01 +04:00
|
|
|
/**
|
2012-10-09 05:00:07 +04:00
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
2012-08-15 01:09:01 +04:00
|
|
|
* Smartcard Device Service Virtual Channel
|
|
|
|
*
|
|
|
|
* Copyright (C) Alexi Volkov <alexi@myrealbox.com> 2006
|
|
|
|
* Copyright 2011 O.S. Systems Software Ltda.
|
|
|
|
* Copyright 2011 Anthony Tong <atong@trustedcs.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
2011-10-15 19:30:10 +04:00
|
|
|
#include "config.h"
|
2012-08-15 01:09:01 +04:00
|
|
|
#endif
|
2011-10-15 19:30:10 +04:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2014-02-11 07:12:13 +04:00
|
|
|
#include <assert.h>
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-20 08:49:08 +04:00
|
|
|
#include <winpr/crt.h>
|
2012-12-14 08:54:54 +04:00
|
|
|
#include <winpr/print.h>
|
2013-03-22 00:45:25 +04:00
|
|
|
#include <winpr/stream.h>
|
2014-04-03 23:27:55 +04:00
|
|
|
#include <winpr/smartcard.h>
|
2013-03-22 01:58:18 +04:00
|
|
|
|
|
|
|
#include <freerdp/freerdp.h>
|
2012-10-09 05:00:07 +04:00
|
|
|
#include <freerdp/channels/rdpdr.h>
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
#include "smartcard_main.h"
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
const char* smartcard_get_ioctl_string(UINT32 ioControlCode, BOOL funcName)
|
2014-04-04 08:56:24 +04:00
|
|
|
{
|
|
|
|
switch (ioControlCode)
|
|
|
|
{
|
|
|
|
case SCARD_IOCTL_ESTABLISHCONTEXT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardEstablishContext" : "SCARD_IOCTL_ESTABLISHCONTEXT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_RELEASECONTEXT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardReleaseContext" : "SCARD_IOCTL_RELEASECONTEXT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_ISVALIDCONTEXT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardIsValidContext" : "SCARD_IOCTL_ISVALIDCONTEXT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERGROUPSA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardListReaderGroupsA" : "SCARD_IOCTL_LISTREADERGROUPSA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERGROUPSW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardListReaderGroupsW" : "SCARD_IOCTL_LISTREADERGROUPSW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERSA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardListReadersA" : "SCARD_IOCTL_LISTREADERSA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERSW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardListReadersW" : "SCARD_IOCTL_LISTREADERSW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERGROUPA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardIntroduceReaderGroupA" : "SCARD_IOCTL_INTRODUCEREADERGROUPA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERGROUPW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardIntroduceReaderGroupW" : "SCARD_IOCTL_INTRODUCEREADERGROUPW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_FORGETREADERGROUPA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardForgetReaderGroupA" : "SCARD_IOCTL_FORGETREADERGROUPA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_FORGETREADERGROUPW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardForgetReaderGroupW" : "SCARD_IOCTL_FORGETREADERGROUPW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardIntroduceReaderA" : "SCARD_IOCTL_INTRODUCEREADERA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardIntroduceReaderW" : "SCARD_IOCTL_INTRODUCEREADERW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_FORGETREADERA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardForgetReaderA" : "SCARD_IOCTL_FORGETREADERA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_FORGETREADERW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardForgetReaderW" : "SCARD_IOCTL_FORGETREADERW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_ADDREADERTOGROUPA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardAddReaderToGroupA" : "SCARD_IOCTL_ADDREADERTOGROUPA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_ADDREADERTOGROUPW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardAddReaderToGroupW" : "SCARD_IOCTL_ADDREADERTOGROUPW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_REMOVEREADERFROMGROUPA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardRemoveReaderFromGroupA" : "SCARD_IOCTL_REMOVEREADERFROMGROUPA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_REMOVEREADERFROMGROUPW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardRemoveReaderFromGroupW" : "SCARD_IOCTL_REMOVEREADERFROMGROUPW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardLocateCardsA" : "SCARD_IOCTL_LOCATECARDSA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardLocateCardsW" : "SCARD_IOCTL_LOCATECARDSW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETSTATUSCHANGEA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetStatusChangeA" : "SCARD_IOCTL_GETSTATUSCHANGEA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETSTATUSCHANGEW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetStatusChangeW" : "SCARD_IOCTL_GETSTATUSCHANGEW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_CANCEL:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardCancel" : "SCARD_IOCTL_CANCEL";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_CONNECTA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardConnectA" : "SCARD_IOCTL_CONNECTA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_CONNECTW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardConnectW" : "SCARD_IOCTL_CONNECTW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_RECONNECT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardReconnect" : "SCARD_IOCTL_RECONNECT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_DISCONNECT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardDisconnect" : "SCARD_IOCTL_DISCONNECT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_BEGINTRANSACTION:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardBeginTransaction" : "SCARD_IOCTL_BEGINTRANSACTION";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_ENDTRANSACTION:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardEndTransaction" : "SCARD_IOCTL_ENDTRANSACTION";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_STATE:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardState" : "SCARD_IOCTL_STATE";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_STATUSA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardStatusA" : "SCARD_IOCTL_STATUSA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_STATUSW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardStatusW" : "SCARD_IOCTL_STATUSW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_TRANSMIT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardTransmit" : "SCARD_IOCTL_TRANSMIT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_CONTROL:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardControl" : "SCARD_IOCTL_CONTROL";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETATTRIB:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetAttrib" : "SCARD_IOCTL_GETATTRIB";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_SETATTRIB:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardSetAttrib" : "SCARD_IOCTL_SETATTRIB";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_ACCESSSTARTEDEVENT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardAccessStartedEvent" : "SCARD_IOCTL_ACCESSSTARTEDEVENT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSBYATRA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardLocateCardsByATRA" : "SCARD_IOCTL_LOCATECARDSBYATRA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSBYATRW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardLocateCardsByATRB" : "SCARD_IOCTL_LOCATECARDSBYATRW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_READCACHEA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardReadCacheA" : "SCARD_IOCTL_READCACHEA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_READCACHEW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardReadCacheW" : "SCARD_IOCTL_READCACHEW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_WRITECACHEA:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardWriteCacheA" : "SCARD_IOCTL_WRITECACHEA";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_WRITECACHEW:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardWriteCacheW" : "SCARD_IOCTL_WRITECACHEW";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETTRANSMITCOUNT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetTransmitCount" : "SCARD_IOCTL_GETTRANSMITCOUNT";
|
2014-04-06 00:06:46 +04:00
|
|
|
case SCARD_IOCTL_RELEASESTARTEDEVENT:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardReleaseStartedEvent" : "SCARD_IOCTL_RELEASESTARTEDEVENT";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETREADERICON:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetReaderIcon" : "SCARD_IOCTL_GETREADERICON";
|
2014-04-04 08:56:24 +04:00
|
|
|
case SCARD_IOCTL_GETDEVICETYPEID:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardGetDeviceTypeId" : "SCARD_IOCTL_GETDEVICETYPEID";
|
2014-04-04 08:56:24 +04:00
|
|
|
default:
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardUnknown" : "SCARD_IOCTL_UNKNOWN";
|
2014-04-04 08:56:24 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
return funcName ? "SCardUnknown" : "SCARD_IOCTL_UNKNOWN";
|
2014-04-04 08:56:24 +04:00
|
|
|
}
|
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
static void smartcard_output_alignment(IRP* irp, UINT32 seed)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-10-09 11:26:39 +04:00
|
|
|
const UINT32 field_lengths = 20;/* Remove the lengths of the fields
|
2012-08-02 02:05:45 +04:00
|
|
|
* RDPDR_HEADER, DeviceID,
|
|
|
|
* CompletionID, and IoStatus
|
|
|
|
* of Section 2.2.1.5.5 of MS-RDPEFS.
|
|
|
|
*/
|
2013-04-30 06:35:15 +04:00
|
|
|
UINT32 size = Stream_GetPosition(irp->output) - field_lengths;
|
2012-10-09 11:26:39 +04:00
|
|
|
UINT32 add = (seed - (size % seed)) % seed;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
|
|
|
if (add > 0)
|
2013-05-09 00:27:21 +04:00
|
|
|
Stream_Zero(irp->output, add);
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
size_t smartcard_multi_string_length_a(const char* msz)
|
|
|
|
{
|
|
|
|
char* p = (char*) msz;
|
|
|
|
|
|
|
|
if (!p)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
while (p[0] || p[1])
|
|
|
|
p++;
|
|
|
|
|
|
|
|
return (p - msz);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t smartcard_multi_string_length_w(const WCHAR* msz)
|
|
|
|
{
|
|
|
|
WCHAR* p = (WCHAR*) msz;
|
|
|
|
|
|
|
|
if (!p)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
while (p[0] || p[1])
|
|
|
|
p++;
|
|
|
|
|
|
|
|
return (p - msz);
|
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_EstablishContext(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
UINT32 status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDCONTEXT hContext = -1;
|
2014-04-06 05:17:54 +04:00
|
|
|
EstablishContext_Call call;
|
|
|
|
EstablishContext_Return ret;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
status = smartcard_unpack_establish_context_call(smartcard, irp->input, &call);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
status = SCardEstablishContext(call.dwScope, NULL, NULL, &hContext);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
smartcard->hContext = hContext;
|
2014-04-06 05:17:54 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.Context.cbContext = sizeof(ULONG_PTR);
|
|
|
|
ret.Context.pbContext = (UINT64) hContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
smartcard_pack_establish_context_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2014-04-04 01:29:12 +04:00
|
|
|
|
2013-09-13 11:47:18 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_ReleaseContext(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2013-09-12 14:18:35 +04:00
|
|
|
UINT32 status;
|
2014-04-07 05:09:49 +04:00
|
|
|
Context_Call call;
|
|
|
|
SCARDCONTEXT hContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_context_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardReleaseContext(hContext);
|
2014-04-07 05:09:49 +04:00
|
|
|
|
|
|
|
smartcard->hContext = 0;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
|
|
|
|
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_IsValidContext(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
UINT32 status;
|
2014-04-07 05:09:49 +04:00
|
|
|
Context_Call call;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_context_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardIsValidContext(hContext);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
static UINT32 smartcard_ListReadersA(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2013-09-12 14:18:35 +04:00
|
|
|
UINT32 status;
|
2014-04-07 05:09:49 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-04 21:57:28 +04:00
|
|
|
ListReaders_Call call;
|
2014-04-07 08:10:27 +04:00
|
|
|
ListReaders_Return ret;
|
|
|
|
LPSTR mszReaders = NULL;
|
|
|
|
DWORD cchReaders = 0;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
status = smartcard_unpack_list_readers_call(smartcard, irp->input, &call);
|
2014-04-04 08:56:24 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
if (status)
|
2013-09-30 15:29:40 +04:00
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
cchReaders = SCARD_AUTOALLOCATE;
|
|
|
|
status = SCardListReadersA(hContext, (LPCSTR) call.mszGroups, (LPSTR) &mszReaders, &cchReaders);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
if (status != SCARD_S_SUCCESS)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
ret.msz = (BYTE*) mszReaders;
|
|
|
|
ret.cBytes = smartcard_multi_string_length_a((char*) ret.msz) + 2;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
smartcard_pack_list_readers_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-11-22 05:46:58 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
finish:
|
|
|
|
if (mszReaders)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 08:10:27 +04:00
|
|
|
SCardFreeMemory(hContext, mszReaders);
|
|
|
|
}
|
2014-04-07 05:09:49 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
if (call.mszGroups)
|
|
|
|
free(call.mszGroups);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
static UINT32 smartcard_ListReadersW(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
|
|
|
{
|
|
|
|
UINT32 status;
|
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
ListReaders_Call call;
|
|
|
|
ListReaders_Return ret;
|
|
|
|
LPWSTR mszReaders = NULL;
|
|
|
|
DWORD cchReaders = 0;
|
2014-04-07 05:09:49 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
status = smartcard_unpack_list_readers_call(smartcard, irp->input, &call);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
if (status)
|
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
cchReaders = SCARD_AUTOALLOCATE;
|
|
|
|
status = SCardListReadersW(hContext, (LPCWSTR) call.mszGroups, (LPWSTR) &mszReaders, &cchReaders);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:10:27 +04:00
|
|
|
if (status != SCARD_S_SUCCESS)
|
|
|
|
goto finish;
|
|
|
|
|
|
|
|
ret.msz = (BYTE*) mszReaders;
|
|
|
|
ret.cBytes = (smartcard_multi_string_length_w((WCHAR*) ret.msz) + 2) * 2;
|
|
|
|
|
|
|
|
smartcard_pack_list_readers_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-09-12 15:53:15 +04:00
|
|
|
finish:
|
2014-04-07 08:10:27 +04:00
|
|
|
if (mszReaders)
|
2013-09-12 15:53:15 +04:00
|
|
|
{
|
2014-04-07 08:10:27 +04:00
|
|
|
SCardFreeMemory(hContext, mszReaders);
|
2013-09-12 15:53:15 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 21:57:28 +04:00
|
|
|
if (call.mszGroups)
|
|
|
|
free(call.mszGroups);
|
2013-09-30 15:29:40 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_GetStatusChangeA(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2014-04-07 00:29:11 +04:00
|
|
|
UINT32 index;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
GetStatusChangeA_Call call;
|
|
|
|
ReaderStateA* readerState = NULL;
|
|
|
|
LPSCARD_READERSTATEA rgReaderState = NULL;
|
|
|
|
LPSCARD_READERSTATEA rgReaderStates = NULL;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_get_status_change_a_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2014-04-07 00:29:11 +04:00
|
|
|
return status;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
|
|
|
|
|
|
|
rgReaderStates = (SCARD_READERSTATEA*) calloc(call.cReaders, sizeof(SCARD_READERSTATEA));
|
|
|
|
|
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2013-09-12 14:18:35 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
rgReaderState = &rgReaderStates[index];
|
|
|
|
rgReaderState->szReader = (LPCSTR) call.rgReaderStates[index].szReader;
|
|
|
|
rgReaderState->dwCurrentState = call.rgReaderStates[index].Common.dwCurrentState;
|
|
|
|
rgReaderState->dwEventState = call.rgReaderStates[index].Common.dwEventState;
|
|
|
|
rgReaderState->cbAtr = call.rgReaderStates[index].Common.cbAtr;
|
|
|
|
CopyMemory(&(rgReaderState->rgbAtr), &(call.rgReaderStates[index].Common.rgbAtr), 36);
|
2013-09-12 14:18:35 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = SCardGetStatusChangeA(hContext, (DWORD) call.dwTimeOut, rgReaderStates, (DWORD) call.cReaders);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
Stream_Write_UINT32(irp->output, call.cReaders); /* (4 bytes) */
|
|
|
|
Stream_Write_UINT32(irp->output, 0x00084dd8); /* (4 bytes) */
|
|
|
|
Stream_Write_UINT32(irp->output, call.cReaders); /* (4 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2013-09-12 15:53:15 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
rgReaderState = &rgReaderStates[index];
|
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->dwCurrentState);
|
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->dwEventState);
|
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->cbAtr);
|
|
|
|
Stream_Write(irp->output, rgReaderState->rgbAtr, 32);
|
|
|
|
Stream_Zero(irp->output, 4);
|
2013-09-12 15:53:15 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2012-11-20 08:49:08 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
if (call.rgReaderStates)
|
|
|
|
{
|
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
readerState = &call.rgReaderStates[index];
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
if (readerState->szReader)
|
|
|
|
free((void*) readerState->szReader);
|
|
|
|
readerState->szReader = NULL;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
2014-04-07 00:29:11 +04:00
|
|
|
free(call.rgReaderStates);
|
|
|
|
free(rgReaderStates);
|
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
return status;
|
|
|
|
}
|
2014-04-05 00:45:11 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_GetStatusChangeW(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2014-04-07 00:29:11 +04:00
|
|
|
{
|
|
|
|
LONG status;
|
|
|
|
UINT32 index;
|
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
GetStatusChangeW_Call call;
|
|
|
|
ReaderStateW* readerState = NULL;
|
|
|
|
LPSCARD_READERSTATEW rgReaderState = NULL;
|
|
|
|
LPSCARD_READERSTATEW rgReaderStates = NULL;
|
2013-09-12 15:53:15 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_get_status_change_w_call(smartcard, irp->input, &call);
|
2013-09-12 15:53:15 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
2014-04-05 19:22:48 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
rgReaderStates = (SCARD_READERSTATEW*) calloc(call.cReaders, sizeof(SCARD_READERSTATEW));
|
2014-04-05 19:22:48 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
rgReaderState = &rgReaderStates[index];
|
|
|
|
rgReaderState->szReader = (LPCWSTR) call.rgReaderStates[index].szReader;
|
|
|
|
rgReaderState->dwCurrentState = call.rgReaderStates[index].Common.dwCurrentState;
|
|
|
|
rgReaderState->dwEventState = call.rgReaderStates[index].Common.dwEventState;
|
|
|
|
rgReaderState->cbAtr = call.rgReaderStates[index].Common.cbAtr;
|
|
|
|
CopyMemory(&(rgReaderState->rgbAtr), &(call.rgReaderStates[index].Common.rgbAtr), 36);
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = SCardGetStatusChangeW(hContext, (DWORD) call.dwTimeOut, rgReaderStates, (DWORD) call.cReaders);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
Stream_Write_UINT32(irp->output, call.cReaders); /* (4 bytes) */
|
|
|
|
Stream_Write_UINT32(irp->output, 0x00084dd8); /* (4 bytes) */
|
|
|
|
Stream_Write_UINT32(irp->output, call.cReaders); /* (4 bytes) */
|
2014-04-05 19:22:48 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
rgReaderState = &rgReaderStates[index];
|
2014-04-05 19:22:48 +04:00
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->dwCurrentState);
|
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->dwEventState);
|
|
|
|
Stream_Write_UINT32(irp->output, rgReaderState->cbAtr);
|
|
|
|
Stream_Write(irp->output, rgReaderState->rgbAtr, 32);
|
2013-05-09 00:27:21 +04:00
|
|
|
Stream_Zero(irp->output, 4);
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
if (call.rgReaderStates)
|
2013-10-01 15:09:52 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
for (index = 0; index < call.cReaders; index++)
|
2013-10-01 15:09:52 +04:00
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
readerState = &call.rgReaderStates[index];
|
2013-10-01 15:09:52 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
if (readerState->szReader)
|
|
|
|
free((void*) readerState->szReader);
|
|
|
|
readerState->szReader = NULL;
|
2013-10-01 15:09:52 +04:00
|
|
|
}
|
2014-04-05 19:22:48 +04:00
|
|
|
free(call.rgReaderStates);
|
|
|
|
free(rgReaderStates);
|
2013-10-01 15:09:52 +04:00
|
|
|
}
|
2013-09-12 15:53:15 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_Cancel(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2014-04-07 05:09:49 +04:00
|
|
|
Context_Call call;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_context_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hContext = (ULONG_PTR) call.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardCancel(hContext);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-11-22 05:46:58 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
UINT32 smartcard_ConnectA(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2014-04-07 07:08:40 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
ConnectA_Call call;
|
|
|
|
Connect_Return ret;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
call.szReader = NULL;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
status = smartcard_unpack_connect_a_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2014-04-06 00:06:46 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hContext = (ULONG_PTR) call.Common.Context.pbContext;
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
if ((call.Common.dwPreferredProtocols == SCARD_PROTOCOL_UNDEFINED) &&
|
|
|
|
(call.Common.dwShareMode != SCARD_SHARE_DIRECT))
|
|
|
|
{
|
|
|
|
call.Common.dwPreferredProtocols = SCARD_PROTOCOL_T0 | SCARD_PROTOCOL_T1;
|
|
|
|
}
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
status = SCardConnectA(hContext, (char*) call.szReader, (DWORD) call.Common.dwShareMode,
|
|
|
|
(DWORD) call.Common.dwPreferredProtocols, &hCard, (DWORD*) &ret.dwActiveProtocol);
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
smartcard->hCard = hCard;
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.hCard.Context.cbContext = 0;
|
|
|
|
ret.hCard.Context.pbContext = 0;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
ret.hCard.cbHandle = sizeof(ULONG_PTR);
|
2014-04-07 00:29:11 +04:00
|
|
|
ret.hCard.pbHandle = (UINT64) hCard;
|
2014-04-06 04:46:20 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
smartcard_pack_connect_return(smartcard, irp->output, &ret);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
finish:
|
|
|
|
if (call.szReader)
|
|
|
|
free(call.szReader);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
UINT32 smartcard_ConnectW(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2014-04-06 00:06:46 +04:00
|
|
|
{
|
|
|
|
LONG status;
|
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
SCARDHANDLE hCard;
|
|
|
|
ConnectW_Call call;
|
|
|
|
Connect_Return ret;
|
|
|
|
|
|
|
|
call.szReader = NULL;
|
|
|
|
|
|
|
|
status = smartcard_unpack_connect_w_call(smartcard, irp->input, &call);
|
|
|
|
|
|
|
|
if (status)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hContext = (ULONG_PTR) call.Common.Context.pbContext;
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
if ((call.Common.dwPreferredProtocols == SCARD_PROTOCOL_UNDEFINED) &&
|
|
|
|
(call.Common.dwShareMode != SCARD_SHARE_DIRECT))
|
|
|
|
{
|
|
|
|
call.Common.dwPreferredProtocols = SCARD_PROTOCOL_T0 | SCARD_PROTOCOL_T1;
|
|
|
|
}
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
status = SCardConnectW(hContext, (WCHAR*) call.szReader, (DWORD) call.Common.dwShareMode,
|
2014-04-05 19:22:48 +04:00
|
|
|
(DWORD) call.Common.dwPreferredProtocols, &hCard, (DWORD*) &ret.dwActiveProtocol);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 01:29:12 +04:00
|
|
|
smartcard->hCard = hCard;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.hCard.Context.cbContext = 0;
|
|
|
|
ret.hCard.Context.pbContext = 0;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
ret.hCard.cbHandle = sizeof(ULONG_PTR);
|
2014-04-07 00:29:11 +04:00
|
|
|
ret.hCard.pbHandle = (UINT64) hCard;
|
2014-04-06 04:46:20 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
status = smartcard_pack_connect_return(smartcard, irp->output, &ret);
|
2014-04-06 04:46:20 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
if (status)
|
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-09-12 15:53:15 +04:00
|
|
|
finish:
|
2014-04-05 19:22:48 +04:00
|
|
|
if (call.szReader)
|
|
|
|
free(call.szReader);
|
2013-09-12 15:53:15 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_Reconnect(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2014-04-07 00:29:11 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
Reconnect_Call call;
|
|
|
|
Reconnect_Return ret;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_reconnect_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
status = SCardReconnect(hCard, (DWORD) call.dwShareMode, (DWORD) call.dwPreferredProtocols,
|
|
|
|
(DWORD) call.dwInitialization, (LPDWORD) &ret.dwActiveProtocol);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
smartcard_pack_reconnect_return(smartcard, irp->output, &ret);
|
2014-04-07 05:28:59 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_Disconnect(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2014-04-07 00:29:11 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
HCardAndDisposition_Call call;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
status = SCardDisconnect(hCard, (DWORD) call.dwDisposition);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
smartcard->hCard = 0;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_BeginTransaction(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDHANDLE hCard;
|
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
HCardAndDisposition_Call call;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardBeginTransaction(hCard);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_EndTransaction(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDHANDLE hCard;
|
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
HCardAndDisposition_Call call;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
status = SCardEndTransaction(hCard, call.dwDisposition);
|
2011-11-22 05:46:58 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-11-22 05:46:58 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_State(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
State_Call call;
|
|
|
|
State_Return ret;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_state_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2014-04-07 09:19:58 +04:00
|
|
|
return status;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
ret.cbAtrLen = SCARD_ATR_LENGTH;
|
|
|
|
|
2014-04-07 09:19:58 +04:00
|
|
|
status = SCardState(hCard, &ret.dwState, &ret.dwProtocol, (BYTE*) &ret.rgAtr, &ret.cbAtrLen);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
if (status != SCARD_S_SUCCESS)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 08:58:34 +04:00
|
|
|
Stream_Zero(irp->output, 256);
|
2014-04-07 09:19:58 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 09:19:58 +04:00
|
|
|
smartcard_pack_state_return(smartcard, irp->output, &ret);
|
2014-04-07 05:28:59 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
static DWORD smartcard_StatusA(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-07 05:09:49 +04:00
|
|
|
Status_Call call;
|
2014-04-07 08:58:34 +04:00
|
|
|
Status_Return ret;
|
|
|
|
DWORD cchReaderLen = 0;
|
|
|
|
LPSTR mszReaderNames = NULL;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_status_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
if (call.cbAtrLen > 32)
|
|
|
|
call.cbAtrLen = 32;
|
|
|
|
|
|
|
|
ret.cbAtrLen = call.cbAtrLen;
|
|
|
|
ZeroMemory(ret.pbAtr, 32);
|
2014-04-06 03:58:48 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
cchReaderLen = SCARD_AUTOALLOCATE;
|
|
|
|
|
|
|
|
status = SCardStatusA(hCard, (LPSTR) &mszReaderNames, &cchReaderLen,
|
|
|
|
&ret.dwState, &ret.dwProtocol, (BYTE*) &ret.pbAtr, &ret.cbAtrLen);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
if (status != SCARD_S_SUCCESS)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 08:58:34 +04:00
|
|
|
Stream_Zero(irp->output, 256);
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
ret.mszReaderNames = (BYTE*) mszReaderNames;
|
|
|
|
ret.cBytes = smartcard_multi_string_length_a((char*) ret.mszReaderNames) + 2;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
smartcard_pack_status_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
finish:
|
|
|
|
if (mszReaderNames)
|
|
|
|
{
|
|
|
|
SCardFreeMemory(hContext, mszReaderNames);
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD smartcard_StatusW(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
|
|
|
{
|
|
|
|
LONG status;
|
|
|
|
SCARDHANDLE hCard;
|
|
|
|
SCARDCONTEXT hContext;
|
|
|
|
Status_Call call;
|
|
|
|
Status_Return ret;
|
|
|
|
DWORD cchReaderLen = 0;
|
|
|
|
LPWSTR mszReaderNames = NULL;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 08:58:34 +04:00
|
|
|
status = smartcard_unpack_status_call(smartcard, irp->input, &call);
|
|
|
|
|
|
|
|
if (status)
|
|
|
|
goto finish;
|
|
|
|
|
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
|
|
|
|
|
|
|
if (call.cbAtrLen > 32)
|
|
|
|
call.cbAtrLen = 32;
|
|
|
|
|
|
|
|
ret.cbAtrLen = call.cbAtrLen;
|
|
|
|
ZeroMemory(ret.pbAtr, 32);
|
|
|
|
|
|
|
|
cchReaderLen = SCARD_AUTOALLOCATE;
|
|
|
|
|
|
|
|
status = SCardStatusW(hCard, (LPWSTR) &mszReaderNames, &cchReaderLen,
|
|
|
|
&ret.dwState, &ret.dwProtocol, (BYTE*) &ret.pbAtr, &ret.cbAtrLen);
|
|
|
|
|
|
|
|
if (status != SCARD_S_SUCCESS)
|
|
|
|
{
|
|
|
|
Stream_Zero(irp->output, 256);
|
|
|
|
goto finish;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.mszReaderNames = (BYTE*) mszReaderNames;
|
|
|
|
ret.cBytes = (smartcard_multi_string_length_w((WCHAR*) ret.mszReaderNames) + 2) * 2;
|
|
|
|
|
|
|
|
smartcard_pack_status_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-09-12 15:53:15 +04:00
|
|
|
finish:
|
2014-04-07 08:58:34 +04:00
|
|
|
if (mszReaderNames)
|
2013-09-12 15:53:15 +04:00
|
|
|
{
|
2014-04-07 08:58:34 +04:00
|
|
|
SCardFreeMemory(hContext, mszReaderNames);
|
2013-09-12 15:53:15 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_Transmit(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDHANDLE hCard;
|
|
|
|
SCARDCONTEXT hContext;
|
2014-04-07 05:09:49 +04:00
|
|
|
Transmit_Call call;
|
2014-04-07 07:08:40 +04:00
|
|
|
Transmit_Return ret;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_transmit_call(smartcard, irp->input, &call);
|
2014-04-06 00:06:46 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2013-09-12 14:30:46 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2013-09-12 17:50:12 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.cbRecvLength = 0;
|
|
|
|
ret.pbRecvBuffer = NULL;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
if (call.cbRecvLength && !call.fpbRecvBufferIsNULL)
|
2014-04-04 01:29:12 +04:00
|
|
|
{
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.cbRecvLength = call.cbRecvLength;
|
|
|
|
ret.pbRecvBuffer = (BYTE*) malloc(ret.cbRecvLength);
|
2014-04-04 01:29:12 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
ret.pioRecvPci = call.pioRecvPci;
|
2014-04-04 01:29:12 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
status = SCardTransmit(hCard, call.pioSendPci, call.pbSendBuffer, call.cbSendLength,
|
|
|
|
ret.pioRecvPci, ret.pbRecvBuffer, &(ret.cbRecvLength));
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
status = smartcard_pack_transmit_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-09-12 15:53:15 +04:00
|
|
|
finish:
|
2014-04-07 05:09:49 +04:00
|
|
|
if (call.pbSendBuffer)
|
|
|
|
free(call.pbSendBuffer);
|
2014-04-07 07:08:40 +04:00
|
|
|
if (ret.pbRecvBuffer)
|
|
|
|
free(ret.pbRecvBuffer);
|
2014-04-07 05:09:49 +04:00
|
|
|
if (call.pioSendPci)
|
|
|
|
free(call.pioSendPci);
|
|
|
|
if (call.pioRecvPci)
|
|
|
|
free(call.pioRecvPci);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_Control(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2014-04-07 05:09:49 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
Control_Call call;
|
|
|
|
Control_Return ret;
|
2014-04-07 05:09:49 +04:00
|
|
|
UINT32 controlFunction;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_control_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
2013-09-12 15:53:15 +04:00
|
|
|
goto finish;
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
if (DEVICE_TYPE_FROM_CTL_CODE(call.dwControlCode) == FILE_DEVICE_SMARTCARD)
|
2011-11-23 04:22:17 +04:00
|
|
|
{
|
2014-04-05 19:22:48 +04:00
|
|
|
controlFunction = FUNCTION_FROM_CTL_CODE(call.dwControlCode);
|
|
|
|
call.dwControlCode = SCARD_CTL_CODE(controlFunction);
|
2011-11-23 04:22:17 +04:00
|
|
|
}
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
ret.cbOutBufferSize = call.cbOutBufferSize;
|
2014-04-06 03:58:48 +04:00
|
|
|
ret.pvOutBuffer = (BYTE*) malloc(call.cbOutBufferSize);
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
status = SCardControl(hCard, (DWORD) call.dwControlCode,
|
|
|
|
call.pvInBuffer, (DWORD) call.cbInBufferSize,
|
|
|
|
ret.pvOutBuffer, (DWORD) call.cbOutBufferSize, &ret.cbOutBufferSize);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 09:19:58 +04:00
|
|
|
smartcard_pack_control_return(smartcard, irp->output, &ret);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-09-12 15:53:15 +04:00
|
|
|
finish:
|
2014-04-05 19:22:48 +04:00
|
|
|
if (call.pvInBuffer)
|
|
|
|
free(call.pvInBuffer);
|
|
|
|
if (ret.pvOutBuffer)
|
|
|
|
free(ret.pvOutBuffer);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_GetAttrib(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
LONG status;
|
2014-04-05 19:22:48 +04:00
|
|
|
DWORD cbAttrLen;
|
2011-10-15 19:30:10 +04:00
|
|
|
SCARDHANDLE hCard;
|
2013-09-12 14:18:35 +04:00
|
|
|
SCARDCONTEXT hContext;
|
2014-04-05 19:22:48 +04:00
|
|
|
GetAttrib_Call call;
|
|
|
|
GetAttrib_Return ret;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
status = smartcard_unpack_get_attrib_call(smartcard, irp->input, &call);
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
2014-04-07 05:09:49 +04:00
|
|
|
hCard = (ULONG_PTR) call.hCard.pbHandle;
|
|
|
|
hContext = (ULONG_PTR) call.hCard.Context.pbContext;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
ret.pbAttr = NULL;
|
2014-04-06 05:17:54 +04:00
|
|
|
cbAttrLen = (!call.cbAttrLen || call.fpbAttrIsNULL) ? 0 : SCARD_AUTOALLOCATE;
|
2014-04-05 19:22:48 +04:00
|
|
|
|
|
|
|
status = SCardGetAttrib(hCard, call.dwAttrId, (cbAttrLen == 0) ? NULL : (BYTE*) &ret.pbAttr, &cbAttrLen);
|
2012-11-21 20:56:40 +04:00
|
|
|
|
|
|
|
if (status != SCARD_S_SUCCESS)
|
|
|
|
{
|
2014-04-05 19:22:48 +04:00
|
|
|
cbAttrLen = (call.cbAttrLen == 0) ? 0 : SCARD_AUTOALLOCATE;
|
2011-12-22 23:23:08 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
if (call.dwAttrId == SCARD_ATTR_DEVICE_FRIENDLY_NAME_A && status == SCARD_E_UNSUPPORTED_FEATURE)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardGetAttrib(hCard, SCARD_ATTR_DEVICE_FRIENDLY_NAME_W,
|
2014-04-05 19:22:48 +04:00
|
|
|
(cbAttrLen == 0) ? NULL : (BYTE*) &ret.pbAttr, &cbAttrLen);
|
2012-11-21 20:56:40 +04:00
|
|
|
|
|
|
|
if (status != SCARD_S_SUCCESS)
|
|
|
|
{
|
2014-04-05 19:22:48 +04:00
|
|
|
cbAttrLen = (call.cbAttrLen == 0) ? 0 : SCARD_AUTOALLOCATE;
|
2011-12-22 23:23:08 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
2014-04-04 01:29:12 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
if (call.dwAttrId == SCARD_ATTR_DEVICE_FRIENDLY_NAME_W && status == SCARD_E_UNSUPPORTED_FEATURE)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCardGetAttrib(hCard, SCARD_ATTR_DEVICE_FRIENDLY_NAME_A,
|
2014-04-05 19:22:48 +04:00
|
|
|
(cbAttrLen == 0) ? NULL : (BYTE*) &ret.pbAttr, &cbAttrLen);
|
2012-11-21 20:56:40 +04:00
|
|
|
|
|
|
|
if (status != SCARD_S_SUCCESS)
|
|
|
|
{
|
2014-04-05 19:22:48 +04:00
|
|
|
cbAttrLen = (call.cbAttrLen == 0) ? 0 : SCARD_AUTOALLOCATE;
|
2011-12-22 23:23:08 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
2014-04-05 19:22:48 +04:00
|
|
|
|
|
|
|
if ((cbAttrLen > call.cbAttrLen) && (ret.pbAttr != NULL))
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-11-21 20:56:40 +04:00
|
|
|
status = SCARD_E_INSUFFICIENT_BUFFER;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
2014-04-07 09:19:58 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
call.cbAttrLen = cbAttrLen;
|
2014-04-07 09:19:58 +04:00
|
|
|
ret.cbAttrLen = call.cbAttrLen;
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
if (status != SCARD_S_SUCCESS)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-07 08:58:34 +04:00
|
|
|
Stream_Zero(irp->output, 256);
|
2013-10-01 15:09:52 +04:00
|
|
|
goto finish;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 09:19:58 +04:00
|
|
|
smartcard_pack_get_attrib_return(smartcard, irp->output, &ret);
|
2014-04-07 05:28:59 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2013-10-01 15:09:52 +04:00
|
|
|
finish:
|
2014-04-05 19:22:48 +04:00
|
|
|
SCardFreeMemory(hContext, ret.pbAttr);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
return status;
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-07 07:08:40 +04:00
|
|
|
static UINT32 smartcard_AccessStartedEvent(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2013-09-12 14:18:35 +04:00
|
|
|
if (Stream_GetRemainingLength(irp->input) < 4)
|
|
|
|
{
|
2014-04-07 05:09:49 +04:00
|
|
|
WLog_Print(smartcard->log, WLOG_WARN, "AccessStartedEvent is too short: %d",
|
|
|
|
(int) Stream_GetRemainingLength(irp->input));
|
2013-09-12 14:18:35 +04:00
|
|
|
return SCARD_F_INTERNAL_ERROR;
|
|
|
|
}
|
2013-10-24 23:34:14 +04:00
|
|
|
|
2014-04-05 19:22:48 +04:00
|
|
|
Stream_Seek(irp->input, 4); /* Unused (4 bytes) */
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2012-11-21 20:56:40 +04:00
|
|
|
smartcard_output_alignment(irp, 8);
|
2013-09-12 14:18:35 +04:00
|
|
|
|
2011-10-15 19:30:10 +04:00
|
|
|
return SCARD_S_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2014-04-06 01:51:13 +04:00
|
|
|
void smartcard_irp_device_control_peek_io_control_code(SMARTCARD_DEVICE* smartcard, IRP* irp, UINT32* ioControlCode)
|
|
|
|
{
|
|
|
|
*ioControlCode = 0;
|
|
|
|
|
|
|
|
if (Stream_GetRemainingLength(irp->input) < 32)
|
|
|
|
{
|
|
|
|
WLog_Print(smartcard->log, WLOG_WARN, "Device Control Request is too short: %d",
|
|
|
|
(int) Stream_GetRemainingLength(irp->input));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream_Seek_UINT32(irp->input); /* OutputBufferLength (4 bytes) */
|
|
|
|
Stream_Seek_UINT32(irp->input); /* InputBufferLength (4 bytes) */
|
|
|
|
Stream_Read_UINT32(irp->input, *ioControlCode); /* IoControlCode (4 bytes) */
|
|
|
|
Stream_Rewind(irp->input, (4 + 4 + 4));
|
|
|
|
}
|
|
|
|
|
2014-04-06 01:15:17 +04:00
|
|
|
void smartcard_irp_device_control(SMARTCARD_DEVICE* smartcard, IRP* irp)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2012-10-09 11:26:39 +04:00
|
|
|
UINT32 result;
|
2014-04-05 19:56:02 +04:00
|
|
|
UINT32 status;
|
2014-04-06 00:06:46 +04:00
|
|
|
UINT32 ioControlCode;
|
2014-04-04 08:56:24 +04:00
|
|
|
UINT32 outputBufferLength;
|
|
|
|
UINT32 inputBufferLength;
|
2014-04-06 00:06:46 +04:00
|
|
|
UINT32 objectBufferLength;
|
|
|
|
|
|
|
|
/* Device Control Request */
|
2012-11-21 20:56:40 +04:00
|
|
|
|
2013-09-12 14:18:35 +04:00
|
|
|
if (Stream_GetRemainingLength(irp->input) < 32)
|
|
|
|
{
|
2014-04-06 00:06:46 +04:00
|
|
|
WLog_Print(smartcard->log, WLOG_WARN, "Device Control Request is too short: %d",
|
|
|
|
(int) Stream_GetRemainingLength(irp->input));
|
2013-09-12 14:18:35 +04:00
|
|
|
return;
|
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 08:56:24 +04:00
|
|
|
Stream_Read_UINT32(irp->input, outputBufferLength); /* OutputBufferLength (4 bytes) */
|
|
|
|
Stream_Read_UINT32(irp->input, inputBufferLength); /* InputBufferLength (4 bytes) */
|
|
|
|
Stream_Read_UINT32(irp->input, ioControlCode); /* IoControlCode (4 bytes) */
|
|
|
|
Stream_Seek(irp->input, 20); /* Padding (20 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
if (Stream_Length(irp->input) != (Stream_GetPosition(irp->input) + inputBufferLength))
|
2014-04-05 19:56:02 +04:00
|
|
|
{
|
2014-04-06 00:06:46 +04:00
|
|
|
WLog_Print(smartcard->log, WLOG_WARN,
|
|
|
|
"InputBufferLength mismatch: Actual: %d Expected: %d\n",
|
|
|
|
Stream_Length(irp->input), Stream_GetPosition(irp->input) + inputBufferLength);
|
2014-04-05 19:56:02 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-04-07 00:29:11 +04:00
|
|
|
WLog_Print(smartcard->log, WLOG_DEBUG, "%s (0x%08X) FileId: %d CompletionId: %d",
|
|
|
|
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode, irp->FileId, irp->CompletionId);
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-06 03:58:48 +04:00
|
|
|
#if 1
|
2014-04-07 05:09:49 +04:00
|
|
|
if (/*(ioControlCode != SCARD_IOCTL_TRANSMIT) &&*/
|
2014-04-06 03:58:48 +04:00
|
|
|
(ioControlCode != SCARD_IOCTL_GETSTATUSCHANGEA) &&
|
|
|
|
(ioControlCode != SCARD_IOCTL_GETSTATUSCHANGEW))
|
|
|
|
{
|
2014-04-07 00:29:11 +04:00
|
|
|
printf("%s (0x%08X) FileId: %d CompletionId: %d\n",
|
|
|
|
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode, irp->FileId, irp->CompletionId);
|
2014-04-06 03:58:48 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
if ((ioControlCode != SCARD_IOCTL_ACCESSSTARTEDEVENT) &&
|
|
|
|
(ioControlCode != SCARD_IOCTL_RELEASESTARTEDEVENT))
|
2014-04-05 19:56:02 +04:00
|
|
|
{
|
2014-04-06 00:06:46 +04:00
|
|
|
status = smartcard_unpack_common_type_header(smartcard, irp->input);
|
2014-04-05 19:56:02 +04:00
|
|
|
|
|
|
|
if (status)
|
|
|
|
return;
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
status = smartcard_unpack_private_type_header(smartcard, irp->input);
|
2014-04-05 19:56:02 +04:00
|
|
|
|
|
|
|
if (status)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
/**
|
|
|
|
* [MS-RDPESC] 3.2.5.1: Sending Outgoing Messages:
|
|
|
|
* the output buffer length SHOULD be set to 2048
|
|
|
|
*
|
|
|
|
* Since it's a SHOULD and not a MUST, we don't care
|
|
|
|
* about it, but we still reserve at least 2048 bytes.
|
|
|
|
*/
|
2013-05-02 02:15:55 +04:00
|
|
|
Stream_EnsureRemainingCapacity(irp->output, 2048);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
/* Device Control Response */
|
|
|
|
Stream_Seek_UINT32(irp->output); /* OutputBufferLength (4 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
Stream_Seek(irp->output, SMARTCARD_COMMON_TYPE_HEADER_LENGTH); /* CommonTypeHeader (8 bytes) */
|
|
|
|
Stream_Seek(irp->output, SMARTCARD_PRIVATE_TYPE_HEADER_LENGTH); /* PrivateTypeHeader (8 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
Stream_Seek_UINT32(irp->output); /* Result (4 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
switch (ioControlCode)
|
2011-10-15 19:30:10 +04:00
|
|
|
{
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_ESTABLISHCONTEXT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_EstablishContext(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_RELEASECONTEXT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_ReleaseContext(smartcard, irp);
|
2014-04-05 19:56:02 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_ISVALIDCONTEXT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_IsValidContext(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERGROUPSA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_LISTREADERGROUPSW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERSA:
|
2014-04-07 08:10:27 +04:00
|
|
|
result = smartcard_ListReadersA(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_LISTREADERSW:
|
2014-04-07 08:10:27 +04:00
|
|
|
result = smartcard_ListReadersW(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERGROUPA:
|
2014-04-04 21:57:28 +04:00
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_INTRODUCEREADERGROUPW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_FORGETREADERGROUPA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_FORGETREADERGROUPW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_INTRODUCEREADERA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_INTRODUCEREADERW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_FORGETREADERA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_FORGETREADERW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_ADDREADERTOGROUPA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_ADDREADERTOGROUPW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_REMOVEREADERFROMGROUPA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_REMOVEREADERFROMGROUPW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_LOCATECARDSA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_LOCATECARDSW:
|
2011-10-15 19:30:10 +04:00
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_GETSTATUSCHANGEA:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_GetStatusChangeA(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_GETSTATUSCHANGEW:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_GetStatusChangeW(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_CANCEL:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_Cancel(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_CONNECTA:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_ConnectA(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_CONNECTW:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_ConnectW(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_RECONNECT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_Reconnect(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_DISCONNECT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_Disconnect(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_BEGINTRANSACTION:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_BeginTransaction(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_ENDTRANSACTION:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_EndTransaction(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_STATE:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_State(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_STATUSA:
|
2014-04-07 08:58:34 +04:00
|
|
|
result = smartcard_StatusA(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_STATUSW:
|
2014-04-07 08:58:34 +04:00
|
|
|
result = smartcard_StatusW(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_TRANSMIT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_Transmit(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_CONTROL:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_Control(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_GETATTRIB:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_GetAttrib(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_SETATTRIB:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_ACCESSSTARTEDEVENT:
|
2014-04-07 07:08:40 +04:00
|
|
|
result = smartcard_AccessStartedEvent(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSBYATRA:
|
2014-04-07 05:28:59 +04:00
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
2014-04-05 19:56:02 +04:00
|
|
|
|
2014-04-04 02:47:18 +04:00
|
|
|
case SCARD_IOCTL_LOCATECARDSBYATRW:
|
2014-04-07 05:28:59 +04:00
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
|
2014-04-05 19:56:02 +04:00
|
|
|
case SCARD_IOCTL_READCACHEA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_READCACHEW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_WRITECACHEA:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_WRITECACHEW:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_GETTRANSMITCOUNT:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_RELEASESTARTEDEVENT:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_GETREADERICON:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SCARD_IOCTL_GETDEVICETYPEID:
|
|
|
|
result = SCARD_F_INTERNAL_ERROR;
|
|
|
|
break;
|
|
|
|
|
2011-10-15 19:30:10 +04:00
|
|
|
default:
|
2014-04-04 08:56:24 +04:00
|
|
|
result = STATUS_UNSUCCESSFUL;
|
2011-10-15 19:30:10 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
if ((result != SCARD_S_SUCCESS) && (result != SCARD_E_TIMEOUT))
|
2014-04-05 19:56:02 +04:00
|
|
|
{
|
2014-04-06 00:06:46 +04:00
|
|
|
WLog_Print(smartcard->log, WLOG_WARN,
|
2014-04-07 00:29:11 +04:00
|
|
|
"IRP failure: %s (0x%08X), status: %s (0x%08X)",
|
|
|
|
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode,
|
2014-04-06 00:06:46 +04:00
|
|
|
SCardGetErrorString(result), result);
|
2014-04-05 19:56:02 +04:00
|
|
|
}
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
Stream_SealLength(irp->output);
|
|
|
|
outputBufferLength = Stream_Length(irp->output) - RDPDR_DEVICE_IO_RESPONSE_LENGTH - 4;
|
|
|
|
objectBufferLength = outputBufferLength - RDPDR_DEVICE_IO_RESPONSE_LENGTH;
|
|
|
|
Stream_SetPosition(irp->output, RDPDR_DEVICE_IO_RESPONSE_LENGTH);
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
/* Device Control Response */
|
|
|
|
Stream_Write_UINT32(irp->output, outputBufferLength); /* OutputBufferLength (4 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
smartcard_pack_common_type_header(smartcard, irp->output); /* CommonTypeHeader (8 bytes) */
|
|
|
|
smartcard_pack_private_type_header(smartcard, irp->output, objectBufferLength); /* PrivateTypeHeader (8 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
Stream_Write_UINT32(irp->output, result); /* Result (4 bytes) */
|
2011-10-15 19:30:10 +04:00
|
|
|
|
2014-04-06 00:06:46 +04:00
|
|
|
Stream_SetPosition(irp->output, Stream_Length(irp->output));
|
2011-10-15 19:30:10 +04:00
|
|
|
|
|
|
|
irp->IoStatus = 0;
|
2014-04-06 00:57:31 +04:00
|
|
|
smartcard_complete_irp(smartcard, irp);
|
2011-10-15 19:30:10 +04:00
|
|
|
}
|