FreeRDP/channels/smartcard/client/smartcard_main.c

215 lines
5.1 KiB
C
Raw Normal View History

/**
2012-10-09 07:02:04 +04:00
* FreeRDP: A Remote Desktop Protocol Implementation
* Smartcard Device Service Virtual Channel
*
* Copyright 2011 O.S. Systems Software Ltda.
* Copyright 2011 Eduardo Fiss Beloni <beloni@ossystems.com.br>
* 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>
#include <winpr/crt.h>
#include <winpr/smartcard.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
static void smartcard_free(DEVICE* device)
2011-10-15 19:30:10 +04:00
{
SMARTCARD_DEVICE* smartcard = (SMARTCARD_DEVICE*) device;
2011-10-15 19:30:10 +04:00
MessageQueue_PostQuit(smartcard->IrpQueue, 0);
WaitForSingleObject(smartcard->thread, INFINITE);
CloseHandle(smartcard->thread);
Stream_Free(smartcard->device.data, TRUE);
MessageQueue_Free(smartcard->IrpQueue);
ListDictionary_Free(smartcard->OutstandingIrps);
free(device);
2011-10-15 19:30:10 +04:00
}
/**
* Initialization occurs when the protocol server sends a device announce message.
* At that time, dwDeviceId MUST receive the unique device ID announced.
* The OutstandingIrps list MUST be set to the empty list.
*/
static void smartcard_init(DEVICE* device)
{
SMARTCARD_DEVICE* smartcard = (SMARTCARD_DEVICE*) device;
if (ListDictionary_Count(smartcard->OutstandingIrps) > 0)
{
fprintf(stderr, "Warning: smartcard device initialized with outstanding IRPs\n");
}
fprintf(stderr, "SmartCard Init\n");
}
IRP* smartcard_get_outstanding_irp_by_id(SMARTCARD_DEVICE* smartcard, UINT32 completionId)
{
IRP* irp = NULL;
void* key = (void*) (size_t) completionId;
irp = (IRP*) ListDictionary_GetItemValue(smartcard->OutstandingIrps, key);
return irp;
}
void smartcard_complete_irp(SMARTCARD_DEVICE* smartcard, IRP* irp)
{
void* key;
key = (void*) (size_t) irp->CompletionId;
ListDictionary_Remove(smartcard->OutstandingIrps, key);
irp->Complete(irp);
}
static void smartcard_process_irp(SMARTCARD_DEVICE* smartcard, IRP* irp)
2011-10-15 19:30:10 +04:00
{
void* key;
key = (void*) (size_t) irp->CompletionId;
ListDictionary_Add(smartcard->OutstandingIrps, key, irp);
2011-10-15 19:30:10 +04:00
switch (irp->MajorFunction)
{
case IRP_MJ_DEVICE_CONTROL:
smartcard_device_control(smartcard, irp);
2011-10-15 19:30:10 +04:00
break;
default:
fprintf(stderr, "MajorFunction 0x%X unexpected for smartcards.", irp->MajorFunction);
2011-10-15 19:30:10 +04:00
irp->IoStatus = STATUS_NOT_SUPPORTED;
smartcard_complete_irp(smartcard, irp);
2011-10-15 19:30:10 +04:00
break;
}
}
/**
* Multiple threads and SCardGetStatusChange:
* http://musclecard.996296.n3.nabble.com/Multiple-threads-and-SCardGetStatusChange-td4430.html
*/
static void* smartcard_thread_func(void* arg)
2011-10-15 19:30:10 +04:00
{
IRP* irp;
wMessage message;
SMARTCARD_DEVICE* smartcard = (SMARTCARD_DEVICE*) arg;
2011-10-15 19:30:10 +04:00
while (1)
2011-10-15 19:30:10 +04:00
{
if (!MessageQueue_Wait(smartcard->IrpQueue))
break;
if (!MessageQueue_Peek(smartcard->IrpQueue, &message, TRUE))
2011-10-15 19:30:10 +04:00
break;
if (message.id == WMQ_QUIT)
2011-10-15 19:30:10 +04:00
break;
irp = (IRP*) message.wParam;
if (irp)
smartcard_process_irp(smartcard, irp);
}
ExitThread(0);
return NULL;
}
static void smartcard_irp_request(DEVICE* device, IRP* irp)
2011-10-15 19:30:10 +04:00
{
SMARTCARD_DEVICE* smartcard = (SMARTCARD_DEVICE*) device;
MessageQueue_Post(smartcard->IrpQueue, NULL, 0, (void*) irp, NULL);
2011-10-15 19:30:10 +04:00
}
2012-10-14 10:38:58 +04:00
#ifdef STATIC_CHANNELS
#define DeviceServiceEntry smartcard_DeviceServiceEntry
#endif
int DeviceServiceEntry(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints)
2011-10-15 19:30:10 +04:00
{
char* name;
char* path;
int length, ck;
RDPDR_SMARTCARD* device;
SMARTCARD_DEVICE* smartcard;
2011-10-15 19:30:10 +04:00
device = (RDPDR_SMARTCARD*) pEntryPoints->device;
2014-04-04 01:29:12 +04:00
name = device->Name;
path = device->Path;
2011-10-15 19:30:10 +04:00
smartcard = (SMARTCARD_DEVICE*) calloc(1, sizeof(SMARTCARD_DEVICE));
if (!smartcard)
return -1;
2011-10-15 19:30:10 +04:00
smartcard->device.type = RDPDR_DTYP_SMARTCARD;
smartcard->device.name = "SCARD";
smartcard->device.IRPRequest = smartcard_irp_request;
smartcard->device.Init = smartcard_init;
smartcard->device.Free = smartcard_free;
2011-10-15 19:30:10 +04:00
length = strlen(smartcard->device.name);
smartcard->device.data = Stream_New(NULL, length + 1);
2011-10-15 19:30:10 +04:00
2013-09-12 16:55:25 +04:00
Stream_Write(smartcard->device.data, "SCARD", 6);
2011-10-15 19:30:10 +04:00
smartcard->name = NULL;
smartcard->path = NULL;
if (path)
{
smartcard->path = path;
smartcard->name = name;
}
2013-09-12 16:55:25 +04:00
else if (name)
{
if (1 == sscanf(name, "%d", &ck))
smartcard->path = name;
else
smartcard->name = name;
}
2011-10-15 19:30:10 +04:00
2014-04-04 01:29:12 +04:00
smartcard->log = WLog_Get("com.freerdp.channel.smartcard.client");
smartcard->IrpQueue = MessageQueue_New(NULL);
smartcard->OutstandingIrps = ListDictionary_New(TRUE);
smartcard->thread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE) smartcard_thread_func,
smartcard, CREATE_SUSPENDED, NULL);
2011-10-15 19:30:10 +04:00
pEntryPoints->RegisterDevice(pEntryPoints->devman, (DEVICE*) smartcard);
2011-10-15 19:30:10 +04:00
ResumeThread(smartcard->thread);
2011-10-15 19:30:10 +04:00
return 0;
}