FreeRDP
Loading...
Searching...
No Matches
pf_channel_smartcard.c
1
21#include <winpr/assert.h>
22#include <winpr/string.h>
23
24#include <winpr/smartcard.h>
25#include <winpr/pool.h>
26
27#include <freerdp/server/proxy/proxy_log.h>
28#include <freerdp/emulate/scard/smartcard_emulate.h>
29#include <freerdp/channels/scard.h>
30#include <freerdp/channels/rdpdr.h>
31#include <freerdp/utils/rdpdr_utils.h>
32
33#include <freerdp/utils/smartcard_operations.h>
34#include <freerdp/utils/smartcard_call.h>
35
36#include "pf_channel_smartcard.h"
37#include "pf_channel_rdpdr.h"
38
39#define TAG PROXY_TAG("channel.scard")
40
41#define SCARD_SVC_CHANNEL_NAME "SCARD"
42
43typedef struct
44{
46 scard_call_context* callctx;
47 wArrayList* workObjects;
48} pf_channel_client_context;
49
50typedef struct
51{
53 wStream* out;
54 pClientContext* pc;
55 wLog* log;
56 pf_scard_send_fkt_t send_fkt;
57} pf_channel_client_queue_element;
58
59WINPR_ATTR_NODISCARD
60static pf_channel_client_context* scard_get_client_context(pClientContext* pc)
61{
62 pf_channel_client_context* scard = nullptr;
63
64 WINPR_ASSERT(pc);
65 WINPR_ASSERT(pc->interceptContextMap);
66
67 scard = HashTable_GetItemValue(pc->interceptContextMap, SCARD_SVC_CHANNEL_NAME);
68 if (!scard)
69 WLog_WARN(TAG, "[%s] missing in pc->interceptContextMap", SCARD_SVC_CHANNEL_NAME);
70 return scard;
71}
72
73WINPR_ATTR_NODISCARD
74static BOOL pf_channel_client_write_iostatus(wStream* out, const SMARTCARD_OPERATION* op,
75 NTSTATUS ioStatus)
76{
77 UINT16 component = 0;
78 UINT16 packetid = 0;
79 UINT32 dID = 0;
80 UINT32 cID = 0;
81 size_t pos = 0;
82
83 WINPR_ASSERT(op);
84 WINPR_ASSERT(out);
85
86 pos = Stream_GetPosition(out);
87 Stream_SetPosition(out, 0);
88 if (!Stream_CheckAndLogRequiredLength(TAG, out, 16))
89 return FALSE;
90
91 Stream_Read_UINT16(out, component);
92 Stream_Read_UINT16(out, packetid);
93
94 Stream_Read_UINT32(out, dID);
95 Stream_Read_UINT32(out, cID);
96
97 WINPR_ASSERT(component == RDPDR_CTYP_CORE);
98 WINPR_ASSERT(packetid == PAKID_CORE_DEVICE_IOCOMPLETION);
99 WINPR_ASSERT(dID == op->deviceID);
100 WINPR_ASSERT(cID == op->completionID);
101
102 Stream_Write_INT32(out, ioStatus);
103 Stream_SetPosition(out, pos);
104 return TRUE;
105}
106
107struct thread_arg
108{
109 pf_channel_client_context* scard;
110 pf_channel_client_queue_element* e;
111};
112
113static void queue_free(void* obj);
114
115WINPR_ATTR_MALLOC(queue_free, 1)
116WINPR_ATTR_NODISCARD
117static void* queue_copy(const void* obj);
118
119static VOID irp_thread(WINPR_ATTR_UNUSED PTP_CALLBACK_INSTANCE Instance, PVOID Context,
120 PTP_WORK Work)
121{
122 struct thread_arg* arg = Context;
123 pf_channel_client_context* scard = arg->scard;
124 {
125 NTSTATUS ioStatus = 0;
126 LONG rc = smartcard_irp_device_control_call(arg->scard->callctx, arg->e->out, &ioStatus,
127 &arg->e->op);
128 if (rc == CHANNEL_RC_OK)
129 {
130 if (pf_channel_client_write_iostatus(arg->e->out, &arg->e->op, ioStatus))
131 arg->e->send_fkt(arg->e->log, arg->e->pc, arg->e->out);
132 }
133 }
134 queue_free(arg->e);
135 free(arg);
136 ArrayList_Remove(scard->workObjects, Work);
137}
138
139WINPR_ATTR_NODISCARD
140static BOOL start_irp_thread(pf_channel_client_context* scard,
141 const pf_channel_client_queue_element* e)
142{
143 PTP_WORK work = nullptr;
144 struct thread_arg* arg = calloc(1, sizeof(struct thread_arg));
145 if (!arg)
146 return FALSE;
147 arg->scard = scard;
148 arg->e = queue_copy(e);
149 if (!arg->e)
150 goto fail;
151
152 work = CreateThreadpoolWork(irp_thread, arg, nullptr);
153 if (!work)
154 goto fail;
155 ArrayList_Append(scard->workObjects, work);
156 SubmitThreadpoolWork(work);
157
158 return TRUE;
159
160fail:
161 if (arg)
162 queue_free(arg->e);
163 free(arg);
164 return FALSE;
165}
166
167BOOL pf_channel_smartcard_client_handle(wLog* log, pClientContext* pc, wStream* s, wStream* out,
168 pf_scard_send_fkt_t send_fkt)
169{
170 BOOL rc = FALSE;
171 LONG status = 0;
172 UINT32 FileId = 0;
173 UINT32 CompletionId = 0;
174 NTSTATUS ioStatus = 0;
175 pf_channel_client_queue_element e = WINPR_C_ARRAY_INIT;
176 pf_channel_client_context* scard = scard_get_client_context(pc);
177
178 WINPR_ASSERT(log);
179 WINPR_ASSERT(send_fkt);
180 WINPR_ASSERT(s);
181
182 if (!scard)
183 return FALSE;
184
185 e.log = log;
186 e.pc = pc;
187 e.out = out;
188 e.send_fkt = send_fkt;
189
190 /* Skip IRP header */
191 if (!Stream_CheckAndLogRequiredLength(TAG, s, 20))
192 return FALSE;
193 else
194 {
195 const uint32_t DeviceId = Stream_Get_UINT32(s); /* DeviceId (4 bytes) */
196 FileId = Stream_Get_UINT32(s); /* FileId (4 bytes) */
197 CompletionId = Stream_Get_UINT32(s); /* CompletionId (4 bytes) */
198 const uint32_t MajorFunction = Stream_Get_UINT32(s); /* MajorFunction (4 bytes) */
199 const uint32_t MinorFunction = Stream_Get_UINT32(s); /* MinorFunction (4 bytes) */
200
201 if (MajorFunction != IRP_MJ_DEVICE_CONTROL)
202 {
203 WLog_WARN(TAG, "[%s] Invalid IRP received, expected %s, got %s [0x%08" PRIx32 "]",
204 SCARD_SVC_CHANNEL_NAME, rdpdr_irp_string(IRP_MJ_DEVICE_CONTROL),
205 rdpdr_irp_string(MajorFunction), MinorFunction);
206 return FALSE;
207 }
208 e.op.completionID = CompletionId;
209 e.op.deviceID = DeviceId;
210
211 if (!rdpdr_write_iocompletion_header(out, DeviceId, CompletionId, 0))
212 return FALSE;
213 }
214
215 status = smartcard_irp_device_control_decode(s, CompletionId, FileId, &e.op);
216 if (status != 0)
217 goto fail;
218
219 switch (e.op.ioControlCode)
220 {
221 case SCARD_IOCTL_LISTREADERGROUPSA:
222 case SCARD_IOCTL_LISTREADERGROUPSW:
223 case SCARD_IOCTL_LISTREADERSA:
224 case SCARD_IOCTL_LISTREADERSW:
225 case SCARD_IOCTL_LOCATECARDSA:
226 case SCARD_IOCTL_LOCATECARDSW:
227 case SCARD_IOCTL_LOCATECARDSBYATRA:
228 case SCARD_IOCTL_LOCATECARDSBYATRW:
229 case SCARD_IOCTL_GETSTATUSCHANGEA:
230 case SCARD_IOCTL_GETSTATUSCHANGEW:
231 case SCARD_IOCTL_CONNECTA:
232 case SCARD_IOCTL_CONNECTW:
233 case SCARD_IOCTL_RECONNECT:
234 case SCARD_IOCTL_DISCONNECT:
235 case SCARD_IOCTL_BEGINTRANSACTION:
236 case SCARD_IOCTL_ENDTRANSACTION:
237 case SCARD_IOCTL_STATE:
238 case SCARD_IOCTL_STATUSA:
239 case SCARD_IOCTL_STATUSW:
240 case SCARD_IOCTL_TRANSMIT:
241 case SCARD_IOCTL_CONTROL:
242 case SCARD_IOCTL_GETATTRIB:
243 case SCARD_IOCTL_SETATTRIB:
244 if (!start_irp_thread(scard, &e))
245 goto fail;
246 return TRUE;
247
248 default:
249 status = smartcard_irp_device_control_call(scard->callctx, out, &ioStatus, &e.op);
250 if (status != 0)
251 goto fail;
252 if (!pf_channel_client_write_iostatus(out, &e.op, ioStatus))
253 goto fail;
254 break;
255 }
256
257 rc = send_fkt(log, pc, out) == CHANNEL_RC_OK;
258
259fail:
260 smartcard_operation_free(&e.op, FALSE);
261 return rc;
262}
263
264BOOL pf_channel_smartcard_server_handle(WINPR_ATTR_UNUSED pServerContext* ps,
265 WINPR_ATTR_UNUSED wStream* s)
266{
267 WLog_ERR(TAG, "TODO: unimplemented");
268 return TRUE;
269}
270
271static void channel_stop_and_wait(pf_channel_client_context* scard, BOOL reset)
272{
273 WINPR_ASSERT(scard);
274 smartcard_call_context_signal_stop(scard->callctx, FALSE);
275
276 while (ArrayList_Count(scard->workObjects) > 0)
277 {
278 PTP_WORK work = ArrayList_GetItem(scard->workObjects, 0);
279 if (!work)
280 continue;
281 WaitForThreadpoolWorkCallbacks(work, TRUE);
282 }
283
284 smartcard_call_context_signal_stop(scard->callctx, reset);
285}
286
287static void pf_channel_scard_client_context_free(InterceptContextMapEntry* base)
288{
289 pf_channel_client_context* entry = (pf_channel_client_context*)base;
290 if (!entry)
291 return;
292
293 /* Set the stop event.
294 * All threads waiting in blocking operations will abort at the next
295 * available polling slot */
296 channel_stop_and_wait(entry, FALSE);
297 ArrayList_Free(entry->workObjects);
298
299 smartcard_call_context_free(entry->callctx);
300 free(entry);
301}
302
303static void queue_free(void* obj)
304{
305 pf_channel_client_queue_element* element = obj;
306 if (!element)
307 return;
308 smartcard_operation_free(&element->op, FALSE);
309 Stream_Free(element->out, TRUE);
310 free(element);
311}
312
313WINPR_ATTR_MALLOC(queue_free, 1)
314WINPR_ATTR_NODISCARD
315static void* queue_copy(const void* obj)
316{
317 const pf_channel_client_queue_element* other = obj;
318 pf_channel_client_queue_element* copy = nullptr;
319 if (!other)
320 return nullptr;
321 copy = calloc(1, sizeof(pf_channel_client_queue_element));
322 if (!copy)
323 return nullptr;
324
325 *copy = *other;
326 copy->out = Stream_New(nullptr, Stream_Capacity(other->out));
327 if (!copy->out)
328 goto fail;
329 Stream_Write(copy->out, Stream_Buffer(other->out), Stream_GetPosition(other->out));
330 return copy;
331fail:
332 queue_free(copy);
333 return nullptr;
334}
335
336static void work_object_free(void* arg)
337{
338 PTP_WORK work = arg;
339 CloseThreadpoolWork(work);
340}
341
342BOOL pf_channel_smartcard_client_new(pClientContext* pc)
343{
344 pf_channel_client_context* scard = nullptr;
345 wObject* obj = nullptr;
346
347 WINPR_ASSERT(pc);
348 WINPR_ASSERT(pc->interceptContextMap);
349
350 scard = calloc(1, sizeof(pf_channel_client_context));
351 if (!scard)
352 return FALSE;
353 scard->base.free = pf_channel_scard_client_context_free;
354 scard->callctx = smartcard_call_context_new(pc->context.settings);
355 if (!scard->callctx)
356 goto fail;
357
358 scard->workObjects = ArrayList_New(TRUE);
359 if (!scard->workObjects)
360 goto fail;
361 obj = ArrayList_Object(scard->workObjects);
362 WINPR_ASSERT(obj);
363 obj->fnObjectFree = work_object_free;
364
365 return HashTable_Insert(pc->interceptContextMap, SCARD_SVC_CHANNEL_NAME, scard);
366fail:
367 pf_channel_scard_client_context_free(&scard->base);
368 return FALSE;
369}
370
371void pf_channel_smartcard_client_free(pClientContext* pc)
372{
373 WINPR_ASSERT(pc);
374 WINPR_ASSERT(pc->interceptContextMap);
375 HashTable_Remove(pc->interceptContextMap, SCARD_SVC_CHANNEL_NAME);
376}
377
378BOOL pf_channel_smartcard_client_emulate(pClientContext* pc)
379{
380 pf_channel_client_context* scard = scard_get_client_context(pc);
381 if (!scard)
382 return FALSE;
383 return smartcard_call_is_configured(scard->callctx);
384}
385
386BOOL pf_channel_smartcard_client_reset(pClientContext* pc)
387{
388 pf_channel_client_context* scard = scard_get_client_context(pc);
389 if (!scard)
390 return TRUE;
391
392 channel_stop_and_wait(scard, TRUE);
393 return TRUE;
394}
This struct contains function pointer to initialize/free objects.
Definition collections.h:52
OBJECT_FREE_FN fnObjectFree
Definition collections.h:58