FreeRDP/channels/smartcard/client/smartcard_operations.c

1712 lines
47 KiB
C
Raw Normal View History

/**
2012-10-09 05:00:07 +04:00
* FreeRDP: A Remote Desktop Protocol Implementation
* 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"
#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
#include <winpr/crt.h>
#include <winpr/print.h>
#include <winpr/stream.h>
#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
#include "smartcard_main.h"
2011-10-15 19:30:10 +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:
return funcName ? "SCardEstablishContext" : "SCARD_IOCTL_ESTABLISHCONTEXT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_RELEASECONTEXT:
return funcName ? "SCardReleaseContext" : "SCARD_IOCTL_RELEASECONTEXT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_ISVALIDCONTEXT:
return funcName ? "SCardIsValidContext" : "SCARD_IOCTL_ISVALIDCONTEXT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LISTREADERGROUPSA:
return funcName ? "SCardListReaderGroupsA" : "SCARD_IOCTL_LISTREADERGROUPSA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LISTREADERGROUPSW:
return funcName ? "SCardListReaderGroupsW" : "SCARD_IOCTL_LISTREADERGROUPSW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LISTREADERSA:
return funcName ? "SCardListReadersA" : "SCARD_IOCTL_LISTREADERSA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LISTREADERSW:
return funcName ? "SCardListReadersW" : "SCARD_IOCTL_LISTREADERSW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_INTRODUCEREADERGROUPA:
return funcName ? "SCardIntroduceReaderGroupA" : "SCARD_IOCTL_INTRODUCEREADERGROUPA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_INTRODUCEREADERGROUPW:
return funcName ? "SCardIntroduceReaderGroupW" : "SCARD_IOCTL_INTRODUCEREADERGROUPW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_FORGETREADERGROUPA:
return funcName ? "SCardForgetReaderGroupA" : "SCARD_IOCTL_FORGETREADERGROUPA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_FORGETREADERGROUPW:
return funcName ? "SCardForgetReaderGroupW" : "SCARD_IOCTL_FORGETREADERGROUPW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_INTRODUCEREADERA:
return funcName ? "SCardIntroduceReaderA" : "SCARD_IOCTL_INTRODUCEREADERA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_INTRODUCEREADERW:
return funcName ? "SCardIntroduceReaderW" : "SCARD_IOCTL_INTRODUCEREADERW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_FORGETREADERA:
return funcName ? "SCardForgetReaderA" : "SCARD_IOCTL_FORGETREADERA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_FORGETREADERW:
return funcName ? "SCardForgetReaderW" : "SCARD_IOCTL_FORGETREADERW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_ADDREADERTOGROUPA:
return funcName ? "SCardAddReaderToGroupA" : "SCARD_IOCTL_ADDREADERTOGROUPA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_ADDREADERTOGROUPW:
return funcName ? "SCardAddReaderToGroupW" : "SCARD_IOCTL_ADDREADERTOGROUPW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_REMOVEREADERFROMGROUPA:
return funcName ? "SCardRemoveReaderFromGroupA" : "SCARD_IOCTL_REMOVEREADERFROMGROUPA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_REMOVEREADERFROMGROUPW:
return funcName ? "SCardRemoveReaderFromGroupW" : "SCARD_IOCTL_REMOVEREADERFROMGROUPW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LOCATECARDSA:
return funcName ? "SCardLocateCardsA" : "SCARD_IOCTL_LOCATECARDSA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LOCATECARDSW:
return funcName ? "SCardLocateCardsW" : "SCARD_IOCTL_LOCATECARDSW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETSTATUSCHANGEA:
return funcName ? "SCardGetStatusChangeA" : "SCARD_IOCTL_GETSTATUSCHANGEA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETSTATUSCHANGEW:
return funcName ? "SCardGetStatusChangeW" : "SCARD_IOCTL_GETSTATUSCHANGEW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_CANCEL:
return funcName ? "SCardCancel" : "SCARD_IOCTL_CANCEL";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_CONNECTA:
return funcName ? "SCardConnectA" : "SCARD_IOCTL_CONNECTA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_CONNECTW:
return funcName ? "SCardConnectW" : "SCARD_IOCTL_CONNECTW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_RECONNECT:
return funcName ? "SCardReconnect" : "SCARD_IOCTL_RECONNECT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_DISCONNECT:
return funcName ? "SCardDisconnect" : "SCARD_IOCTL_DISCONNECT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_BEGINTRANSACTION:
return funcName ? "SCardBeginTransaction" : "SCARD_IOCTL_BEGINTRANSACTION";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_ENDTRANSACTION:
return funcName ? "SCardEndTransaction" : "SCARD_IOCTL_ENDTRANSACTION";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_STATE:
return funcName ? "SCardState" : "SCARD_IOCTL_STATE";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_STATUSA:
return funcName ? "SCardStatusA" : "SCARD_IOCTL_STATUSA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_STATUSW:
return funcName ? "SCardStatusW" : "SCARD_IOCTL_STATUSW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_TRANSMIT:
return funcName ? "SCardTransmit" : "SCARD_IOCTL_TRANSMIT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_CONTROL:
return funcName ? "SCardControl" : "SCARD_IOCTL_CONTROL";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETATTRIB:
return funcName ? "SCardGetAttrib" : "SCARD_IOCTL_GETATTRIB";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_SETATTRIB:
return funcName ? "SCardSetAttrib" : "SCARD_IOCTL_SETATTRIB";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_ACCESSSTARTEDEVENT:
return funcName ? "SCardAccessStartedEvent" : "SCARD_IOCTL_ACCESSSTARTEDEVENT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LOCATECARDSBYATRA:
return funcName ? "SCardLocateCardsByATRA" : "SCARD_IOCTL_LOCATECARDSBYATRA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_LOCATECARDSBYATRW:
return funcName ? "SCardLocateCardsByATRB" : "SCARD_IOCTL_LOCATECARDSBYATRW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_READCACHEA:
return funcName ? "SCardReadCacheA" : "SCARD_IOCTL_READCACHEA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_READCACHEW:
return funcName ? "SCardReadCacheW" : "SCARD_IOCTL_READCACHEW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_WRITECACHEA:
return funcName ? "SCardWriteCacheA" : "SCARD_IOCTL_WRITECACHEA";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_WRITECACHEW:
return funcName ? "SCardWriteCacheW" : "SCARD_IOCTL_WRITECACHEW";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETTRANSMITCOUNT:
return funcName ? "SCardGetTransmitCount" : "SCARD_IOCTL_GETTRANSMITCOUNT";
case SCARD_IOCTL_RELEASESTARTEDEVENT:
return funcName ? "SCardReleaseStartedEvent" : "SCARD_IOCTL_RELEASESTARTEDEVENT";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETREADERICON:
return funcName ? "SCardGetReaderIcon" : "SCARD_IOCTL_GETREADERICON";
2014-04-04 08:56:24 +04:00
case SCARD_IOCTL_GETDEVICETYPEID:
return funcName ? "SCardGetDeviceTypeId" : "SCARD_IOCTL_GETDEVICETYPEID";
2014-04-04 08:56:24 +04:00
default:
return funcName ? "SCardUnknown" : "SCARD_IOCTL_UNKNOWN";
2014-04-04 08:56:24 +04:00
}
return funcName ? "SCardUnknown" : "SCARD_IOCTL_UNKNOWN";
2014-04-04 08:56:24 +04:00
}
static UINT32 smartcard_EstablishContext_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, EstablishContext_Call* call)
2011-10-15 19:30:10 +04:00
{
UINT32 status;
2011-10-15 19:30:10 +04:00
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_establish_context_call(smartcard, irp->input, call);
2011-10-15 19:30:10 +04:00
smartcard_trace_establish_context_call(smartcard, call);
return status;
}
static UINT32 smartcard_EstablishContext_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, EstablishContext_Call* call)
{
UINT32 status;
SCARDCONTEXT hContext = -1;
EstablishContext_Return ret;
2011-10-15 19:30:10 +04:00
status = ret.ReturnCode = SCardEstablishContext(call->dwScope, NULL, NULL, &hContext);
2011-10-15 19:30:10 +04:00
if (ret.ReturnCode == SCARD_S_SUCCESS)
{
SMARTCARD_CONTEXT* pContext;
void* key = (void*) (size_t) hContext;
pContext = smartcard_context_new(smartcard, hContext);
ListDictionary_Add(smartcard->rgSCardContextList, key, (void*) pContext);
}
smartcard_scard_context_native_to_redir(smartcard, &(ret.hContext), hContext);
smartcard_trace_establish_context_return(smartcard, &ret);
2011-10-15 19:30:10 +04:00
status = smartcard_pack_establish_context_return(smartcard, irp->output, &ret);
2011-10-15 19:30:10 +04:00
if (status)
return status;
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_ReleaseContext_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
UINT32 status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_context_call(smartcard, irp->input, call);
smartcard_trace_context_call(smartcard, call, "ReleaseContext");
return status;
}
static UINT32 smartcard_ReleaseContext_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
UINT32 status;
Long_Return ret;
SCARDCONTEXT hContext;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2011-10-15 19:30:10 +04:00
status = ret.ReturnCode = SCardReleaseContext(hContext);
2014-04-07 05:09:49 +04:00
if (ret.ReturnCode == SCARD_S_SUCCESS)
{
SMARTCARD_CONTEXT* pContext;
void* key = (void*) (size_t) hContext;
pContext = (SMARTCARD_CONTEXT*) ListDictionary_Remove(smartcard->rgSCardContextList, key);
smartcard_context_free(pContext);
}
smartcard_trace_long_return(smartcard, &ret, "ReleaseContext");
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_IsValidContext_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
UINT32 status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_context_call(smartcard, irp->input, call);
smartcard_trace_context_call(smartcard, call, "IsValidContext");
return status;
}
static UINT32 smartcard_IsValidContext_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
UINT32 status;
Long_Return ret;
SCARDCONTEXT hContext;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2011-10-15 19:30:10 +04:00
status = ret.ReturnCode = SCardIsValidContext(hContext);
2011-10-15 19:30:10 +04:00
smartcard_trace_long_return(smartcard, &ret, "IsValidContext");
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_ListReadersA_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, ListReaders_Call* call)
{
UINT32 status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_list_readers_call(smartcard, irp->input, call);
smartcard_trace_list_readers_call(smartcard, call, FALSE);
return status;
}
static UINT32 smartcard_ListReadersA_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, ListReaders_Call* call)
2011-10-15 19:30:10 +04:00
{
UINT32 status;
2014-04-07 05:09:49 +04:00
SCARDCONTEXT hContext;
ListReaders_Return ret;
LPSTR mszReaders = NULL;
DWORD cchReaders = 0;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2011-10-15 19:30:10 +04:00
cchReaders = SCARD_AUTOALLOCATE;
status = ret.ReturnCode = SCardListReadersA(hContext, (LPCSTR) call->mszGroups, (LPSTR) &mszReaders, &cchReaders);
2011-10-15 19:30:10 +04:00
ret.msz = (BYTE*) mszReaders;
2014-04-09 22:27:44 +04:00
ret.cBytes = cchReaders;
2011-10-15 19:30:10 +04:00
2014-04-11 01:36:42 +04:00
if (status)
return status;
smartcard_trace_list_readers_return(smartcard, &ret, FALSE);
status = smartcard_pack_list_readers_return(smartcard, irp->output, &ret);
if (status)
return status;
2011-10-15 19:30:10 +04:00
if (mszReaders)
SCardFreeMemory(hContext, mszReaders);
2014-04-07 05:09:49 +04:00
if (call->mszGroups)
free(call->mszGroups);
return ret.ReturnCode;
}
2011-10-15 19:30:10 +04:00
static UINT32 smartcard_ListReadersW_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, ListReaders_Call* call)
{
UINT32 status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_list_readers_call(smartcard, irp->input, call);
smartcard_trace_list_readers_call(smartcard, call, TRUE);
return status;
}
static UINT32 smartcard_ListReadersW_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, ListReaders_Call* call)
{
UINT32 status;
SCARDCONTEXT hContext;
ListReaders_Return ret;
LPWSTR mszReaders = NULL;
DWORD cchReaders = 0;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2011-10-15 19:30:10 +04:00
cchReaders = SCARD_AUTOALLOCATE;
status = ret.ReturnCode = SCardListReadersW(hContext, (LPCWSTR) call->mszGroups, (LPWSTR) &mszReaders, &cchReaders);
2011-10-15 19:30:10 +04:00
2014-04-11 01:36:42 +04:00
ret.msz = (BYTE*) mszReaders;
ret.cBytes = cchReaders * 2;
if (status != SCARD_S_SUCCESS)
return status;
smartcard_trace_list_readers_return(smartcard, &ret, TRUE);
status = smartcard_pack_list_readers_return(smartcard, irp->output, &ret);
if (status)
return status;
2011-10-15 19:30:10 +04:00
if (mszReaders)
SCardFreeMemory(hContext, mszReaders);
2011-10-15 19:30:10 +04:00
if (call->mszGroups)
free(call->mszGroups);
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_GetStatusChangeA_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, GetStatusChangeA_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_get_status_change_a_call(smartcard, irp->input, call);
smartcard_trace_get_status_change_a_call(smartcard, call);
return status;
}
static UINT32 smartcard_GetStatusChangeA_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, GetStatusChangeA_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
UINT32 index;
2011-10-15 19:30:10 +04:00
SCARDCONTEXT hContext;
GetStatusChange_Return ret;
LPSCARD_READERSTATEA rgReaderState = NULL;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2013-10-24 23:34:14 +04:00
status = ret.ReturnCode = SCardGetStatusChangeA(hContext, call->dwTimeOut, call->rgReaderStates, call->cReaders);
2013-10-24 23:34:14 +04:00
if (status && (status != SCARD_E_TIMEOUT) && (status != SCARD_E_CANCELLED))
return status;
ret.cReaders = call->cReaders;
ret.rgReaderStates = (ReaderState_Return*) calloc(ret.cReaders, sizeof(ReaderState_Return));
2011-10-15 19:30:10 +04:00
for (index = 0; index < ret.cReaders; index++)
{
ret.rgReaderStates[index].dwCurrentState = call->rgReaderStates[index].dwCurrentState;
ret.rgReaderStates[index].dwEventState = call->rgReaderStates[index].dwEventState;
ret.rgReaderStates[index].cbAtr = call->rgReaderStates[index].cbAtr;
CopyMemory(&(ret.rgReaderStates[index].rgbAtr), &(call->rgReaderStates[index].rgbAtr), 32);
}
2011-10-15 19:30:10 +04:00
smartcard_trace_get_status_change_return(smartcard, &ret, FALSE);
status = smartcard_pack_get_status_change_return(smartcard, irp->output, &ret);
if (status)
return status;
if (call->rgReaderStates)
{
for (index = 0; index < call->cReaders; index++)
2011-10-15 19:30:10 +04:00
{
rgReaderState = &call->rgReaderStates[index];
2011-10-15 19:30:10 +04:00
if (rgReaderState->szReader)
free((void*) rgReaderState->szReader);
2011-10-15 19:30:10 +04:00
}
free(call->rgReaderStates);
}
2011-10-15 19:30:10 +04:00
free(ret.rgReaderStates);
return ret.ReturnCode;
}
static UINT32 smartcard_GetStatusChangeW_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, GetStatusChangeW_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_get_status_change_w_call(smartcard, irp->input, call);
smartcard_trace_get_status_change_w_call(smartcard, call);
return status;
}
static UINT32 smartcard_GetStatusChangeW_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, GetStatusChangeW_Call* call)
{
LONG status;
UINT32 index;
SCARDCONTEXT hContext;
GetStatusChange_Return ret;
LPSCARD_READERSTATEW rgReaderState = NULL;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
status = ret.ReturnCode = SCardGetStatusChangeW(hContext, call->dwTimeOut, call->rgReaderStates, call->cReaders);
2011-10-15 19:30:10 +04:00
if (status && (status != SCARD_E_TIMEOUT) && (status != SCARD_E_CANCELLED))
return status;
ret.cReaders = call->cReaders;
ret.rgReaderStates = (ReaderState_Return*) calloc(ret.cReaders, sizeof(ReaderState_Return));
for (index = 0; index < ret.cReaders; index++)
2011-10-15 19:30:10 +04:00
{
ret.rgReaderStates[index].dwCurrentState = call->rgReaderStates[index].dwCurrentState;
ret.rgReaderStates[index].dwEventState = call->rgReaderStates[index].dwEventState;
ret.rgReaderStates[index].cbAtr = call->rgReaderStates[index].cbAtr;
CopyMemory(&(ret.rgReaderStates[index].rgbAtr), &(call->rgReaderStates[index].rgbAtr), 32);
2011-10-15 19:30:10 +04:00
}
smartcard_trace_get_status_change_return(smartcard, &ret, TRUE);
status = smartcard_pack_get_status_change_return(smartcard, irp->output, &ret);
2011-10-15 19:30:10 +04:00
if (status)
return status;
if (call->rgReaderStates)
{
for (index = 0; index < call->cReaders; index++)
{
rgReaderState = &call->rgReaderStates[index];
if (rgReaderState->szReader)
free((void*) rgReaderState->szReader);
}
free(call->rgReaderStates);
}
free(ret.rgReaderStates);
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_Cancel_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_context_call(smartcard, irp->input, call);
smartcard_trace_context_call(smartcard, call, "Cancel");
return status;
}
static UINT32 smartcard_Cancel_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Context_Call* call)
{
LONG status;
SCARDCONTEXT hContext;
Long_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
2011-10-15 19:30:10 +04:00
status = ret.ReturnCode = SCardCancel(hContext);
smartcard_trace_long_return(smartcard, &ret, "Cancel");
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_ConnectA_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, ConnectA_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
2011-10-15 19:30:10 +04:00
status = smartcard_unpack_connect_a_call(smartcard, irp->input, call);
2013-10-24 23:34:14 +04:00
smartcard_trace_connect_a_call(smartcard, call);
return status;
}
static UINT32 smartcard_ConnectA_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, ConnectA_Call* call)
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Connect_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->Common.hContext));
if ((call->Common.dwPreferredProtocols == SCARD_PROTOCOL_UNDEFINED) &&
(call->Common.dwShareMode != SCARD_SHARE_DIRECT))
2014-04-07 07:08:40 +04:00
{
call->Common.dwPreferredProtocols = SCARD_PROTOCOL_Tx;
2014-04-07 07:08:40 +04:00
}
status = ret.ReturnCode = SCardConnectA(hContext, (char*) call->szReader, call->Common.dwShareMode,
call->Common.dwPreferredProtocols, &hCard, &ret.dwActiveProtocol);
if (status)
return status;
smartcard_scard_context_native_to_redir(smartcard, &(ret.hContext), hContext);
smartcard_scard_handle_native_to_redir(smartcard, &(ret.hCard), hCard);
smartcard_trace_connect_return(smartcard, &ret);
status = smartcard_pack_connect_return(smartcard, irp->output, &ret);
if (status)
return status;
2013-10-24 23:34:14 +04:00
if (call->szReader)
free(call->szReader);
return ret.ReturnCode;
}
static UINT32 smartcard_ConnectW_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, ConnectW_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_connect_w_call(smartcard, irp->input, call);
smartcard_trace_connect_w_call(smartcard, call);
return status;
}
static UINT32 smartcard_ConnectW_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, ConnectW_Call* call)
{
LONG status;
SCARDCONTEXT hContext;
SCARDHANDLE hCard;
Connect_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->Common.hContext));
if ((call->Common.dwPreferredProtocols == SCARD_PROTOCOL_UNDEFINED) &&
(call->Common.dwShareMode != SCARD_SHARE_DIRECT))
2014-04-07 07:08:40 +04:00
{
call->Common.dwPreferredProtocols = SCARD_PROTOCOL_Tx;
2014-04-07 07:08:40 +04:00
}
status = ret.ReturnCode = SCardConnectW(hContext, (WCHAR*) call->szReader, call->Common.dwShareMode,
call->Common.dwPreferredProtocols, &hCard, &ret.dwActiveProtocol);
2011-10-15 19:30:10 +04:00
if (status)
return status;
smartcard_scard_context_native_to_redir(smartcard, &(ret.hContext), hContext);
smartcard_scard_handle_native_to_redir(smartcard, &(ret.hCard), hCard);
smartcard_trace_connect_return(smartcard, &ret);
2014-04-07 07:08:40 +04:00
status = smartcard_pack_connect_return(smartcard, irp->output, &ret);
if (status)
return status;
if (call->szReader)
free(call->szReader);
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_Reconnect_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Reconnect_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_reconnect_call(smartcard, irp->input, call);
smartcard_trace_reconnect_call(smartcard, call);
return status;
}
static UINT32 smartcard_Reconnect_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Reconnect_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
2011-10-15 19:30:10 +04:00
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Reconnect_Return ret;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2013-10-24 23:34:14 +04:00
status = ret.ReturnCode = SCardReconnect(hCard, call->dwShareMode,
call->dwPreferredProtocols, call->dwInitialization, &ret.dwActiveProtocol);
smartcard_trace_reconnect_return(smartcard, &ret);
status = smartcard_pack_reconnect_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
return ret.ReturnCode;
}
static UINT32 smartcard_Disconnect_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, call);
smartcard_trace_hcard_and_disposition_call(smartcard, call, "Disconnect");
return status;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_Disconnect_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
2011-10-15 19:30:10 +04:00
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Long_Return ret;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2013-10-24 23:34:14 +04:00
status = ret.ReturnCode = SCardDisconnect(hCard, call->dwDisposition);
smartcard_trace_long_return(smartcard, &ret, "Disconnect");
if (status != SCARD_S_SUCCESS)
return status;
return ret.ReturnCode;
}
2011-10-15 19:30:10 +04:00
static UINT32 smartcard_BeginTransaction_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, call);
smartcard_trace_hcard_and_disposition_call(smartcard, call, "BeginTransaction");
return status;
}
static UINT32 smartcard_BeginTransaction_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Long_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2013-10-24 23:34:14 +04:00
status = ret.ReturnCode = SCardBeginTransaction(hCard);
smartcard_trace_long_return(smartcard, &ret, "BeginTransaction");
return ret.ReturnCode;
}
static UINT32 smartcard_EndTransaction_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_hcard_and_disposition_call(smartcard, irp->input, call);
smartcard_trace_hcard_and_disposition_call(smartcard, call, "EndTransaction");
return status;
}
static UINT32 smartcard_EndTransaction_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, HCardAndDisposition_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Long_Return ret;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2013-10-24 23:34:14 +04:00
status = ret.ReturnCode = SCardEndTransaction(hCard, call->dwDisposition);
smartcard_trace_long_return(smartcard, &ret, "EndTransaction");
return ret.ReturnCode;
}
static UINT32 smartcard_State_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, State_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_state_call(smartcard, irp->input, call);
return status;
}
static UINT32 smartcard_State_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, State_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
2011-10-15 19:30:10 +04:00
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
State_Return ret;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2011-10-15 19:30:10 +04:00
ret.cbAtrLen = SCARD_ATR_LENGTH;
status = ret.ReturnCode = SCardState(hCard, &ret.dwState, &ret.dwProtocol, (BYTE*) &ret.rgAtr, &ret.cbAtrLen);
2011-10-15 19:30:10 +04:00
status = smartcard_pack_state_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static DWORD smartcard_StatusA_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Status_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_status_call(smartcard, irp->input, call);
smartcard_trace_status_call(smartcard, call, FALSE);
return status;
}
static DWORD smartcard_StatusA_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Status_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
2011-10-15 19:30:10 +04:00
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Status_Return ret = { 0 };
DWORD cchReaderLen = 0;
LPSTR mszReaderNames = NULL;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
if (call->cbAtrLen > 32)
call->cbAtrLen = 32;
2011-10-15 19:30:10 +04:00
ret.cbAtrLen = call->cbAtrLen;
ZeroMemory(ret.pbAtr, 32);
2014-04-06 03:58:48 +04:00
cchReaderLen = SCARD_AUTOALLOCATE;
status = ret.ReturnCode = SCardStatusA(hCard, (LPSTR) &mszReaderNames, &cchReaderLen,
&ret.dwState, &ret.dwProtocol, (BYTE*) &ret.pbAtr, &ret.cbAtrLen);
2011-10-15 19:30:10 +04:00
if (status == SCARD_S_SUCCESS)
{
ret.mszReaderNames = (BYTE*) mszReaderNames;
ret.cBytes = cchReaderLen;
}
2011-10-15 19:30:10 +04:00
smartcard_trace_status_return(smartcard, &ret, FALSE);
status = smartcard_pack_status_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
2011-10-15 19:30:10 +04:00
if (mszReaderNames)
SCardFreeMemory(hContext, mszReaderNames);
return ret.ReturnCode;
}
static DWORD smartcard_StatusW_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Status_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_status_call(smartcard, irp->input, call);
smartcard_trace_status_call(smartcard, call, TRUE);
return status;
}
static DWORD smartcard_StatusW_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Status_Call* call)
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Status_Return ret;
DWORD cchReaderLen = 0;
LPWSTR mszReaderNames = NULL;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
if (call->cbAtrLen > 32)
call->cbAtrLen = 32;
ret.cbAtrLen = call->cbAtrLen;
ZeroMemory(ret.pbAtr, 32);
cchReaderLen = SCARD_AUTOALLOCATE;
status = ret.ReturnCode = SCardStatusW(hCard, (LPWSTR) &mszReaderNames, &cchReaderLen,
&ret.dwState, &ret.dwProtocol, (BYTE*) &ret.pbAtr, &ret.cbAtrLen);
ret.mszReaderNames = (BYTE*) mszReaderNames;
2014-04-11 01:36:42 +04:00
ret.cBytes = cchReaderLen * 2;
smartcard_trace_status_return(smartcard, &ret, TRUE);
status = smartcard_pack_status_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
2011-10-15 19:30:10 +04:00
if (mszReaderNames)
SCardFreeMemory(hContext, mszReaderNames);
2011-10-15 19:30:10 +04:00
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_Transmit_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Transmit_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_transmit_call(smartcard, irp->input, call);
smartcard_trace_transmit_call(smartcard, call);
return status;
}
static UINT32 smartcard_Transmit_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Transmit_Call* call)
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Transmit_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2014-04-07 07:08:40 +04:00
ret.cbRecvLength = 0;
ret.pbRecvBuffer = NULL;
if (call->cbRecvLength && !call->fpbRecvBufferIsNULL)
2014-04-04 01:29:12 +04:00
{
if (call->cbRecvLength >= 66560)
call->cbRecvLength = 66560;
ret.cbRecvLength = call->cbRecvLength;
2014-04-07 07:08:40 +04:00
ret.pbRecvBuffer = (BYTE*) malloc(ret.cbRecvLength);
2014-04-04 01:29:12 +04:00
}
2011-10-15 19:30:10 +04:00
ret.pioRecvPci = call->pioRecvPci;
2014-04-04 01:29:12 +04:00
status = ret.ReturnCode = SCardTransmit(hCard, call->pioSendPci, call->pbSendBuffer,
call->cbSendLength, ret.pioRecvPci, ret.pbRecvBuffer, &(ret.cbRecvLength));
smartcard_trace_transmit_return(smartcard, &ret);
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
if (status != SCARD_S_SUCCESS)
return status;
if (call->pbSendBuffer)
free(call->pbSendBuffer);
2014-04-07 07:08:40 +04:00
if (ret.pbRecvBuffer)
free(ret.pbRecvBuffer);
if (call->pioSendPci)
free(call->pioSendPci);
if (call->pioRecvPci)
free(call->pioRecvPci);
2011-10-15 19:30:10 +04:00
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_Control_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Control_Call* call)
{
LONG status;
2011-10-15 19:30:10 +04:00
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_control_call(smartcard, irp->input, call);
smartcard_trace_control_call(smartcard, call);
2011-10-15 19:30:10 +04:00
return status;
}
static UINT32 smartcard_Control_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Control_Call* call)
{
LONG status;
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
Control_Return ret;
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
ret.cbOutBufferSize = call->cbOutBufferSize;
ret.pvOutBuffer = (BYTE*) malloc(call->cbOutBufferSize);
status = ret.ReturnCode = SCardControl(hCard,
call->dwControlCode, call->pvInBuffer, call->cbInBufferSize,
ret.pvOutBuffer, call->cbOutBufferSize, &ret.cbOutBufferSize);
smartcard_trace_control_return(smartcard, &ret);
status = smartcard_pack_control_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
2011-10-15 19:30:10 +04:00
if (call->pvInBuffer)
free(call->pvInBuffer);
if (ret.pvOutBuffer)
free(ret.pvOutBuffer);
2011-10-15 19:30:10 +04:00
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_GetAttrib_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, GetAttrib_Call* call)
{
LONG status;
if (!call)
return STATUS_NO_MEMORY;
status = smartcard_unpack_get_attrib_call(smartcard, irp->input, call);
smartcard_trace_get_attrib_call(smartcard, call);
return status;
}
static UINT32 smartcard_GetAttrib_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, GetAttrib_Call* call)
2011-10-15 19:30:10 +04:00
{
LONG status;
DWORD cbAttrLen;
2011-10-15 19:30:10 +04:00
SCARDHANDLE hCard;
SCARDCONTEXT hContext;
GetAttrib_Return ret;
2011-10-15 19:30:10 +04:00
hContext = smartcard_scard_context_native_from_redir(smartcard, &(call->hContext));
hCard = smartcard_scard_handle_native_from_redir(smartcard, &(call->hCard));
2011-10-15 19:30:10 +04:00
ret.pbAttr = NULL;
if (call->fpbAttrIsNULL)
call->cbAttrLen = 0;
if (call->cbAttrLen)
ret.pbAttr = (BYTE*) malloc(call->cbAttrLen);
cbAttrLen = call->cbAttrLen;
status = ret.ReturnCode = SCardGetAttrib(hCard, call->dwAttrId, ret.pbAttr, &cbAttrLen);
2014-04-07 09:19:58 +04:00
ret.cbAttrLen = cbAttrLen;
2011-10-15 19:30:10 +04:00
smartcard_trace_get_attrib_return(smartcard, &ret, call->dwAttrId);
if (ret.ReturnCode)
2011-10-15 19:30:10 +04:00
{
WLog_Print(smartcard->log, WLOG_WARN,
"SCardGetAttrib: %s (0x%08X) cbAttrLen: %d\n",
SCardGetAttributeString(call->dwAttrId), call->dwAttrId, call->cbAttrLen);
Stream_Zero(irp->output, 256);
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
status = smartcard_pack_get_attrib_return(smartcard, irp->output, &ret);
if (status != SCARD_S_SUCCESS)
return status;
free(ret.pbAttr);
2011-10-15 19:30:10 +04:00
return ret.ReturnCode;
2011-10-15 19:30:10 +04:00
}
static UINT32 smartcard_AccessStartedEvent_Decode(SMARTCARD_DEVICE* smartcard, IRP* irp, Long_Call* call)
2011-10-15 19:30:10 +04:00
{
if (!call)
return STATUS_NO_MEMORY;
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));
return SCARD_F_INTERNAL_ERROR;
}
2013-10-24 23:34:14 +04:00
Stream_Read_UINT32(irp->input, call->LongValue); /* Unused (4 bytes) */
return SCARD_S_SUCCESS;
}
static UINT32 smartcard_AccessStartedEvent_Call(SMARTCARD_DEVICE* smartcard, IRP* irp, Long_Call* call)
{
UINT32 status;
Long_Return ret;
status = ret.ReturnCode = SCARD_S_SUCCESS;
if (!smartcard->StartedEvent)
smartcard->StartedEvent = SCardAccessStartedEvent();
if (!smartcard->StartedEvent)
status = ret.ReturnCode = SCARD_E_NO_SERVICE;
return status;
2011-10-15 19:30:10 +04:00
}
UINT32 smartcard_irp_device_control_decode(SMARTCARD_DEVICE* smartcard, SMARTCARD_OPERATION* operation)
2011-10-15 19:30:10 +04:00
{
IRP* irp;
UINT32 status;
UINT32 offset;
void* call = NULL;
UINT32 ioControlCode;
2014-04-04 08:56:24 +04:00
UINT32 outputBufferLength;
UINT32 inputBufferLength;
irp = operation->irp;
/* Device Control Request */
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 SCARD_F_INTERNAL_ERROR;
}
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
operation->ioControlCode = ioControlCode;
if (Stream_Length(irp->input) != (Stream_GetPosition(irp->input) + inputBufferLength))
{
WLog_Print(smartcard->log, WLOG_WARN,
"InputBufferLength mismatch: Actual: %d Expected: %d\n",
Stream_Length(irp->input), Stream_GetPosition(irp->input) + inputBufferLength);
return SCARD_F_INTERNAL_ERROR;
}
WLog_Print(smartcard->log, WLOG_DEBUG, "%s (0x%08X) FileId: %d CompletionId: %d",
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode, irp->FileId, irp->CompletionId);
#if 1
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
if ((ioControlCode != SCARD_IOCTL_ACCESSSTARTEDEVENT) &&
(ioControlCode != SCARD_IOCTL_RELEASESTARTEDEVENT))
{
status = smartcard_unpack_common_type_header(smartcard, irp->input);
if (status)
return SCARD_F_INTERNAL_ERROR;
status = smartcard_unpack_private_type_header(smartcard, irp->input);
if (status)
return SCARD_F_INTERNAL_ERROR;
}
/* Decode */
switch (ioControlCode)
{
case SCARD_IOCTL_ESTABLISHCONTEXT:
call = calloc(1, sizeof(EstablishContext_Call));
status = smartcard_EstablishContext_Decode(smartcard, irp, (EstablishContext_Call*) call);
break;
case SCARD_IOCTL_RELEASECONTEXT:
call = calloc(1, sizeof(Context_Call));
status = smartcard_ReleaseContext_Decode(smartcard, irp, (Context_Call*) call);
break;
case SCARD_IOCTL_ISVALIDCONTEXT:
call = calloc(1, sizeof(Context_Call));
status = smartcard_IsValidContext_Decode(smartcard, irp, (Context_Call*) call);
break;
case SCARD_IOCTL_LISTREADERGROUPSA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_LISTREADERGROUPSW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_LISTREADERSA:
call = calloc(1, sizeof(ListReaders_Call));
status = smartcard_ListReadersA_Decode(smartcard, irp, (ListReaders_Call*) call);
break;
case SCARD_IOCTL_LISTREADERSW:
call = calloc(1, sizeof(ListReaders_Call));
status = smartcard_ListReadersW_Decode(smartcard, irp, (ListReaders_Call*) call);
break;
case SCARD_IOCTL_INTRODUCEREADERGROUPA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_INTRODUCEREADERGROUPW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_FORGETREADERGROUPA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_FORGETREADERGROUPW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_INTRODUCEREADERA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_INTRODUCEREADERW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_FORGETREADERA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_FORGETREADERW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_ADDREADERTOGROUPA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_ADDREADERTOGROUPW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_REMOVEREADERFROMGROUPA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_REMOVEREADERFROMGROUPW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_LOCATECARDSA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_LOCATECARDSW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_GETSTATUSCHANGEA:
call = calloc(1, sizeof(GetStatusChangeA_Call));
status = smartcard_GetStatusChangeA_Decode(smartcard, irp, (GetStatusChangeA_Call*) call);
break;
case SCARD_IOCTL_GETSTATUSCHANGEW:
call = calloc(1, sizeof(GetStatusChangeW_Call));
status = smartcard_GetStatusChangeW_Decode(smartcard, irp, (GetStatusChangeW_Call*) call);
break;
case SCARD_IOCTL_CANCEL:
call = calloc(1, sizeof(Context_Call));
status = smartcard_Cancel_Decode(smartcard, irp, (Context_Call*) call);
break;
case SCARD_IOCTL_CONNECTA:
call = calloc(1, sizeof(ConnectA_Call));
status = smartcard_ConnectA_Decode(smartcard, irp, (ConnectA_Call*) call);
break;
case SCARD_IOCTL_CONNECTW:
call = calloc(1, sizeof(ConnectW_Call));
status = smartcard_ConnectW_Decode(smartcard, irp, (ConnectW_Call*) call);
break;
case SCARD_IOCTL_RECONNECT:
call = calloc(1, sizeof(Reconnect_Call));
status = smartcard_Reconnect_Decode(smartcard, irp, (Reconnect_Call*) call);
break;
case SCARD_IOCTL_DISCONNECT:
call = calloc(1, sizeof(HCardAndDisposition_Call));
status = smartcard_Disconnect_Decode(smartcard, irp, (HCardAndDisposition_Call*) call);
break;
case SCARD_IOCTL_BEGINTRANSACTION:
call = calloc(1, sizeof(HCardAndDisposition_Call));
status = smartcard_BeginTransaction_Decode(smartcard, irp, (HCardAndDisposition_Call*) call);
break;
case SCARD_IOCTL_ENDTRANSACTION:
call = calloc(1, sizeof(HCardAndDisposition_Call));
status = smartcard_EndTransaction_Decode(smartcard, irp, (HCardAndDisposition_Call*) call);
break;
case SCARD_IOCTL_STATE:
call = calloc(1, sizeof(State_Call));
status = smartcard_State_Decode(smartcard, irp, (State_Call*) call);
break;
case SCARD_IOCTL_STATUSA:
call = calloc(1, sizeof(Status_Call));
status = smartcard_StatusA_Decode(smartcard, irp, (Status_Call*) call);
break;
case SCARD_IOCTL_STATUSW:
call = calloc(1, sizeof(Status_Call));
status = smartcard_StatusW_Decode(smartcard, irp, (Status_Call*) call);
break;
case SCARD_IOCTL_TRANSMIT:
call = calloc(1, sizeof(Transmit_Call));
status = smartcard_Transmit_Decode(smartcard, irp, (Transmit_Call*) call);
break;
case SCARD_IOCTL_CONTROL:
call = calloc(1, sizeof(Control_Call));
status = smartcard_Control_Decode(smartcard, irp, (Control_Call*) call);
break;
case SCARD_IOCTL_GETATTRIB:
call = calloc(1, sizeof(GetAttrib_Call));
status = smartcard_GetAttrib_Decode(smartcard, irp, (GetAttrib_Call*) call);
break;
case SCARD_IOCTL_SETATTRIB:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_ACCESSSTARTEDEVENT:
call = calloc(1, sizeof(Long_Call));
status = smartcard_AccessStartedEvent_Decode(smartcard, irp, (Long_Call*) call);
break;
case SCARD_IOCTL_LOCATECARDSBYATRA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_LOCATECARDSBYATRW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_READCACHEA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_READCACHEW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_WRITECACHEA:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_WRITECACHEW:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_GETTRANSMITCOUNT:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_RELEASESTARTEDEVENT:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_GETREADERICON:
status = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_GETDEVICETYPEID:
status = SCARD_F_INTERNAL_ERROR;
break;
default:
status = SCARD_F_INTERNAL_ERROR;
break;
}
if ((ioControlCode != SCARD_IOCTL_ACCESSSTARTEDEVENT) &&
(ioControlCode != SCARD_IOCTL_RELEASESTARTEDEVENT))
{
offset = (RDPDR_DEVICE_IO_REQUEST_LENGTH + RDPDR_DEVICE_IO_CONTROL_REQ_HDR_LENGTH);
smartcard_unpack_read_size_align(smartcard, irp->input,
Stream_GetPosition(irp->input) - offset, 8);
}
if (((size_t) Stream_GetPosition(irp->input)) < Stream_Length(irp->input))
{
UINT32 difference;
difference = (int) (Stream_Length(irp->input) - Stream_GetPosition(irp->input));
WLog_Print(smartcard->log, WLOG_WARN,
"IRP was not fully parsed %s (0x%08X): Actual: %d, Expected: %d, Difference: %d",
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode,
(int) Stream_GetPosition(irp->input), (int) Stream_Length(irp->input), difference);
winpr_HexDump(Stream_Pointer(irp->input), difference);
}
if (((size_t) Stream_GetPosition(irp->input)) > Stream_Length(irp->input))
{
UINT32 difference;
difference = (int) (Stream_GetPosition(irp->input) - Stream_Length(irp->input));
WLog_Print(smartcard->log, WLOG_WARN,
"IRP was parsed beyond its end %s (0x%08X): Actual: %d, Expected: %d, Difference: %d",
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode,
(int) Stream_GetPosition(irp->input), (int) Stream_Length(irp->input), difference);
}
if (status != SCARD_S_SUCCESS)
{
free(call);
call = NULL;
}
operation->call = call;
return status;
}
UINT32 smartcard_irp_device_control_call(SMARTCARD_DEVICE* smartcard, SMARTCARD_OPERATION* operation)
{
IRP* irp;
UINT32 result;
UINT32 offset;
ULONG_PTR* call;
UINT32 ioControlCode;
UINT32 outputBufferLength;
UINT32 objectBufferLength;
irp = operation->irp;
call = operation->call;
ioControlCode = operation->ioControlCode;
/**
* [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.
*/
Stream_EnsureRemainingCapacity(irp->output, 2048);
/* Device Control Response */
Stream_Seek_UINT32(irp->output); /* OutputBufferLength (4 bytes) */
Stream_Seek(irp->output, SMARTCARD_COMMON_TYPE_HEADER_LENGTH); /* CommonTypeHeader (8 bytes) */
Stream_Seek(irp->output, SMARTCARD_PRIVATE_TYPE_HEADER_LENGTH); /* PrivateTypeHeader (8 bytes) */
Stream_Seek_UINT32(irp->output); /* Result (4 bytes) */
/* Call */
switch (ioControlCode)
2011-10-15 19:30:10 +04:00
{
case SCARD_IOCTL_ESTABLISHCONTEXT:
result = smartcard_EstablishContext_Call(smartcard, irp, (EstablishContext_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_RELEASECONTEXT:
result = smartcard_ReleaseContext_Call(smartcard, irp, (Context_Call*) call);
break;
case SCARD_IOCTL_ISVALIDCONTEXT:
result = smartcard_IsValidContext_Call(smartcard, irp, (Context_Call*) call);
2011-10-15 19:30:10 +04:00
break;
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;
case SCARD_IOCTL_LISTREADERSA:
result = smartcard_ListReadersA_Call(smartcard, irp, (ListReaders_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_LISTREADERSW:
result = smartcard_ListReadersW_Call(smartcard, irp, (ListReaders_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_INTRODUCEREADERGROUPA:
result = SCARD_F_INTERNAL_ERROR;
break;
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;
case SCARD_IOCTL_GETSTATUSCHANGEA:
result = smartcard_GetStatusChangeA_Call(smartcard, irp, (GetStatusChangeA_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_GETSTATUSCHANGEW:
result = smartcard_GetStatusChangeW_Call(smartcard, irp, (GetStatusChangeW_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_CANCEL:
result = smartcard_Cancel_Call(smartcard, irp, (Context_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_CONNECTA:
result = smartcard_ConnectA_Call(smartcard, irp, (ConnectA_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_CONNECTW:
result = smartcard_ConnectW_Call(smartcard, irp, (ConnectW_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_RECONNECT:
result = smartcard_Reconnect_Call(smartcard, irp, (Reconnect_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_DISCONNECT:
result = smartcard_Disconnect_Call(smartcard, irp, (HCardAndDisposition_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_BEGINTRANSACTION:
result = smartcard_BeginTransaction_Call(smartcard, irp, (HCardAndDisposition_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_ENDTRANSACTION:
result = smartcard_EndTransaction_Call(smartcard, irp, (HCardAndDisposition_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_STATE:
result = smartcard_State_Call(smartcard, irp, (State_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_STATUSA:
result = smartcard_StatusA_Call(smartcard, irp, (Status_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_STATUSW:
result = smartcard_StatusW_Call(smartcard, irp, (Status_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_TRANSMIT:
result = smartcard_Transmit_Call(smartcard, irp, (Transmit_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_CONTROL:
result = smartcard_Control_Call(smartcard, irp, (Control_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_GETATTRIB:
result = smartcard_GetAttrib_Call(smartcard, irp, (GetAttrib_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_SETATTRIB:
result = SCARD_F_INTERNAL_ERROR;
break;
case SCARD_IOCTL_ACCESSSTARTEDEVENT:
result = smartcard_AccessStartedEvent_Call(smartcard, irp, (Long_Call*) call);
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_LOCATECARDSBYATRA:
result = SCARD_F_INTERNAL_ERROR;
2011-10-15 19:30:10 +04:00
break;
case SCARD_IOCTL_LOCATECARDSBYATRW:
result = SCARD_F_INTERNAL_ERROR;
2011-10-15 19:30:10 +04:00
break;
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;
}
free(call);
/**
* [MS-RPCE] 2.2.6.3 Primitive Type Serialization
* The type MUST be aligned on an 8-byte boundary. If the size of the
* primitive type is not a multiple of 8 bytes, the data MUST be padded.
*/
if ((ioControlCode != SCARD_IOCTL_ACCESSSTARTEDEVENT) &&
(ioControlCode != SCARD_IOCTL_RELEASESTARTEDEVENT))
{
offset = (RDPDR_DEVICE_IO_RESPONSE_LENGTH + RDPDR_DEVICE_IO_CONTROL_RSP_HDR_LENGTH);
smartcard_pack_write_size_align(smartcard, irp->output,
Stream_GetPosition(irp->output) - offset, 8);
}
if ((result != SCARD_S_SUCCESS) && (result != SCARD_E_TIMEOUT) &&
(result != SCARD_E_NO_READERS_AVAILABLE) && (result != SCARD_E_NO_SERVICE))
{
WLog_Print(smartcard->log, WLOG_WARN,
"IRP failure: %s (0x%08X), status: %s (0x%08X)",
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode,
SCardGetErrorString(result), result);
}
2011-10-15 19:30:10 +04:00
irp->IoStatus = 0;
if ((result & 0xC0000000) == 0xC0000000)
{
/* NTSTATUS error */
irp->IoStatus = result;
Stream_SetPosition(irp->output, RDPDR_DEVICE_IO_RESPONSE_LENGTH);
WLog_Print(smartcard->log, WLOG_WARN,
"IRP failure: %s (0x%08X), ntstatus: 0x%08X",
smartcard_get_ioctl_string(ioControlCode, TRUE), ioControlCode, result);
}
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
/* Device Control Response */
Stream_Write_UINT32(irp->output, outputBufferLength); /* OutputBufferLength (4 bytes) */
2011-10-15 19:30:10 +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
Stream_Write_UINT32(irp->output, result); /* Result (4 bytes) */
2011-10-15 19:30:10 +04:00
Stream_SetPosition(irp->output, Stream_Length(irp->output));
return SCARD_S_SUCCESS;
2011-10-15 19:30:10 +04:00
}