Coverage Report

Created: 2026-04-12 06:58

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/winpr/libwinpr/smartcard/smartcard_pcsc.c
Line
Count
Source
1
/**
2
 * WinPR: Windows Portable Runtime
3
 * Smart Card API
4
 *
5
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6
 * Copyright 2020 Armin Novak <armin.novak@thincast.com>
7
 * Copyright 2020 Thincast Technologies GmbH
8
 *
9
 * Licensed under the Apache License, Version 2.0 (the "License");
10
 * you may not use this file except in compliance with the License.
11
 * You may obtain a copy of the License at
12
 *
13
 *     http://www.apache.org/licenses/LICENSE-2.0
14
 *
15
 * Unless required by applicable law or agreed to in writing, software
16
 * distributed under the License is distributed on an "AS IS" BASIS,
17
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18
 * See the License for the specific language governing permissions and
19
 * limitations under the License.
20
 */
21
22
#include <winpr/config.h>
23
24
#ifndef _WIN32
25
26
#ifdef __APPLE__
27
#include <sys/types.h>
28
#include <sys/param.h>
29
#include <sys/sysctl.h>
30
#include <string.h>
31
#include <ctype.h>
32
#include <errno.h>
33
#endif
34
35
#include <stdio.h>
36
#include <stdlib.h>
37
38
#include <winpr/crt.h>
39
#include <winpr/assert.h>
40
#include <winpr/synch.h>
41
#include <winpr/library.h>
42
#include <winpr/smartcard.h>
43
#include <winpr/collections.h>
44
#include <winpr/environment.h>
45
46
#include "smartcard_pcsc.h"
47
48
#include "../log.h"
49
#define TAG WINPR_TAG("smartcard")
50
51
#ifndef MIN
52
0
#define MIN(x, y) (((x) < (y)) ? (x) : (y))
53
#endif
54
55
#define WINSCARD_LOAD_PROC_EX(module, pcsc, _fname, _name)                   \
56
0
  do                                                                       \
57
0
  {                                                                        \
58
0
    WINPR_PRAGMA_DIAG_PUSH                                               \
59
0
    WINPR_PRAGMA_DIAG_IGNORED_PEDANTIC                                   \
60
0
    pcsc.pfn##_fname = GetProcAddressAs(module, #_name, fnPCSC##_fname); \
61
0
    WINPR_PRAGMA_DIAG_POP                                                \
62
0
  } while (0)
63
64
0
#define WINSCARD_LOAD_PROC(module, pcsc, _name) WINSCARD_LOAD_PROC_EX(module, pcsc, _name, _name)
65
66
/**
67
 * PC/SC transactions:
68
 * http://developersblog.wwpass.com/?p=180
69
 */
70
71
/**
72
 * Smart Card Logon on Windows Vista:
73
 * http://blogs.msdn.com/b/shivaram/archive/2007/02/26/smart-card-logon-on-windows-vista.aspx
74
 */
75
76
/**
77
 * The Smart Card Cryptographic Service Provider Cookbook:
78
 * http://msdn.microsoft.com/en-us/library/ms953432.aspx
79
 *
80
 * SCARDCONTEXT
81
 *
82
 * The context is a communication channel with the smart card resource manager and
83
 * all calls to the resource manager must go through this link.
84
 *
85
 * All functions that take a context as a parameter or a card handle as parameter,
86
 * which is indirectly associated with a particular context, may be blocking calls.
87
 * Examples of these are SCardGetStatusChange and SCardBeginTransaction, which takes
88
 * a card handle as a parameter. If such a function blocks then all operations wanting
89
 * to use the context are blocked as well. So, it is recommended that a CSP using
90
 * monitoring establishes at least two contexts with the resource manager; one for
91
 * monitoring (with SCardGetStatusChange) and one for other operations.
92
 *
93
 * If multiple cards are present, it is recommended that a separate context or pair
94
 * of contexts be established for each card to prevent operations on one card from
95
 * blocking operations on another.
96
 *
97
 * Example one
98
 *
99
 * The example below shows what can happen if a CSP using SCardGetStatusChange for
100
 * monitoring does not establish two contexts with the resource manager.
101
 * The context becomes unusable until SCardGetStatusChange unblocks.
102
 *
103
 * In this example, there is one process running called P1.
104
 * P1 calls SCardEstablishContext, which returns the context hCtx.
105
 * P1 calls SCardConnect (with the hCtx context) which returns a handle to the card, hCard.
106
 * P1 calls SCardGetStatusChange (with the hCtx context) which blocks because
107
 * there are no status changes to report.
108
 * Until the thread running SCardGetStatusChange unblocks, another thread in P1 trying to
109
 * perform an operation using the context hCtx (or the card hCard) will also be blocked.
110
 *
111
 * Example two
112
 *
113
 * The example below shows how transaction control ensures that operations meant to be
114
 * performed without interruption can do so safely within a transaction.
115
 *
116
 * In this example, there are two different processes running; P1 and P2.
117
 * P1 calls SCardEstablishContext, which returns the context hCtx1.
118
 * P2 calls SCardEstablishContext, which returns the context hCtx2.
119
 * P1 calls SCardConnect (with the hCtx1 context) which returns a handle to the card, hCard1.
120
 * P2 calls SCardConnect (with the hCtx2 context) which returns a handle to the same card, hCard2.
121
 * P1 calls SCardBeginTransaction (with the hCard 1 context).
122
 * Until P1 calls SCardEndTransaction (with the hCard1 context),
123
 * any operation using hCard2 will be blocked.
124
 * Once an operation using hCard2 is blocked and until it's returning,
125
 * any operation using hCtx2 (and hCard2) will also be blocked.
126
 */
127
128
//#define DISABLE_PCSC_SCARD_AUTOALLOCATE
129
#include "smartcard_pcsc.h"
130
131
0
#define PCSC_SCARD_PCI_T0 (&g_PCSC_rgSCardT0Pci)
132
0
#define PCSC_SCARD_PCI_T1 (&g_PCSC_rgSCardT1Pci)
133
0
#define PCSC_SCARD_PCI_RAW (&g_PCSC_rgSCardRawPci)
134
135
typedef PCSC_LONG (*fnPCSCSCardEstablishContext)(PCSC_DWORD dwScope, LPCVOID pvReserved1,
136
                                                 LPCVOID pvReserved2, LPSCARDCONTEXT phContext);
137
typedef PCSC_LONG (*fnPCSCSCardReleaseContext)(SCARDCONTEXT hContext);
138
typedef PCSC_LONG (*fnPCSCSCardIsValidContext)(SCARDCONTEXT hContext);
139
typedef PCSC_LONG (*fnPCSCSCardConnect)(SCARDCONTEXT hContext, LPCSTR szReader,
140
                                        PCSC_DWORD dwShareMode, PCSC_DWORD dwPreferredProtocols,
141
                                        LPSCARDHANDLE phCard, PCSC_LPDWORD pdwActiveProtocol);
142
typedef PCSC_LONG (*fnPCSCSCardReconnect)(SCARDHANDLE hCard, PCSC_DWORD dwShareMode,
143
                                          PCSC_DWORD dwPreferredProtocols,
144
                                          PCSC_DWORD dwInitialization,
145
                                          PCSC_LPDWORD pdwActiveProtocol);
146
typedef PCSC_LONG (*fnPCSCSCardDisconnect)(SCARDHANDLE hCard, PCSC_DWORD dwDisposition);
147
typedef PCSC_LONG (*fnPCSCSCardBeginTransaction)(SCARDHANDLE hCard);
148
typedef PCSC_LONG (*fnPCSCSCardEndTransaction)(SCARDHANDLE hCard, PCSC_DWORD dwDisposition);
149
typedef PCSC_LONG (*fnPCSCSCardStatus)(SCARDHANDLE hCard, LPSTR mszReaderName,
150
                                       PCSC_LPDWORD pcchReaderLen, PCSC_LPDWORD pdwState,
151
                                       PCSC_LPDWORD pdwProtocol, LPBYTE pbAtr,
152
                                       PCSC_LPDWORD pcbAtrLen);
153
typedef PCSC_LONG (*fnPCSCSCardGetStatusChange)(SCARDCONTEXT hContext, PCSC_DWORD dwTimeout,
154
                                                PCSC_SCARD_READERSTATE* rgReaderStates,
155
                                                PCSC_DWORD cReaders);
156
typedef PCSC_LONG (*fnPCSCSCardControl)(SCARDHANDLE hCard, PCSC_DWORD dwControlCode,
157
                                        LPCVOID pbSendBuffer, PCSC_DWORD cbSendLength,
158
                                        LPVOID pbRecvBuffer, PCSC_DWORD cbRecvLength,
159
                                        PCSC_LPDWORD lpBytesReturned);
160
typedef PCSC_LONG (*fnPCSCSCardTransmit)(SCARDHANDLE hCard, const PCSC_SCARD_IO_REQUEST* pioSendPci,
161
                                         LPCBYTE pbSendBuffer, PCSC_DWORD cbSendLength,
162
                                         PCSC_SCARD_IO_REQUEST* pioRecvPci, LPBYTE pbRecvBuffer,
163
                                         PCSC_LPDWORD pcbRecvLength);
164
typedef PCSC_LONG (*fnPCSCSCardListReaderGroups)(SCARDCONTEXT hContext, LPSTR mszGroups,
165
                                                 PCSC_LPDWORD pcchGroups);
166
typedef PCSC_LONG (*fnPCSCSCardListReaders)(SCARDCONTEXT hContext, LPCSTR mszGroups,
167
                                            LPSTR mszReaders, PCSC_LPDWORD pcchReaders);
168
typedef PCSC_LONG (*fnPCSCSCardFreeMemory)(SCARDCONTEXT hContext, LPCVOID pvMem);
169
typedef PCSC_LONG (*fnPCSCSCardCancel)(SCARDCONTEXT hContext);
170
typedef PCSC_LONG (*fnPCSCSCardGetAttrib)(SCARDHANDLE hCard, PCSC_DWORD dwAttrId, LPBYTE pbAttr,
171
                                          PCSC_LPDWORD pcbAttrLen);
172
typedef PCSC_LONG (*fnPCSCSCardSetAttrib)(SCARDHANDLE hCard, PCSC_DWORD dwAttrId, LPCBYTE pbAttr,
173
                                          PCSC_DWORD cbAttrLen);
174
175
typedef struct
176
{
177
  WINPR_ATTR_NODISCARD fnPCSCSCardEstablishContext pfnSCardEstablishContext;
178
  WINPR_ATTR_NODISCARD fnPCSCSCardReleaseContext pfnSCardReleaseContext;
179
  WINPR_ATTR_NODISCARD fnPCSCSCardIsValidContext pfnSCardIsValidContext;
180
  WINPR_ATTR_NODISCARD fnPCSCSCardConnect pfnSCardConnect;
181
  WINPR_ATTR_NODISCARD fnPCSCSCardReconnect pfnSCardReconnect;
182
  WINPR_ATTR_NODISCARD fnPCSCSCardDisconnect pfnSCardDisconnect;
183
  WINPR_ATTR_NODISCARD fnPCSCSCardBeginTransaction pfnSCardBeginTransaction;
184
  WINPR_ATTR_NODISCARD fnPCSCSCardEndTransaction pfnSCardEndTransaction;
185
  WINPR_ATTR_NODISCARD fnPCSCSCardStatus pfnSCardStatus;
186
  WINPR_ATTR_NODISCARD fnPCSCSCardGetStatusChange pfnSCardGetStatusChange;
187
  WINPR_ATTR_NODISCARD fnPCSCSCardControl pfnSCardControl;
188
  WINPR_ATTR_NODISCARD fnPCSCSCardTransmit pfnSCardTransmit;
189
  WINPR_ATTR_NODISCARD fnPCSCSCardListReaderGroups pfnSCardListReaderGroups;
190
  WINPR_ATTR_NODISCARD fnPCSCSCardListReaders pfnSCardListReaders;
191
  fnPCSCSCardFreeMemory pfnSCardFreeMemory;
192
  fnPCSCSCardCancel pfnSCardCancel;
193
  WINPR_ATTR_NODISCARD fnPCSCSCardGetAttrib pfnSCardGetAttrib;
194
  WINPR_ATTR_NODISCARD fnPCSCSCardSetAttrib pfnSCardSetAttrib;
195
} PCSCFunctionTable;
196
197
typedef struct
198
{
199
  DWORD len;
200
  DWORD freshness;
201
  BYTE* data;
202
} PCSC_CACHE_ITEM;
203
204
typedef struct
205
{
206
  SCARDHANDLE owner;
207
  CRITICAL_SECTION lock;
208
  SCARDCONTEXT hContext;
209
  DWORD dwCardHandleCount;
210
  BOOL isTransactionLocked;
211
  wHashTable* cache;
212
} PCSC_SCARDCONTEXT;
213
214
typedef struct
215
{
216
  BOOL shared;
217
  SCARDCONTEXT hSharedContext;
218
} PCSC_SCARDHANDLE;
219
220
static HMODULE g_PCSCModule = nullptr;
221
static PCSCFunctionTable g_PCSC = WINPR_C_ARRAY_INIT;
222
223
static HANDLE g_StartedEvent = nullptr;
224
static int g_StartedEventRefCount = 0;
225
226
static BOOL g_SCardAutoAllocate = FALSE;
227
static BOOL g_PnP_Notification = TRUE;
228
229
#ifdef __MACOSX__
230
static unsigned int OSXVersion = 0;
231
#endif
232
233
static wListDictionary* g_CardHandles = nullptr;
234
static wListDictionary* g_CardContexts = nullptr;
235
static wListDictionary* g_MemoryBlocks = nullptr;
236
static INIT_ONCE g_CardHandleInitializer = INIT_ONCE_STATIC_INIT;
237
238
static const char SMARTCARD_PNP_NOTIFICATION_A[] = "\\\\?PnP?\\Notification";
239
240
static const PCSC_SCARD_IO_REQUEST g_PCSC_rgSCardT0Pci = { SCARD_PROTOCOL_T0,
241
                                                         sizeof(PCSC_SCARD_IO_REQUEST) };
242
static const PCSC_SCARD_IO_REQUEST g_PCSC_rgSCardT1Pci = { SCARD_PROTOCOL_T1,
243
                                                         sizeof(PCSC_SCARD_IO_REQUEST) };
244
static const PCSC_SCARD_IO_REQUEST g_PCSC_rgSCardRawPci = { PCSC_SCARD_PROTOCOL_RAW,
245
                                                          sizeof(PCSC_SCARD_IO_REQUEST) };
246
247
static void clearHandles(void)
248
0
{
249
0
  ListDictionary_Free(g_CardHandles);
250
0
  ListDictionary_Free(g_CardContexts);
251
0
  ListDictionary_Free(g_MemoryBlocks);
252
253
0
  g_CardHandles = nullptr;
254
0
  g_CardContexts = nullptr;
255
0
  g_MemoryBlocks = nullptr;
256
0
}
257
258
WINPR_ATTR_NODISCARD
259
static wListDictionary* setupWithValueObjectFree(BOOL key)
260
0
{
261
0
  wListDictionary* list = ListDictionary_New(TRUE);
262
0
  if (!list)
263
0
    return nullptr;
264
265
0
  {
266
0
    wObject* obj = ListDictionary_ValueObject(list);
267
0
    if (key)
268
0
      obj = ListDictionary_KeyObject(list);
269
0
    WINPR_ASSERT(obj);
270
0
    obj->fnObjectFree = free;
271
0
  }
272
0
  return list;
273
0
}
274
275
static void cardContextFree(PCSC_SCARDCONTEXT* pContext)
276
0
{
277
0
  if (!pContext)
278
0
    return;
279
280
0
  DeleteCriticalSection(&(pContext->lock));
281
0
  HashTable_Free(pContext->cache);
282
0
  free(pContext);
283
0
}
284
285
static void cardContextFreeVoid(void* obj)
286
0
{
287
0
  PCSC_SCARDCONTEXT* pContext = obj;
288
0
  cardContextFree(pContext);
289
0
}
290
291
WINPR_ATTR_NODISCARD
292
static BOOL initializeHandles(WINPR_ATTR_UNUSED PINIT_ONCE InitOnce,
293
                              WINPR_ATTR_UNUSED PVOID Parameter, WINPR_ATTR_UNUSED PVOID* Context)
294
0
{
295
0
  (void)atexit(clearHandles);
296
0
  g_CardHandles = setupWithValueObjectFree(FALSE);
297
0
  if (!g_CardHandles)
298
0
    return FALSE;
299
0
  g_CardContexts = ListDictionary_New(TRUE);
300
0
  if (!g_CardContexts)
301
0
    return FALSE;
302
0
  {
303
0
    wObject* obj = ListDictionary_ValueObject(g_CardContexts);
304
0
    if (!obj)
305
0
      return FALSE;
306
0
    obj->fnObjectFree = cardContextFreeVoid;
307
0
  }
308
309
0
  g_MemoryBlocks = setupWithValueObjectFree(TRUE);
310
0
  return g_MemoryBlocks != nullptr;
311
0
}
312
313
WINPR_ATTR_NODISCARD
314
static BOOL init(void)
315
0
{
316
0
  return InitOnceExecuteOnce(&g_CardHandleInitializer, initializeHandles, nullptr, nullptr);
317
0
}
318
319
static LONG WINAPI PCSC_SCardFreeMemory_Internal(SCARDCONTEXT hContext, LPVOID pvMem);
320
321
WINPR_ATTR_NODISCARD
322
static LONG WINAPI PCSC_SCardEstablishContext_Internal(DWORD dwScope, LPCVOID pvReserved1,
323
                                                       LPCVOID pvReserved2,
324
                                                       LPSCARDCONTEXT phContext);
325
326
WINPR_ATTR_NODISCARD
327
static LONG WINAPI PCSC_SCardReleaseContext_Internal(SCARDCONTEXT hContext);
328
329
WINPR_ATTR_NODISCARD
330
static LONG PCSC_SCard_LogError(const char* what)
331
0
{
332
0
  WLog_WARN(TAG, "Missing function pointer %s=nullptr", what);
333
0
  return SCARD_E_UNSUPPORTED_FEATURE;
334
0
}
335
336
WINPR_ATTR_NODISCARD
337
static LONG PCSC_MapErrorCodeToWinSCard(PCSC_LONG errorCode)
338
0
{
339
  /**
340
   * pcsc-lite returns SCARD_E_UNEXPECTED when it
341
   * should return SCARD_E_UNSUPPORTED_FEATURE.
342
   *
343
   * Additionally, the pcsc-lite headers incorrectly
344
   * define SCARD_E_UNSUPPORTED_FEATURE to 0x8010001F,
345
   * when the real value should be 0x80100022.
346
   */
347
0
  if (errorCode != SCARD_S_SUCCESS)
348
0
  {
349
0
    if (errorCode == SCARD_E_UNEXPECTED)
350
0
      errorCode = SCARD_E_UNSUPPORTED_FEATURE;
351
0
  }
352
353
0
  return (LONG)errorCode;
354
0
}
355
356
WINPR_ATTR_NODISCARD
357
static DWORD PCSC_ConvertCardStateToWinSCard(DWORD dwCardState, PCSC_LONG status)
358
0
{
359
  /**
360
   * pcsc-lite's SCardStatus returns a bit-field, not an enumerated value.
361
   *
362
   *         State             WinSCard           pcsc-lite
363
   *
364
   *      SCARD_UNKNOWN           0                0x0001
365
   *      SCARD_ABSENT            1                0x0002
366
   *      SCARD_PRESENT           2                0x0004
367
   *      SCARD_SWALLOWED         3                0x0008
368
   *      SCARD_POWERED           4                0x0010
369
   *      SCARD_NEGOTIABLE        5                0x0020
370
   *      SCARD_SPECIFIC          6                0x0040
371
   *
372
   * pcsc-lite also never sets SCARD_SPECIFIC,
373
   * which is expected by some windows applications.
374
   */
375
0
  if (status == SCARD_S_SUCCESS)
376
0
  {
377
0
    if ((dwCardState & PCSC_SCARD_NEGOTIABLE) || (dwCardState & PCSC_SCARD_SPECIFIC))
378
0
      return SCARD_SPECIFIC;
379
0
  }
380
381
0
  if (dwCardState & PCSC_SCARD_POWERED)
382
0
    return SCARD_POWERED;
383
384
0
  if (dwCardState & PCSC_SCARD_NEGOTIABLE)
385
0
    return SCARD_NEGOTIABLE;
386
387
0
  if (dwCardState & PCSC_SCARD_SPECIFIC)
388
0
    return SCARD_SPECIFIC;
389
390
0
  if (dwCardState & PCSC_SCARD_ABSENT)
391
0
    return SCARD_ABSENT;
392
393
0
  if (dwCardState & PCSC_SCARD_PRESENT)
394
0
    return SCARD_PRESENT;
395
396
0
  if (dwCardState & PCSC_SCARD_SWALLOWED)
397
0
    return SCARD_SWALLOWED;
398
399
0
  if (dwCardState & PCSC_SCARD_UNKNOWN)
400
0
    return SCARD_UNKNOWN;
401
402
0
  return SCARD_UNKNOWN;
403
0
}
404
405
WINPR_ATTR_NODISCARD
406
static DWORD PCSC_ConvertProtocolsToWinSCard(PCSC_DWORD dwProtocols)
407
0
{
408
  /**
409
   * pcsc-lite uses a different value for SCARD_PROTOCOL_RAW,
410
   * and also has SCARD_PROTOCOL_T15 which is not in WinSCard.
411
   */
412
0
  if (dwProtocols & PCSC_SCARD_PROTOCOL_RAW)
413
0
  {
414
0
    dwProtocols &= ~PCSC_SCARD_PROTOCOL_RAW;
415
0
    dwProtocols |= SCARD_PROTOCOL_RAW;
416
0
  }
417
418
0
  if (dwProtocols & PCSC_SCARD_PROTOCOL_T15)
419
0
  {
420
0
    dwProtocols &= ~PCSC_SCARD_PROTOCOL_T15;
421
0
  }
422
423
0
  return (DWORD)dwProtocols;
424
0
}
425
426
WINPR_ATTR_NODISCARD
427
static DWORD PCSC_ConvertProtocolsFromWinSCard(DWORD dwProtocols)
428
0
{
429
  /**
430
   * pcsc-lite uses a different value for SCARD_PROTOCOL_RAW,
431
   * and it does not define WinSCard's SCARD_PROTOCOL_DEFAULT.
432
   */
433
0
  if (dwProtocols & SCARD_PROTOCOL_RAW)
434
0
  {
435
0
    dwProtocols &= ~SCARD_PROTOCOL_RAW;
436
0
    dwProtocols |= PCSC_SCARD_PROTOCOL_RAW;
437
0
  }
438
439
0
  if (dwProtocols & SCARD_PROTOCOL_DEFAULT)
440
0
  {
441
0
    dwProtocols &= ~SCARD_PROTOCOL_DEFAULT;
442
0
  }
443
444
0
  if (dwProtocols == SCARD_PROTOCOL_UNDEFINED)
445
0
  {
446
0
    dwProtocols = SCARD_PROTOCOL_Tx;
447
0
  }
448
449
0
  return dwProtocols;
450
0
}
451
452
WINPR_ATTR_NODISCARD
453
static PCSC_SCARDCONTEXT* PCSC_GetCardContextData(SCARDCONTEXT hContext)
454
0
{
455
0
  if (!init())
456
0
    return nullptr;
457
458
0
  PCSC_SCARDCONTEXT* pContext =
459
0
      (PCSC_SCARDCONTEXT*)ListDictionary_GetItemValue(g_CardContexts, (void*)hContext);
460
461
0
  if (!pContext)
462
0
    return nullptr;
463
464
0
  return pContext;
465
0
}
466
467
static void pcsc_cache_item_free(void* ptr)
468
0
{
469
0
  PCSC_CACHE_ITEM* data = ptr;
470
0
  if (data)
471
0
    free(data->data);
472
0
  free(data);
473
0
}
474
475
static PCSC_SCARDCONTEXT* PCSC_EstablishCardContext(SCARDCONTEXT hContext)
476
0
{
477
0
  if (!init())
478
0
    return nullptr;
479
480
0
  PCSC_SCARDCONTEXT* pContext = (PCSC_SCARDCONTEXT*)calloc(1, sizeof(PCSC_SCARDCONTEXT));
481
482
0
  if (!pContext)
483
0
    return nullptr;
484
485
0
  pContext->hContext = hContext;
486
487
0
  if (!InitializeCriticalSectionAndSpinCount(&(pContext->lock), 4000))
488
0
    goto errors;
489
490
0
  pContext->cache = HashTable_New(FALSE);
491
0
  if (!pContext->cache)
492
0
    goto errors;
493
0
  if (!HashTable_SetupForStringData(pContext->cache, FALSE))
494
0
    goto errors;
495
0
  {
496
0
    wObject* obj = HashTable_ValueObject(pContext->cache);
497
0
    obj->fnObjectFree = pcsc_cache_item_free;
498
0
  }
499
500
0
  if (!ListDictionary_Add(g_CardContexts, (void*)hContext, (void*)pContext))
501
0
    goto errors;
502
503
0
  return pContext;
504
0
errors:
505
0
  cardContextFree(pContext);
506
0
  return nullptr;
507
0
}
508
509
static void PCSC_ReleaseCardContext(SCARDCONTEXT hContext)
510
0
{
511
0
  if (init())
512
0
  {
513
0
    PCSC_SCARDCONTEXT* pContext = ListDictionary_Take(g_CardContexts, (void*)hContext);
514
0
    cardContextFree(pContext);
515
0
  }
516
0
}
517
518
WINPR_ATTR_NODISCARD
519
static BOOL PCSC_LockCardContext(SCARDCONTEXT hContext)
520
0
{
521
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
522
0
  pContext = PCSC_GetCardContextData(hContext);
523
524
0
  if (!pContext)
525
0
  {
526
0
    WLog_ERR(TAG, "PCSC_LockCardContext: invalid context (%p)", (void*)hContext);
527
0
    return FALSE;
528
0
  }
529
530
0
  EnterCriticalSection(&(pContext->lock));
531
0
  return TRUE;
532
0
}
533
534
WINPR_ATTR_NODISCARD
535
static BOOL PCSC_UnlockCardContext(SCARDCONTEXT hContext)
536
0
{
537
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
538
0
  pContext = PCSC_GetCardContextData(hContext);
539
540
0
  if (!pContext)
541
0
  {
542
0
    WLog_ERR(TAG, "PCSC_UnlockCardContext: invalid context (%p)", (void*)hContext);
543
0
    return FALSE;
544
0
  }
545
546
0
  LeaveCriticalSection(&(pContext->lock));
547
0
  return TRUE;
548
0
}
549
550
WINPR_ATTR_NODISCARD
551
static PCSC_SCARDHANDLE* PCSC_GetCardHandleData(SCARDHANDLE hCard)
552
0
{
553
0
  if (!init())
554
0
    return nullptr;
555
556
0
  PCSC_SCARDHANDLE* pCard =
557
0
      (PCSC_SCARDHANDLE*)ListDictionary_GetItemValue(g_CardHandles, (void*)hCard);
558
559
0
  if (!pCard)
560
0
    return nullptr;
561
562
0
  return pCard;
563
0
}
564
565
WINPR_ATTR_NODISCARD
566
static SCARDCONTEXT PCSC_GetCardContextFromHandle(SCARDHANDLE hCard)
567
0
{
568
0
  PCSC_SCARDHANDLE* pCard = nullptr;
569
0
  pCard = PCSC_GetCardHandleData(hCard);
570
571
0
  if (!pCard)
572
0
    return 0;
573
574
0
  return pCard->hSharedContext;
575
0
}
576
577
static BOOL PCSC_WaitForCardAccess(SCARDCONTEXT hContext, SCARDHANDLE hCard, BOOL shared)
578
0
{
579
0
  BOOL status = TRUE;
580
0
  PCSC_SCARDHANDLE* pCard = nullptr;
581
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
582
583
0
  if (!hCard)
584
0
  {
585
    /* SCardConnect */
586
0
    pContext = PCSC_GetCardContextData(hContext);
587
588
0
    if (!pContext)
589
0
      return FALSE;
590
591
0
    if (!pContext->owner)
592
0
      return TRUE;
593
594
    /* wait for card ownership */
595
0
    return TRUE;
596
0
  }
597
598
0
  pCard = PCSC_GetCardHandleData(hCard);
599
600
0
  if (!pCard)
601
0
    return FALSE;
602
603
0
  shared = pCard->shared;
604
0
  hContext = pCard->hSharedContext;
605
0
  pContext = PCSC_GetCardContextData(hContext);
606
607
0
  if (!pContext)
608
0
    return FALSE;
609
610
0
  if (!pContext->owner)
611
0
  {
612
    /* card is not owned */
613
0
    if (!shared)
614
0
      pContext->owner = hCard;
615
616
0
    return TRUE;
617
0
  }
618
619
0
  if (pContext->owner == hCard)
620
0
  {
621
    /* already card owner */
622
0
  }
623
0
  else
624
0
  {
625
    /* wait for card ownership */
626
0
  }
627
628
0
  return status;
629
0
}
630
631
static BOOL PCSC_ReleaseCardAccess(SCARDCONTEXT hContext, SCARDHANDLE hCard)
632
0
{
633
0
  PCSC_SCARDHANDLE* pCard = nullptr;
634
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
635
636
0
  if (!hCard)
637
0
  {
638
    /* release current owner */
639
0
    pContext = PCSC_GetCardContextData(hContext);
640
641
0
    if (!pContext)
642
0
      return FALSE;
643
644
0
    hCard = pContext->owner;
645
646
0
    if (!hCard)
647
0
      return TRUE;
648
649
0
    pCard = PCSC_GetCardHandleData(hCard);
650
651
0
    if (!pCard)
652
0
      return FALSE;
653
654
    /* release card ownership */
655
0
    pContext->owner = 0;
656
0
    return TRUE;
657
0
  }
658
659
0
  pCard = PCSC_GetCardHandleData(hCard);
660
661
0
  if (!pCard)
662
0
    return FALSE;
663
664
0
  hContext = pCard->hSharedContext;
665
0
  pContext = PCSC_GetCardContextData(hContext);
666
667
0
  if (!pContext)
668
0
    return FALSE;
669
670
0
  if (pContext->owner == hCard)
671
0
  {
672
    /* release card ownership */
673
0
    pContext->owner = 0;
674
0
  }
675
676
0
  return TRUE;
677
0
}
678
679
static void PCSC_DisconnectCardHandle(PCSC_SCARDHANDLE* pCard)
680
0
{
681
0
  if (!pCard)
682
0
    return;
683
684
0
  PCSC_SCARDCONTEXT* pContext = PCSC_GetCardContextData(pCard->hSharedContext);
685
0
  if (init())
686
0
    ListDictionary_Remove(g_CardHandles, (void*)pCard);
687
688
0
  if (!pContext)
689
0
  {
690
0
    WLog_ERR(TAG, "PCSC_DisconnectCardHandle: null pContext!");
691
0
    return;
692
0
  }
693
694
0
  pContext->dwCardHandleCount--;
695
0
}
696
697
WINPR_ATTR_MALLOC(PCSC_DisconnectCardHandle, 1)
698
static PCSC_SCARDHANDLE* PCSC_ConnectCardHandle(SCARDCONTEXT hSharedContext, SCARDHANDLE hCard)
699
0
{
700
0
  if (!init())
701
0
    return nullptr;
702
703
0
  PCSC_SCARDCONTEXT* pContext = PCSC_GetCardContextData(hSharedContext);
704
705
0
  if (!pContext)
706
0
  {
707
0
    WLog_ERR(TAG, "PCSC_ConnectCardHandle: null pContext!");
708
0
    return nullptr;
709
0
  }
710
711
0
  PCSC_SCARDHANDLE* pCard = (PCSC_SCARDHANDLE*)calloc(1, sizeof(PCSC_SCARDHANDLE));
712
713
0
  if (!pCard)
714
0
    return nullptr;
715
716
0
  pCard->hSharedContext = hSharedContext;
717
718
0
  if (!ListDictionary_Add(g_CardHandles, (void*)hCard, (void*)pCard))
719
0
    goto error;
720
721
0
  pContext->dwCardHandleCount++;
722
0
  return pCard;
723
0
error:
724
0
  free(pCard);
725
0
  return nullptr;
726
0
}
727
728
static BOOL PCSC_AddMemoryBlock(SCARDCONTEXT hContext, void* pvMem)
729
0
{
730
0
  if (!init())
731
0
    return FALSE;
732
0
  return ListDictionary_Add(g_MemoryBlocks, pvMem, (void*)hContext);
733
0
}
734
735
WINPR_ATTR_NODISCARD
736
static void* PCSC_RemoveMemoryBlock(SCARDCONTEXT hContext, void* pvMem)
737
0
{
738
0
  WINPR_UNUSED(hContext);
739
740
0
  if (!init())
741
0
    return nullptr;
742
743
0
  return ListDictionary_Take(g_MemoryBlocks, pvMem);
744
0
}
745
746
/**
747
 * Standard Windows Smart Card API (PCSC)
748
 */
749
WINPR_ATTR_NODISCARD
750
static LONG WINAPI PCSC_SCardEstablishContext_Internal(DWORD dwScope, LPCVOID pvReserved1,
751
                                                       LPCVOID pvReserved2,
752
                                                       LPSCARDCONTEXT phContext)
753
0
{
754
0
  WINPR_UNUSED(dwScope); /* SCARD_SCOPE_SYSTEM is the only scope supported by pcsc-lite */
755
0
  PCSC_LONG status = SCARD_S_SUCCESS;
756
757
0
  if (!g_PCSC.pfnSCardEstablishContext)
758
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardEstablishContext");
759
760
0
  status =
761
0
      g_PCSC.pfnSCardEstablishContext(SCARD_SCOPE_SYSTEM, pvReserved1, pvReserved2, phContext);
762
0
  return PCSC_MapErrorCodeToWinSCard(status);
763
0
}
764
765
WINPR_ATTR_NODISCARD
766
static LONG WINAPI PCSC_SCardEstablishContext(DWORD dwScope, LPCVOID pvReserved1,
767
                                              LPCVOID pvReserved2, LPSCARDCONTEXT phContext)
768
0
{
769
0
  LONG status = 0;
770
771
0
  status = PCSC_SCardEstablishContext_Internal(dwScope, pvReserved1, pvReserved2, phContext);
772
773
0
  if (status == SCARD_S_SUCCESS)
774
0
    PCSC_EstablishCardContext(*phContext);
775
776
0
  return status;
777
0
}
778
779
WINPR_ATTR_NODISCARD
780
static LONG WINAPI PCSC_SCardReleaseContext_Internal(SCARDCONTEXT hContext)
781
0
{
782
0
  PCSC_LONG status = SCARD_S_SUCCESS;
783
784
0
  if (!g_PCSC.pfnSCardReleaseContext)
785
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardReleaseContext");
786
787
0
  if (!hContext)
788
0
  {
789
0
    WLog_ERR(TAG, "SCardReleaseContext: null hContext");
790
0
    return PCSC_MapErrorCodeToWinSCard(SCARD_E_INVALID_HANDLE);
791
0
  }
792
793
0
  status = g_PCSC.pfnSCardReleaseContext(hContext);
794
0
  return PCSC_MapErrorCodeToWinSCard(status);
795
0
}
796
797
WINPR_ATTR_NODISCARD
798
static LONG WINAPI PCSC_SCardReleaseContext(SCARDCONTEXT hContext)
799
0
{
800
0
  LONG status = SCARD_S_SUCCESS;
801
802
0
  status = PCSC_SCardReleaseContext_Internal(hContext);
803
0
  PCSC_ReleaseCardContext(hContext);
804
805
0
  return status;
806
0
}
807
808
WINPR_ATTR_NODISCARD
809
static LONG WINAPI PCSC_SCardIsValidContext(SCARDCONTEXT hContext)
810
0
{
811
0
  PCSC_LONG status = SCARD_S_SUCCESS;
812
813
0
  if (!g_PCSC.pfnSCardIsValidContext)
814
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardIsValidContext");
815
816
0
  status = g_PCSC.pfnSCardIsValidContext(hContext);
817
0
  return PCSC_MapErrorCodeToWinSCard(status);
818
0
}
819
820
WINPR_ATTR_NODISCARD
821
static LONG WINAPI PCSC_SCardListReaderGroups_Internal(SCARDCONTEXT hContext, LPSTR mszGroups,
822
                                                       LPDWORD pcchGroups)
823
0
{
824
0
  PCSC_LONG status = SCARD_S_SUCCESS;
825
0
  BOOL pcchGroupsAlloc = FALSE;
826
0
  PCSC_DWORD pcsc_cchGroups = 0;
827
828
0
  if (!pcchGroups)
829
0
    return SCARD_E_INVALID_PARAMETER;
830
831
0
  if (!g_PCSC.pfnSCardListReaderGroups)
832
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaderGroups");
833
834
0
  if (*pcchGroups == SCARD_AUTOALLOCATE)
835
0
    pcchGroupsAlloc = TRUE;
836
837
0
  pcsc_cchGroups = pcchGroupsAlloc ? PCSC_SCARD_AUTOALLOCATE : (PCSC_DWORD)*pcchGroups;
838
839
0
  if (pcchGroupsAlloc && !g_SCardAutoAllocate)
840
0
  {
841
0
    pcsc_cchGroups = 0;
842
0
    status = g_PCSC.pfnSCardListReaderGroups(hContext, nullptr, &pcsc_cchGroups);
843
844
0
    if (status == SCARD_S_SUCCESS)
845
0
    {
846
0
      LPSTR tmp = calloc(1, pcsc_cchGroups);
847
848
0
      if (!tmp)
849
0
        return SCARD_E_NO_MEMORY;
850
851
0
      status = g_PCSC.pfnSCardListReaderGroups(hContext, tmp, &pcsc_cchGroups);
852
853
0
      if (status != SCARD_S_SUCCESS)
854
0
      {
855
0
        free(tmp);
856
0
        tmp = nullptr;
857
0
      }
858
0
      else
859
0
        PCSC_AddMemoryBlock(hContext, tmp);
860
861
0
      *(LPSTR*)mszGroups = tmp;
862
0
    }
863
0
  }
864
0
  else
865
0
  {
866
0
    status = g_PCSC.pfnSCardListReaderGroups(hContext, mszGroups, &pcsc_cchGroups);
867
0
  }
868
869
0
  *pcchGroups = (DWORD)pcsc_cchGroups;
870
0
  return PCSC_MapErrorCodeToWinSCard(status);
871
0
}
872
873
WINPR_ATTR_NODISCARD
874
static LONG WINAPI PCSC_SCardListReaderGroupsA(SCARDCONTEXT hContext, LPSTR mszGroups,
875
                                               LPDWORD pcchGroups)
876
0
{
877
0
  LONG status = SCARD_S_SUCCESS;
878
879
0
  if (!g_PCSC.pfnSCardListReaderGroups)
880
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaderGroups");
881
882
0
  if (!PCSC_LockCardContext(hContext))
883
0
    return SCARD_E_INVALID_HANDLE;
884
885
0
  status = PCSC_SCardListReaderGroups_Internal(hContext, mszGroups, pcchGroups);
886
887
0
  if (!PCSC_UnlockCardContext(hContext))
888
0
    return SCARD_E_INVALID_HANDLE;
889
890
0
  return status;
891
0
}
892
893
WINPR_ATTR_NODISCARD
894
static LONG WINAPI PCSC_SCardListReaderGroupsW(SCARDCONTEXT hContext, LPWSTR mszGroups,
895
                                               LPDWORD pcchGroups)
896
0
{
897
0
  LPSTR mszGroupsA = nullptr;
898
0
  LPSTR* pMszGroupsA = &mszGroupsA;
899
0
  LONG status = SCARD_S_SUCCESS;
900
901
0
  if (!g_PCSC.pfnSCardListReaderGroups)
902
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaderGroups");
903
904
0
  if (!PCSC_LockCardContext(hContext))
905
0
    return SCARD_E_INVALID_HANDLE;
906
907
0
  status = PCSC_SCardListReaderGroups_Internal(hContext, (LPSTR)&mszGroupsA, pcchGroups);
908
909
0
  if (status == SCARD_S_SUCCESS)
910
0
  {
911
0
    size_t size = 0;
912
0
    WCHAR* str = ConvertMszUtf8NToWCharAlloc(*pMszGroupsA, *pcchGroups, &size);
913
0
    if (!str)
914
0
    {
915
0
      status = SCARD_E_NO_MEMORY;
916
0
      goto fail;
917
0
    }
918
0
    *(WCHAR**)mszGroups = str;
919
0
    *pcchGroups = (DWORD)size;
920
0
    PCSC_AddMemoryBlock(hContext, str);
921
0
    PCSC_SCardFreeMemory_Internal(hContext, *pMszGroupsA);
922
0
  }
923
924
0
fail:
925
0
  if (!PCSC_UnlockCardContext(hContext))
926
0
    return SCARD_E_INVALID_HANDLE;
927
928
0
  return status;
929
0
}
930
931
WINPR_ATTR_NODISCARD
932
static LONG WINAPI PCSC_SCardListReaders_Internal(SCARDCONTEXT hContext, LPCSTR mszGroups,
933
                                                  LPSTR mszReaders, LPDWORD pcchReaders)
934
0
{
935
0
  PCSC_LONG status = SCARD_S_SUCCESS;
936
0
  BOOL pcchReadersAlloc = FALSE;
937
0
  PCSC_DWORD pcsc_cchReaders = 0;
938
0
  if (!pcchReaders)
939
0
    return SCARD_E_INVALID_PARAMETER;
940
941
0
  if (!g_PCSC.pfnSCardListReaders)
942
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaders");
943
944
0
  mszGroups = nullptr; /* mszGroups is not supported by pcsc-lite */
945
946
0
  if (*pcchReaders == SCARD_AUTOALLOCATE)
947
0
    pcchReadersAlloc = TRUE;
948
949
0
  pcsc_cchReaders = pcchReadersAlloc ? PCSC_SCARD_AUTOALLOCATE : (PCSC_DWORD)*pcchReaders;
950
951
0
  if (pcchReadersAlloc && !g_SCardAutoAllocate)
952
0
  {
953
0
    pcsc_cchReaders = 0;
954
0
    status = g_PCSC.pfnSCardListReaders(hContext, mszGroups, nullptr, &pcsc_cchReaders);
955
956
0
    if (status == SCARD_S_SUCCESS)
957
0
    {
958
0
      char* tmp = calloc(1, pcsc_cchReaders);
959
960
0
      if (!tmp)
961
0
        return SCARD_E_NO_MEMORY;
962
963
0
      status = g_PCSC.pfnSCardListReaders(hContext, mszGroups, tmp, &pcsc_cchReaders);
964
965
0
      if (status != SCARD_S_SUCCESS)
966
0
      {
967
0
        free(tmp);
968
0
        tmp = nullptr;
969
0
      }
970
0
      else
971
0
        PCSC_AddMemoryBlock(hContext, tmp);
972
973
0
      *(char**)mszReaders = tmp;
974
0
    }
975
0
  }
976
0
  else
977
0
  {
978
0
    status = g_PCSC.pfnSCardListReaders(hContext, mszGroups, mszReaders, &pcsc_cchReaders);
979
0
  }
980
981
0
  *pcchReaders = (DWORD)pcsc_cchReaders;
982
0
  return PCSC_MapErrorCodeToWinSCard(status);
983
0
}
984
985
WINPR_ATTR_NODISCARD
986
static LONG WINAPI PCSC_SCardListReadersA(SCARDCONTEXT hContext, LPCSTR mszGroups, LPSTR mszReaders,
987
                                          LPDWORD pcchReaders)
988
0
{
989
0
  LONG status = SCARD_S_SUCCESS;
990
0
  BOOL nullCardContext = FALSE;
991
992
0
  if (!g_PCSC.pfnSCardListReaders)
993
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaders");
994
995
0
  if (!hContext)
996
0
  {
997
0
    status = PCSC_SCardEstablishContext(SCARD_SCOPE_SYSTEM, nullptr, nullptr, &hContext);
998
999
0
    if (status != SCARD_S_SUCCESS)
1000
1001
0
      return status;
1002
1003
0
    nullCardContext = TRUE;
1004
0
  }
1005
1006
0
  if (!PCSC_LockCardContext(hContext))
1007
0
  {
1008
0
    status = SCARD_E_INVALID_HANDLE;
1009
0
    goto release;
1010
0
  }
1011
1012
0
  status = PCSC_SCardListReaders_Internal(hContext, mszGroups, mszReaders, pcchReaders);
1013
1014
0
  if (!PCSC_UnlockCardContext(hContext))
1015
0
    status = SCARD_E_INVALID_HANDLE;
1016
1017
0
release:
1018
0
  if (nullCardContext)
1019
0
  {
1020
0
    const LONG rc = PCSC_SCardReleaseContext(hContext);
1021
0
    if (rc != SCARD_S_SUCCESS)
1022
0
      status = rc;
1023
0
  }
1024
1025
0
  return status;
1026
0
}
1027
1028
WINPR_ATTR_NODISCARD
1029
static LONG WINAPI PCSC_SCardListReadersW(SCARDCONTEXT hContext, LPCWSTR mszGroups,
1030
                                          LPWSTR mszReaders, LPDWORD pcchReaders)
1031
0
{
1032
0
  LPSTR mszGroupsA = nullptr;
1033
0
  LPSTR mszReadersA = nullptr;
1034
0
  LONG status = SCARD_S_SUCCESS;
1035
0
  BOOL nullCardContext = FALSE;
1036
1037
0
  if (!g_PCSC.pfnSCardListReaders)
1038
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardListReaders");
1039
1040
0
  if (!hContext)
1041
0
  {
1042
0
    status = PCSC_SCardEstablishContext(SCARD_SCOPE_SYSTEM, nullptr, nullptr, &hContext);
1043
1044
0
    if (status != SCARD_S_SUCCESS)
1045
1046
0
      return status;
1047
1048
0
    nullCardContext = TRUE;
1049
0
  }
1050
1051
0
  if (!PCSC_LockCardContext(hContext))
1052
0
  {
1053
0
    status = SCARD_E_INVALID_HANDLE;
1054
0
    goto release;
1055
0
  }
1056
1057
0
  if (mszGroups)
1058
0
  {
1059
0
    mszGroupsA = ConvertWCharToUtf8Alloc(mszGroups, nullptr);
1060
0
    if (!mszGroupsA)
1061
0
    {
1062
0
      status = SCARD_E_NO_MEMORY;
1063
0
      goto fail;
1064
0
    }
1065
0
  }
1066
1067
0
  union
1068
0
  {
1069
0
    LPSTR* ppc;
1070
0
    LPSTR pc;
1071
0
  } cnv;
1072
0
  cnv.ppc = &mszReadersA;
1073
1074
0
  status = PCSC_SCardListReaders_Internal(hContext, mszGroupsA, cnv.pc, pcchReaders);
1075
0
  free(mszGroupsA);
1076
0
  if (status == SCARD_S_SUCCESS)
1077
0
  {
1078
0
    size_t size = 0;
1079
0
    WCHAR* str = ConvertMszUtf8NToWCharAlloc(mszReadersA, *pcchReaders, &size);
1080
0
    PCSC_SCardFreeMemory_Internal(hContext, mszReadersA);
1081
0
    if (!str || (size > UINT32_MAX))
1082
0
    {
1083
0
      status = SCARD_E_NO_MEMORY;
1084
0
      free(str);
1085
0
      goto fail;
1086
0
    }
1087
1088
0
    *(LPWSTR*)mszReaders = str;
1089
0
    *pcchReaders = (DWORD)size;
1090
0
    PCSC_AddMemoryBlock(hContext, str);
1091
0
  }
1092
1093
0
fail:
1094
0
  if (!PCSC_UnlockCardContext(hContext))
1095
0
    status = SCARD_E_INVALID_HANDLE;
1096
1097
0
release:
1098
0
  if (nullCardContext)
1099
0
  {
1100
0
    const LONG rc = PCSC_SCardReleaseContext(hContext);
1101
0
    if (rc != SCARD_S_SUCCESS)
1102
0
      status = rc;
1103
0
  }
1104
1105
0
  return status;
1106
0
}
1107
1108
typedef struct
1109
{
1110
  BYTE atr[64];
1111
  size_t atrLen;
1112
  const char* cardName;
1113
} PcscKnownAtr;
1114
1115
static PcscKnownAtr knownAtrs[] = {
1116
  /* Yubico YubiKey 5 NFC (PKI) */
1117
  { { 0x3B, 0xFD, 0x13, 0x00, 0x00, 0x81, 0x31, 0xFE, 0x15, 0x80, 0x73, 0xC0,
1118
      0x21, 0xC0, 0x57, 0x59, 0x75, 0x62, 0x69, 0x4B, 0x65, 0x79, 0x40 },
1119
    23,
1120
    "NIST SP 800-73 [PIV]" },
1121
  /* PIVKey C910 PKI Smart Card (eID) */
1122
  { { 0x3B, 0xFC, 0x18, 0x00, 0x00, 0x81, 0x31, 0x80, 0x45, 0x90, 0x67,
1123
      0x46, 0x4A, 0x00, 0x64, 0x16, 0x06, 0xF2, 0x72, 0x7E, 0x00, 0xE0 },
1124
    22,
1125
    "PIVKey Feitian (E0)" }
1126
};
1127
1128
#ifndef ARRAY_LENGTH
1129
0
#define ARRAY_LENGTH(a) (sizeof(a) / sizeof(a)[0])
1130
#endif
1131
1132
WINPR_ATTR_NODISCARD
1133
static const char* findCardByAtr(LPCBYTE pbAtr)
1134
0
{
1135
0
  for (size_t i = 0; i < ARRAY_LENGTH(knownAtrs); i++)
1136
0
  {
1137
0
    if (memcmp(knownAtrs[i].atr, pbAtr, knownAtrs[i].atrLen) == 0)
1138
0
      return knownAtrs[i].cardName;
1139
0
  }
1140
1141
0
  return nullptr;
1142
0
}
1143
1144
WINPR_ATTR_NODISCARD
1145
static LONG WINAPI PCSC_SCardListCardsA(WINPR_ATTR_UNUSED SCARDCONTEXT hContext, LPCBYTE pbAtr,
1146
                                        LPCGUID rgquidInterfaces, DWORD cguidInterfaceCount,
1147
                                        CHAR* mszCards, LPDWORD pcchCards)
1148
0
{
1149
0
  const char* cardName = nullptr;
1150
0
  DWORD outputLen = 1;
1151
0
  CHAR* output = nullptr;
1152
0
  BOOL autoAllocate = 0;
1153
1154
0
  if (!pbAtr || rgquidInterfaces || cguidInterfaceCount)
1155
0
    return SCARD_E_UNSUPPORTED_FEATURE;
1156
1157
0
  if (!pcchCards)
1158
0
    return SCARD_E_INVALID_PARAMETER;
1159
1160
0
  autoAllocate = (*pcchCards == SCARD_AUTOALLOCATE);
1161
1162
0
  cardName = findCardByAtr(pbAtr);
1163
0
  if (cardName)
1164
0
    outputLen += strlen(cardName) + 1;
1165
1166
0
  *pcchCards = outputLen;
1167
0
  if (autoAllocate)
1168
0
  {
1169
0
    output = malloc(outputLen);
1170
0
    if (!output)
1171
0
      return SCARD_E_NO_MEMORY;
1172
1173
0
    *((LPSTR*)mszCards) = output;
1174
0
  }
1175
0
  else
1176
0
  {
1177
0
    if (!mszCards)
1178
0
      return SCARD_S_SUCCESS;
1179
1180
0
    if (*pcchCards < outputLen)
1181
0
      return SCARD_E_INSUFFICIENT_BUFFER;
1182
1183
0
    output = mszCards;
1184
0
  }
1185
1186
0
  if (cardName)
1187
0
  {
1188
0
    size_t toCopy = strlen(cardName) + 1;
1189
0
    memcpy(output, cardName, toCopy);
1190
0
    output += toCopy;
1191
0
  }
1192
1193
0
  *output = '\0';
1194
1195
0
  return SCARD_S_SUCCESS;
1196
0
}
1197
1198
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardListCardsW(
1199
    WINPR_ATTR_UNUSED SCARDCONTEXT hContext, LPCBYTE pbAtr, LPCGUID rgquidInterfaces,
1200
    DWORD cguidInterfaceCount, WCHAR* mszCards, LPDWORD pcchCards)
1201
0
{
1202
0
  const char* cardName = nullptr;
1203
0
  DWORD outputLen = 1;
1204
0
  WCHAR* output = nullptr;
1205
0
  BOOL autoAllocate = 0;
1206
1207
0
  if (!pbAtr || rgquidInterfaces || cguidInterfaceCount)
1208
0
    return SCARD_E_UNSUPPORTED_FEATURE;
1209
1210
0
  if (!pcchCards)
1211
0
    return SCARD_E_INVALID_PARAMETER;
1212
1213
0
  autoAllocate = (*pcchCards == SCARD_AUTOALLOCATE);
1214
1215
0
  cardName = findCardByAtr(pbAtr);
1216
0
  if (cardName)
1217
0
    outputLen += strlen(cardName) + 1;
1218
1219
0
  *pcchCards = outputLen;
1220
0
  if (autoAllocate)
1221
0
  {
1222
0
    output = calloc(outputLen, sizeof(WCHAR));
1223
0
    if (!output)
1224
0
      return SCARD_E_NO_MEMORY;
1225
1226
0
    *((LPWSTR*)mszCards) = output;
1227
0
  }
1228
0
  else
1229
0
  {
1230
0
    if (!mszCards)
1231
0
      return SCARD_S_SUCCESS;
1232
1233
0
    if (*pcchCards < outputLen)
1234
0
      return SCARD_E_INSUFFICIENT_BUFFER;
1235
1236
0
    output = mszCards;
1237
0
  }
1238
1239
0
  if (cardName)
1240
0
  {
1241
0
    size_t toCopy = strlen(cardName) + 1;
1242
0
    if (ConvertUtf8ToWChar(cardName, output, toCopy) < 0)
1243
0
      return SCARD_F_INTERNAL_ERROR;
1244
0
    output += toCopy;
1245
0
  }
1246
1247
0
  *output = 0;
1248
1249
0
  return SCARD_S_SUCCESS;
1250
0
}
1251
1252
WINPR_ATTR_NODISCARD
1253
static LONG WINAPI
1254
PCSC_SCardListInterfacesA(SCARDCONTEXT hContext, LPCSTR szCard, LPGUID pguidInterfaces,
1255
                          LPDWORD pcguidInterfaces /* NOLINT(readability-non-const-parameter) */)
1256
0
{
1257
0
  WINPR_UNUSED(hContext);
1258
0
  WINPR_UNUSED(szCard);
1259
0
  WINPR_UNUSED(pguidInterfaces);
1260
0
  WINPR_UNUSED(pcguidInterfaces);
1261
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1262
0
}
1263
1264
WINPR_ATTR_NODISCARD
1265
static LONG WINAPI
1266
PCSC_SCardListInterfacesW(SCARDCONTEXT hContext, LPCWSTR szCard, LPGUID pguidInterfaces,
1267
                          LPDWORD pcguidInterfaces /* NOLINT(readability-non-const-parameter) */)
1268
0
{
1269
0
  WINPR_UNUSED(hContext);
1270
0
  WINPR_UNUSED(szCard);
1271
0
  WINPR_UNUSED(pguidInterfaces);
1272
0
  WINPR_UNUSED(pcguidInterfaces);
1273
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1274
0
}
1275
1276
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetProviderIdA(SCARDCONTEXT hContext,
1277
                                                                 LPCSTR szCard,
1278
                                                                 LPGUID pguidProviderId)
1279
0
{
1280
0
  WINPR_UNUSED(hContext);
1281
0
  WINPR_UNUSED(szCard);
1282
0
  WINPR_UNUSED(pguidProviderId);
1283
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1284
0
}
1285
1286
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetProviderIdW(SCARDCONTEXT hContext,
1287
                                                                 LPCWSTR szCard,
1288
                                                                 LPGUID pguidProviderId)
1289
0
{
1290
0
  WINPR_UNUSED(hContext);
1291
0
  WINPR_UNUSED(szCard);
1292
0
  WINPR_UNUSED(pguidProviderId);
1293
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1294
0
}
1295
1296
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetCardTypeProviderNameA(
1297
    SCARDCONTEXT hContext, LPCSTR szCardName, DWORD dwProviderId,
1298
    CHAR* szProvider /* NOLINT(readability-non-const-parameter) */,
1299
    LPDWORD pcchProvider /* NOLINT(readability-non-const-parameter) */)
1300
0
{
1301
0
  WINPR_UNUSED(hContext);
1302
0
  WINPR_UNUSED(szCardName);
1303
0
  WINPR_UNUSED(dwProviderId);
1304
0
  WINPR_UNUSED(szProvider);
1305
0
  WINPR_UNUSED(pcchProvider);
1306
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1307
0
}
1308
1309
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetCardTypeProviderNameW(
1310
    SCARDCONTEXT hContext, LPCWSTR szCardName, DWORD dwProviderId,
1311
    WCHAR* szProvider /* NOLINT(readability-non-const-parameter) */,
1312
    LPDWORD pcchProvider /* NOLINT(readability-non-const-parameter) */)
1313
0
{
1314
0
  WINPR_UNUSED(hContext);
1315
0
  WINPR_UNUSED(szCardName);
1316
0
  WINPR_UNUSED(dwProviderId);
1317
0
  WINPR_UNUSED(szProvider);
1318
0
  WINPR_UNUSED(pcchProvider);
1319
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1320
0
}
1321
1322
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardIntroduceReaderGroupA(SCARDCONTEXT hContext,
1323
                                                                        LPCSTR szGroupName)
1324
0
{
1325
0
  WINPR_UNUSED(hContext);
1326
0
  WINPR_UNUSED(szGroupName);
1327
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1328
0
}
1329
1330
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardIntroduceReaderGroupW(SCARDCONTEXT hContext,
1331
                                                                        LPCWSTR szGroupName)
1332
0
{
1333
0
  WINPR_UNUSED(hContext);
1334
0
  WINPR_UNUSED(szGroupName);
1335
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1336
0
}
1337
1338
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetReaderGroupA(SCARDCONTEXT hContext,
1339
                                                                     LPCSTR szGroupName)
1340
0
{
1341
0
  WINPR_UNUSED(hContext);
1342
0
  WINPR_UNUSED(szGroupName);
1343
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1344
0
}
1345
1346
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetReaderGroupW(SCARDCONTEXT hContext,
1347
                                                                     LPCWSTR szGroupName)
1348
0
{
1349
0
  WINPR_UNUSED(hContext);
1350
0
  WINPR_UNUSED(szGroupName);
1351
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1352
0
}
1353
1354
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardIntroduceReaderA(SCARDCONTEXT hContext,
1355
                                                                   LPCSTR szReaderName,
1356
                                                                   LPCSTR szDeviceName)
1357
0
{
1358
0
  WINPR_UNUSED(hContext);
1359
0
  WINPR_UNUSED(szReaderName);
1360
0
  WINPR_UNUSED(szDeviceName);
1361
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1362
0
}
1363
1364
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardIntroduceReaderW(SCARDCONTEXT hContext,
1365
                                                                   LPCWSTR szReaderName,
1366
                                                                   LPCWSTR szDeviceName)
1367
0
{
1368
0
  WINPR_UNUSED(hContext);
1369
0
  WINPR_UNUSED(szReaderName);
1370
0
  WINPR_UNUSED(szDeviceName);
1371
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1372
0
}
1373
1374
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetReaderA(SCARDCONTEXT hContext,
1375
                                                                LPCSTR szReaderName)
1376
0
{
1377
0
  WINPR_UNUSED(hContext);
1378
0
  WINPR_UNUSED(szReaderName);
1379
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1380
0
}
1381
1382
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetReaderW(SCARDCONTEXT hContext,
1383
                                                                LPCWSTR szReaderName)
1384
0
{
1385
0
  WINPR_UNUSED(hContext);
1386
0
  WINPR_UNUSED(szReaderName);
1387
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1388
0
}
1389
1390
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardAddReaderToGroupA(SCARDCONTEXT hContext,
1391
                                                                    LPCSTR szReaderName,
1392
                                                                    LPCSTR szGroupName)
1393
0
{
1394
0
  WINPR_UNUSED(hContext);
1395
0
  WINPR_UNUSED(szReaderName);
1396
0
  WINPR_UNUSED(szGroupName);
1397
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1398
0
}
1399
1400
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardAddReaderToGroupW(SCARDCONTEXT hContext,
1401
                                                                    LPCWSTR szReaderName,
1402
                                                                    LPCWSTR szGroupName)
1403
0
{
1404
0
  WINPR_UNUSED(hContext);
1405
0
  WINPR_UNUSED(szReaderName);
1406
0
  WINPR_UNUSED(szGroupName);
1407
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1408
0
}
1409
1410
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardRemoveReaderFromGroupA(SCARDCONTEXT hContext,
1411
                                                                         LPCSTR szReaderName,
1412
                                                                         LPCSTR szGroupName)
1413
0
{
1414
0
  WINPR_UNUSED(hContext);
1415
0
  WINPR_UNUSED(szReaderName);
1416
0
  WINPR_UNUSED(szGroupName);
1417
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1418
0
}
1419
1420
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardRemoveReaderFromGroupW(SCARDCONTEXT hContext,
1421
                                                                         LPCWSTR szReaderName,
1422
                                                                         LPCWSTR szGroupName)
1423
0
{
1424
0
  WINPR_UNUSED(hContext);
1425
0
  WINPR_UNUSED(szReaderName);
1426
0
  WINPR_UNUSED(szGroupName);
1427
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1428
0
}
1429
1430
WINPR_ATTR_NODISCARD static LONG WINAPI
1431
PCSC_SCardIntroduceCardTypeA(SCARDCONTEXT hContext, LPCSTR szCardName, LPCGUID pguidPrimaryProvider,
1432
                             LPCGUID rgguidInterfaces, DWORD dwInterfaceCount, LPCBYTE pbAtr,
1433
                             LPCBYTE pbAtrMask, DWORD cbAtrLen)
1434
0
{
1435
0
  WINPR_UNUSED(hContext);
1436
0
  WINPR_UNUSED(szCardName);
1437
0
  WINPR_UNUSED(pguidPrimaryProvider);
1438
0
  WINPR_UNUSED(rgguidInterfaces);
1439
0
  WINPR_UNUSED(dwInterfaceCount);
1440
0
  WINPR_UNUSED(pbAtr);
1441
0
  WINPR_UNUSED(pbAtrMask);
1442
0
  WINPR_UNUSED(cbAtrLen);
1443
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1444
0
}
1445
1446
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardIntroduceCardTypeW(
1447
    SCARDCONTEXT hContext, LPCWSTR szCardName, LPCGUID pguidPrimaryProvider,
1448
    LPCGUID rgguidInterfaces, DWORD dwInterfaceCount, LPCBYTE pbAtr, LPCBYTE pbAtrMask,
1449
    DWORD cbAtrLen)
1450
0
{
1451
0
  WINPR_UNUSED(hContext);
1452
0
  WINPR_UNUSED(szCardName);
1453
0
  WINPR_UNUSED(pguidPrimaryProvider);
1454
0
  WINPR_UNUSED(rgguidInterfaces);
1455
0
  WINPR_UNUSED(dwInterfaceCount);
1456
0
  WINPR_UNUSED(pbAtr);
1457
0
  WINPR_UNUSED(pbAtrMask);
1458
0
  WINPR_UNUSED(cbAtrLen);
1459
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1460
0
}
1461
1462
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardSetCardTypeProviderNameA(SCARDCONTEXT hContext,
1463
                                                                           LPCSTR szCardName,
1464
                                                                           DWORD dwProviderId,
1465
                                                                           LPCSTR szProvider)
1466
0
{
1467
0
  WINPR_UNUSED(hContext);
1468
0
  WINPR_UNUSED(szCardName);
1469
0
  WINPR_UNUSED(dwProviderId);
1470
0
  WINPR_UNUSED(szProvider);
1471
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1472
0
}
1473
1474
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardSetCardTypeProviderNameW(SCARDCONTEXT hContext,
1475
                                                                           LPCWSTR szCardName,
1476
                                                                           DWORD dwProviderId,
1477
                                                                           LPCWSTR szProvider)
1478
0
{
1479
0
  WINPR_UNUSED(hContext);
1480
0
  WINPR_UNUSED(szCardName);
1481
0
  WINPR_UNUSED(dwProviderId);
1482
0
  WINPR_UNUSED(szProvider);
1483
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1484
0
}
1485
1486
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetCardTypeA(SCARDCONTEXT hContext,
1487
                                                                  LPCSTR szCardName)
1488
0
{
1489
0
  WINPR_UNUSED(hContext);
1490
0
  WINPR_UNUSED(szCardName);
1491
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1492
0
}
1493
1494
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardForgetCardTypeW(SCARDCONTEXT hContext,
1495
                                                                  LPCWSTR szCardName)
1496
0
{
1497
0
  WINPR_UNUSED(hContext);
1498
0
  WINPR_UNUSED(szCardName);
1499
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1500
0
}
1501
1502
static LONG WINAPI PCSC_SCardFreeMemory_Internal(SCARDCONTEXT hContext, LPVOID pvMem)
1503
0
{
1504
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1505
1506
0
  if (PCSC_RemoveMemoryBlock(hContext, pvMem))
1507
0
    status = SCARD_S_SUCCESS;
1508
0
  else
1509
0
  {
1510
0
    if (g_PCSC.pfnSCardFreeMemory)
1511
0
    {
1512
0
      status = g_PCSC.pfnSCardFreeMemory(hContext, pvMem);
1513
0
    }
1514
0
  }
1515
1516
0
  return PCSC_MapErrorCodeToWinSCard(status);
1517
0
}
1518
1519
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardFreeMemory(SCARDCONTEXT hContext, LPVOID pvMem)
1520
0
{
1521
0
  LONG status = SCARD_S_SUCCESS;
1522
1523
0
  if (hContext)
1524
0
  {
1525
0
    if (!PCSC_LockCardContext(hContext))
1526
0
      return SCARD_E_INVALID_HANDLE;
1527
0
  }
1528
1529
0
  status = PCSC_SCardFreeMemory_Internal(hContext, pvMem);
1530
1531
0
  if (hContext)
1532
0
  {
1533
0
    if (!PCSC_UnlockCardContext(hContext))
1534
0
      return SCARD_E_INVALID_HANDLE;
1535
0
  }
1536
1537
0
  return status;
1538
0
}
1539
1540
WINPR_ATTR_NODISCARD
1541
static HANDLE WINAPI PCSC_SCardAccessStartedEvent(void)
1542
0
{
1543
0
  LONG status = 0;
1544
0
  SCARDCONTEXT hContext = 0;
1545
1546
0
  status = PCSC_SCardEstablishContext(SCARD_SCOPE_SYSTEM, nullptr, nullptr, &hContext);
1547
1548
0
  if (status != SCARD_S_SUCCESS)
1549
0
    return nullptr;
1550
1551
0
  status = PCSC_SCardReleaseContext(hContext);
1552
1553
0
  if (status != SCARD_S_SUCCESS)
1554
0
    return nullptr;
1555
1556
0
  if (!g_StartedEvent)
1557
0
  {
1558
0
    if (!(g_StartedEvent = CreateEvent(nullptr, TRUE, FALSE, nullptr)))
1559
0
      return nullptr;
1560
1561
0
    if (!SetEvent(g_StartedEvent))
1562
0
    {
1563
0
      (void)CloseHandle(g_StartedEvent);
1564
0
      return nullptr;
1565
0
    }
1566
0
  }
1567
1568
0
  g_StartedEventRefCount++;
1569
0
  return g_StartedEvent;
1570
0
}
1571
1572
static void WINAPI PCSC_SCardReleaseStartedEvent(void)
1573
0
{
1574
0
  g_StartedEventRefCount--;
1575
1576
0
  if (g_StartedEventRefCount == 0)
1577
0
  {
1578
0
    if (g_StartedEvent)
1579
0
    {
1580
0
      (void)CloseHandle(g_StartedEvent);
1581
0
      g_StartedEvent = nullptr;
1582
0
    }
1583
0
  }
1584
0
}
1585
1586
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardLocateCardsA(SCARDCONTEXT hContext,
1587
                                                               LPCSTR mszCards,
1588
                                                               LPSCARD_READERSTATEA rgReaderStates,
1589
                                                               DWORD cReaders)
1590
0
{
1591
0
  WINPR_UNUSED(hContext);
1592
0
  WINPR_UNUSED(mszCards);
1593
0
  WINPR_UNUSED(rgReaderStates);
1594
0
  WINPR_UNUSED(cReaders);
1595
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1596
0
}
1597
1598
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardLocateCardsW(SCARDCONTEXT hContext,
1599
                                                               LPCWSTR mszCards,
1600
                                                               LPSCARD_READERSTATEW rgReaderStates,
1601
                                                               DWORD cReaders)
1602
0
{
1603
0
  WINPR_UNUSED(hContext);
1604
0
  WINPR_UNUSED(mszCards);
1605
0
  WINPR_UNUSED(rgReaderStates);
1606
0
  WINPR_UNUSED(cReaders);
1607
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1608
0
}
1609
1610
WINPR_ATTR_NODISCARD static LONG WINAPI
1611
PCSC_SCardLocateCardsByATRA(SCARDCONTEXT hContext, LPSCARD_ATRMASK rgAtrMasks, DWORD cAtrs,
1612
                            LPSCARD_READERSTATEA rgReaderStates, DWORD cReaders)
1613
0
{
1614
0
  WINPR_UNUSED(hContext);
1615
0
  WINPR_UNUSED(rgAtrMasks);
1616
0
  WINPR_UNUSED(cAtrs);
1617
0
  WINPR_UNUSED(rgReaderStates);
1618
0
  WINPR_UNUSED(cReaders);
1619
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1620
0
}
1621
1622
WINPR_ATTR_NODISCARD static LONG WINAPI
1623
PCSC_SCardLocateCardsByATRW(SCARDCONTEXT hContext, LPSCARD_ATRMASK rgAtrMasks, DWORD cAtrs,
1624
                            LPSCARD_READERSTATEW rgReaderStates, DWORD cReaders)
1625
0
{
1626
0
  WINPR_UNUSED(hContext);
1627
0
  WINPR_UNUSED(rgAtrMasks);
1628
0
  WINPR_UNUSED(cAtrs);
1629
0
  WINPR_UNUSED(rgReaderStates);
1630
0
  WINPR_UNUSED(cReaders);
1631
0
  return SCARD_E_UNSUPPORTED_FEATURE;
1632
0
}
1633
1634
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetStatusChange_Internal(
1635
    SCARDCONTEXT hContext, DWORD dwTimeout, LPSCARD_READERSTATEA rgReaderStates, DWORD cReaders)
1636
0
{
1637
0
  INT64* map = nullptr;
1638
0
  PCSC_DWORD cMappedReaders = 0;
1639
0
  PCSC_SCARD_READERSTATE* states = nullptr;
1640
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1641
0
  PCSC_DWORD pcsc_dwTimeout = (PCSC_DWORD)dwTimeout;
1642
0
  PCSC_DWORD pcsc_cReaders = (PCSC_DWORD)cReaders;
1643
1644
0
  if (!g_PCSC.pfnSCardGetStatusChange)
1645
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardGetStatusChange");
1646
1647
0
  if (!cReaders)
1648
0
    return SCARD_S_SUCCESS;
1649
1650
  /* pcsc-lite interprets value 0 as INFINITE, work around the problem by using value 1 */
1651
0
  pcsc_dwTimeout = pcsc_dwTimeout ? pcsc_dwTimeout : 1;
1652
  /**
1653
   * Apple's SmartCard Services (not vanilla pcsc-lite) appears to have trouble with the
1654
   * "\\\\?PnP?\\Notification" reader name. I am always getting EXC_BAD_ACCESS with it.
1655
   *
1656
   * The SmartCard Services tarballs can be found here:
1657
   * http://opensource.apple.com/tarballs/SmartCardServices/
1658
   *
1659
   * The "\\\\?PnP?\\Notification" string cannot be found anywhere in the sources,
1660
   * while this string is present in the vanilla pcsc-lite sources.
1661
   *
1662
   * To work around this apparent lack of "\\\\?PnP?\\Notification" support,
1663
   * we have to filter rgReaderStates to exclude the special PnP reader name.
1664
   */
1665
0
  map = (INT64*)calloc(pcsc_cReaders, sizeof(INT64));
1666
1667
0
  if (!map)
1668
0
    return SCARD_E_NO_MEMORY;
1669
1670
0
  states = (PCSC_SCARD_READERSTATE*)calloc(pcsc_cReaders, sizeof(PCSC_SCARD_READERSTATE));
1671
1672
0
  if (!states)
1673
0
  {
1674
0
    free(map);
1675
0
    return SCARD_E_NO_MEMORY;
1676
0
  }
1677
1678
0
  PCSC_DWORD j = 0;
1679
0
  for (PCSC_DWORD i = 0; i < pcsc_cReaders; i++)
1680
0
  {
1681
0
    if (!g_PnP_Notification)
1682
0
    {
1683
0
      LPSCARD_READERSTATEA reader = &rgReaderStates[i];
1684
0
      if (!reader->szReader)
1685
0
        continue;
1686
0
      if (0 == _stricmp(reader->szReader, SMARTCARD_PNP_NOTIFICATION_A))
1687
0
      {
1688
0
        map[i] = -1; /* unmapped */
1689
0
        continue;
1690
0
      }
1691
0
    }
1692
1693
0
    map[i] = (INT64)j;
1694
0
    states[j].szReader = rgReaderStates[i].szReader;
1695
0
    states[j].dwCurrentState = rgReaderStates[i].dwCurrentState;
1696
0
    states[j].pvUserData = rgReaderStates[i].pvUserData;
1697
0
    states[j].dwEventState = rgReaderStates[i].dwEventState;
1698
0
    states[j].cbAtr = rgReaderStates[i].cbAtr;
1699
0
    CopyMemory(&(states[j].rgbAtr), &(rgReaderStates[i].rgbAtr), PCSC_MAX_ATR_SIZE);
1700
0
    j++;
1701
0
  }
1702
1703
0
  cMappedReaders = j;
1704
1705
0
  if (cMappedReaders > 0)
1706
0
  {
1707
0
    status = g_PCSC.pfnSCardGetStatusChange(hContext, pcsc_dwTimeout, states, cMappedReaders);
1708
0
  }
1709
0
  else
1710
0
  {
1711
0
    status = SCARD_S_SUCCESS;
1712
0
  }
1713
1714
0
  for (PCSC_DWORD i = 0; i < pcsc_cReaders; i++)
1715
0
  {
1716
0
    if (map[i] < 0)
1717
0
      continue; /* unmapped */
1718
1719
0
    PCSC_DWORD k = (PCSC_DWORD)map[i];
1720
0
    rgReaderStates[i].dwCurrentState = (DWORD)states[k].dwCurrentState;
1721
0
    rgReaderStates[i].cbAtr = (DWORD)states[k].cbAtr;
1722
0
    CopyMemory(&(rgReaderStates[i].rgbAtr), &(states[k].rgbAtr), PCSC_MAX_ATR_SIZE);
1723
0
    rgReaderStates[i].dwEventState = (DWORD)states[k].dwEventState;
1724
0
  }
1725
1726
0
  free(map);
1727
0
  free(states);
1728
0
  return PCSC_MapErrorCodeToWinSCard(status);
1729
0
}
1730
1731
WINPR_ATTR_NODISCARD
1732
static LONG WINAPI PCSC_SCardGetStatusChangeA(SCARDCONTEXT hContext, DWORD dwTimeout,
1733
                                              LPSCARD_READERSTATEA rgReaderStates, DWORD cReaders)
1734
0
{
1735
0
  LONG status = SCARD_S_SUCCESS;
1736
1737
0
  if (!PCSC_LockCardContext(hContext))
1738
0
    return SCARD_E_INVALID_HANDLE;
1739
1740
0
  status = PCSC_SCardGetStatusChange_Internal(hContext, dwTimeout, rgReaderStates, cReaders);
1741
1742
0
  if (!PCSC_UnlockCardContext(hContext))
1743
0
    return SCARD_E_INVALID_HANDLE;
1744
1745
0
  return status;
1746
0
}
1747
1748
WINPR_ATTR_NODISCARD
1749
static LONG WINAPI PCSC_SCardGetStatusChangeW(SCARDCONTEXT hContext, DWORD dwTimeout,
1750
                                              LPSCARD_READERSTATEW rgReaderStates, DWORD cReaders)
1751
0
{
1752
0
  LPSCARD_READERSTATEA states = nullptr;
1753
0
  LONG status = SCARD_S_SUCCESS;
1754
1755
0
  if (!g_PCSC.pfnSCardGetStatusChange)
1756
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardGetStatusChange");
1757
1758
0
  if (!PCSC_LockCardContext(hContext))
1759
0
    return SCARD_E_INVALID_HANDLE;
1760
1761
0
  states = (LPSCARD_READERSTATEA)calloc(cReaders, sizeof(SCARD_READERSTATEA));
1762
1763
0
  if (!states)
1764
0
  {
1765
0
    if (!PCSC_UnlockCardContext(hContext))
1766
0
      return SCARD_E_INVALID_HANDLE;
1767
0
    return SCARD_E_NO_MEMORY;
1768
0
  }
1769
1770
0
  for (DWORD index = 0; index < cReaders; index++)
1771
0
  {
1772
0
    const LPSCARD_READERSTATEW curReader = &rgReaderStates[index];
1773
0
    LPSCARD_READERSTATEA cur = &states[index];
1774
1775
0
    cur->szReader = ConvertWCharToUtf8Alloc(curReader->szReader, nullptr);
1776
0
    cur->pvUserData = curReader->pvUserData;
1777
0
    cur->dwCurrentState = curReader->dwCurrentState;
1778
0
    cur->dwEventState = curReader->dwEventState;
1779
0
    cur->cbAtr = curReader->cbAtr;
1780
0
    CopyMemory(&(cur->rgbAtr), &(curReader->rgbAtr), ARRAYSIZE(cur->rgbAtr));
1781
0
  }
1782
1783
0
  status = PCSC_SCardGetStatusChange_Internal(hContext, dwTimeout, states, cReaders);
1784
1785
0
  for (DWORD index = 0; index < cReaders; index++)
1786
0
  {
1787
0
    free((void*)states[index].szReader);
1788
0
    rgReaderStates[index].pvUserData = states[index].pvUserData;
1789
0
    rgReaderStates[index].dwCurrentState = states[index].dwCurrentState;
1790
0
    rgReaderStates[index].dwEventState = states[index].dwEventState;
1791
0
    rgReaderStates[index].cbAtr = states[index].cbAtr;
1792
0
    CopyMemory(&(rgReaderStates[index].rgbAtr), &(states[index].rgbAtr), 36);
1793
0
  }
1794
1795
0
  free(states);
1796
1797
0
  if (!PCSC_UnlockCardContext(hContext))
1798
0
    return SCARD_E_INVALID_HANDLE;
1799
1800
0
  return status;
1801
0
}
1802
1803
static LONG WINAPI PCSC_SCardCancel(SCARDCONTEXT hContext)
1804
0
{
1805
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1806
1807
0
  if (!g_PCSC.pfnSCardCancel)
1808
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardCancel");
1809
1810
0
  status = g_PCSC.pfnSCardCancel(hContext);
1811
0
  return PCSC_MapErrorCodeToWinSCard(status);
1812
0
}
1813
1814
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardConnect_Internal(
1815
    SCARDCONTEXT hContext, LPCSTR szReader, DWORD dwShareMode, DWORD dwPreferredProtocols,
1816
    LPSCARDHANDLE phCard, LPDWORD pdwActiveProtocol)
1817
0
{
1818
0
  BOOL shared = 0;
1819
0
  const char* szReaderPCSC = nullptr;
1820
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1821
0
  PCSC_SCARDHANDLE* pCard = nullptr;
1822
0
  PCSC_DWORD pcsc_dwShareMode = (PCSC_DWORD)dwShareMode;
1823
0
  PCSC_DWORD pcsc_dwPreferredProtocols = 0;
1824
0
  PCSC_DWORD pcsc_dwActiveProtocol = 0;
1825
1826
0
  if (!g_PCSC.pfnSCardConnect)
1827
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardConnect");
1828
1829
0
  shared = (dwShareMode == SCARD_SHARE_DIRECT) != 0;
1830
0
  PCSC_WaitForCardAccess(hContext, 0, shared);
1831
0
  szReaderPCSC = szReader;
1832
1833
  /**
1834
   * As stated here :
1835
   * https://pcsclite.alioth.debian.org/api/group__API.html#ga4e515829752e0a8dbc4d630696a8d6a5
1836
   * SCARD_PROTOCOL_UNDEFINED is valid for dwPreferredProtocols (only) if dwShareMode ==
1837
   * SCARD_SHARE_DIRECT and allows to send control commands to the reader (with SCardControl())
1838
   * even if a card is not present in the reader
1839
   */
1840
0
  if (pcsc_dwShareMode == SCARD_SHARE_DIRECT && dwPreferredProtocols == SCARD_PROTOCOL_UNDEFINED)
1841
0
    pcsc_dwPreferredProtocols = SCARD_PROTOCOL_UNDEFINED;
1842
0
  else
1843
0
    pcsc_dwPreferredProtocols =
1844
0
        (PCSC_DWORD)PCSC_ConvertProtocolsFromWinSCard(dwPreferredProtocols);
1845
1846
0
  status = g_PCSC.pfnSCardConnect(hContext, szReaderPCSC, pcsc_dwShareMode,
1847
0
                                  pcsc_dwPreferredProtocols, phCard, &pcsc_dwActiveProtocol);
1848
1849
0
  if (status == SCARD_S_SUCCESS)
1850
0
  {
1851
0
    pCard = PCSC_ConnectCardHandle(hContext, *phCard);
1852
0
    *pdwActiveProtocol = PCSC_ConvertProtocolsToWinSCard((DWORD)pcsc_dwActiveProtocol);
1853
0
    pCard->shared = shared;
1854
1855
    // NOLINTNEXTLINE(clang-analyzer-unix.Malloc): ListDictionary_Add takes ownership of pCard
1856
0
    PCSC_WaitForCardAccess(hContext, pCard->hSharedContext, shared);
1857
0
  }
1858
1859
0
  return PCSC_MapErrorCodeToWinSCard(status);
1860
0
}
1861
1862
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardConnectA(SCARDCONTEXT hContext, LPCSTR szReader,
1863
                                                           DWORD dwShareMode,
1864
                                                           DWORD dwPreferredProtocols,
1865
                                                           LPSCARDHANDLE phCard,
1866
                                                           LPDWORD pdwActiveProtocol)
1867
0
{
1868
0
  LONG status = SCARD_S_SUCCESS;
1869
1870
0
  if (!PCSC_LockCardContext(hContext))
1871
0
    return SCARD_E_INVALID_HANDLE;
1872
1873
0
  status = PCSC_SCardConnect_Internal(hContext, szReader, dwShareMode, dwPreferredProtocols,
1874
0
                                      phCard, pdwActiveProtocol);
1875
1876
0
  if (!PCSC_UnlockCardContext(hContext))
1877
0
    return SCARD_E_INVALID_HANDLE;
1878
1879
0
  return status;
1880
0
}
1881
1882
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardConnectW(SCARDCONTEXT hContext, LPCWSTR szReader,
1883
                                                           DWORD dwShareMode,
1884
                                                           DWORD dwPreferredProtocols,
1885
                                                           LPSCARDHANDLE phCard,
1886
                                                           LPDWORD pdwActiveProtocol)
1887
0
{
1888
0
  LPSTR szReaderA = nullptr;
1889
0
  LONG status = SCARD_S_SUCCESS;
1890
1891
0
  if (!PCSC_LockCardContext(hContext))
1892
0
    return SCARD_E_INVALID_HANDLE;
1893
1894
0
  if (szReader)
1895
0
  {
1896
0
    szReaderA = ConvertWCharToUtf8Alloc(szReader, nullptr);
1897
0
    if (!szReaderA)
1898
0
    {
1899
0
      status = SCARD_E_INSUFFICIENT_BUFFER;
1900
0
      goto fail;
1901
0
    }
1902
0
  }
1903
1904
0
  status = PCSC_SCardConnect_Internal(hContext, szReaderA, dwShareMode, dwPreferredProtocols,
1905
0
                                      phCard, pdwActiveProtocol);
1906
0
  free(szReaderA);
1907
1908
0
fail:
1909
0
  if (!PCSC_UnlockCardContext(hContext))
1910
0
    return SCARD_E_INVALID_HANDLE;
1911
1912
0
  return status;
1913
0
}
1914
1915
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardReconnect(SCARDHANDLE hCard, DWORD dwShareMode,
1916
                                                            DWORD dwPreferredProtocols,
1917
                                                            DWORD dwInitialization,
1918
                                                            LPDWORD pdwActiveProtocol)
1919
0
{
1920
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1921
0
  PCSC_DWORD pcsc_dwShareMode = (PCSC_DWORD)dwShareMode;
1922
0
  PCSC_DWORD pcsc_dwPreferredProtocols = 0;
1923
0
  PCSC_DWORD pcsc_dwInitialization = (PCSC_DWORD)dwInitialization;
1924
0
  PCSC_DWORD pcsc_dwActiveProtocol = 0;
1925
1926
0
  if (!g_PCSC.pfnSCardReconnect)
1927
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardReconnect");
1928
1929
0
  const BOOL shared = (dwShareMode == SCARD_SHARE_DIRECT) != 0;
1930
0
  PCSC_WaitForCardAccess(0, hCard, shared);
1931
0
  pcsc_dwPreferredProtocols = (PCSC_DWORD)PCSC_ConvertProtocolsFromWinSCard(dwPreferredProtocols);
1932
0
  status = g_PCSC.pfnSCardReconnect(hCard, pcsc_dwShareMode, pcsc_dwPreferredProtocols,
1933
0
                                    pcsc_dwInitialization, &pcsc_dwActiveProtocol);
1934
1935
0
  *pdwActiveProtocol = PCSC_ConvertProtocolsToWinSCard((DWORD)pcsc_dwActiveProtocol);
1936
0
  return PCSC_MapErrorCodeToWinSCard(status);
1937
0
}
1938
1939
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardDisconnect(SCARDHANDLE hCard, DWORD dwDisposition)
1940
0
{
1941
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1942
0
  PCSC_DWORD pcsc_dwDisposition = (PCSC_DWORD)dwDisposition;
1943
1944
0
  if (!g_PCSC.pfnSCardDisconnect)
1945
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardDisconnect");
1946
1947
0
  status = g_PCSC.pfnSCardDisconnect(hCard, pcsc_dwDisposition);
1948
1949
0
  if (status == SCARD_S_SUCCESS)
1950
0
  {
1951
0
    PCSC_SCARDHANDLE* pCard = PCSC_GetCardHandleData(hCard);
1952
0
    PCSC_DisconnectCardHandle(pCard);
1953
0
  }
1954
1955
0
  PCSC_ReleaseCardAccess(0, hCard);
1956
0
  return PCSC_MapErrorCodeToWinSCard(status);
1957
0
}
1958
1959
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardBeginTransaction(SCARDHANDLE hCard)
1960
0
{
1961
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1962
0
  PCSC_SCARDHANDLE* pCard = nullptr;
1963
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
1964
1965
0
  if (!g_PCSC.pfnSCardBeginTransaction)
1966
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardBeginTransaction");
1967
1968
0
  pCard = PCSC_GetCardHandleData(hCard);
1969
1970
0
  if (!pCard)
1971
0
    return SCARD_E_INVALID_HANDLE;
1972
1973
0
  pContext = PCSC_GetCardContextData(pCard->hSharedContext);
1974
1975
0
  if (!pContext)
1976
0
    return SCARD_E_INVALID_HANDLE;
1977
1978
0
  if (pContext->isTransactionLocked)
1979
0
    return SCARD_S_SUCCESS; /* disable nested transactions */
1980
1981
0
  status = g_PCSC.pfnSCardBeginTransaction(hCard);
1982
1983
0
  pContext->isTransactionLocked = TRUE;
1984
0
  return PCSC_MapErrorCodeToWinSCard(status);
1985
0
}
1986
1987
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardEndTransaction(SCARDHANDLE hCard,
1988
                                                                 DWORD dwDisposition)
1989
0
{
1990
0
  PCSC_LONG status = SCARD_S_SUCCESS;
1991
0
  PCSC_SCARDHANDLE* pCard = nullptr;
1992
0
  PCSC_SCARDCONTEXT* pContext = nullptr;
1993
0
  PCSC_DWORD pcsc_dwDisposition = (PCSC_DWORD)dwDisposition;
1994
1995
0
  if (!g_PCSC.pfnSCardEndTransaction)
1996
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardEndTransaction");
1997
1998
0
  pCard = PCSC_GetCardHandleData(hCard);
1999
2000
0
  if (!pCard)
2001
0
    return SCARD_E_INVALID_HANDLE;
2002
2003
0
  pContext = PCSC_GetCardContextData(pCard->hSharedContext);
2004
2005
0
  if (!pContext)
2006
0
    return SCARD_E_INVALID_HANDLE;
2007
2008
0
  PCSC_ReleaseCardAccess(0, hCard);
2009
2010
0
  if (!pContext->isTransactionLocked)
2011
0
    return SCARD_S_SUCCESS; /* disable nested transactions */
2012
2013
0
  status = g_PCSC.pfnSCardEndTransaction(hCard, pcsc_dwDisposition);
2014
2015
0
  pContext->isTransactionLocked = FALSE;
2016
0
  return PCSC_MapErrorCodeToWinSCard(status);
2017
0
}
2018
2019
static LONG WINAPI PCSC_SCardCancelTransaction(SCARDHANDLE hCard)
2020
0
{
2021
0
  WINPR_UNUSED(hCard);
2022
0
  return SCARD_S_SUCCESS;
2023
0
}
2024
2025
/*
2026
 * PCSC returns a string but Windows SCardStatus requires the return to be a multi string.
2027
 * Therefore extra length checks and additional buffer allocation is required
2028
 */
2029
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardStatus_Internal(
2030
    SCARDHANDLE hCard, LPSTR mszReaderNames, LPDWORD pcchReaderLen, LPDWORD pdwState,
2031
    LPDWORD pdwProtocol, LPBYTE pbAtr, LPDWORD pcbAtrLen, BOOL unicode)
2032
0
{
2033
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2034
0
  SCARDCONTEXT hContext = 0;
2035
0
  PCSC_LONG status = 0;
2036
0
  PCSC_DWORD pcsc_cchReaderLen = 0;
2037
0
  PCSC_DWORD pcsc_cbAtrLen = 0;
2038
0
  PCSC_DWORD pcsc_dwState = 0;
2039
0
  PCSC_DWORD pcsc_dwProtocol = 0;
2040
0
  BOOL allocateReader = FALSE;
2041
0
  BOOL allocateAtr = FALSE;
2042
0
  LPSTR readerNames = mszReaderNames;
2043
0
  LPBYTE atr = pbAtr;
2044
0
  LPSTR tReader = nullptr;
2045
0
  LPBYTE tATR = nullptr;
2046
2047
0
  if (!g_PCSC.pfnSCardStatus)
2048
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardStatus");
2049
2050
0
  pCard = PCSC_GetCardHandleData(hCard);
2051
2052
0
  if (!pCard)
2053
0
    return SCARD_E_INVALID_VALUE;
2054
2055
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2056
0
  hContext = PCSC_GetCardContextFromHandle(hCard);
2057
2058
0
  if (!hContext)
2059
0
    return SCARD_E_INVALID_VALUE;
2060
2061
0
  status = g_PCSC.pfnSCardStatus(hCard, nullptr, &pcsc_cchReaderLen, nullptr, nullptr, nullptr,
2062
0
                                 &pcsc_cbAtrLen);
2063
2064
0
  if (status != STATUS_SUCCESS)
2065
0
    return PCSC_MapErrorCodeToWinSCard(status);
2066
2067
0
  pcsc_cchReaderLen++;
2068
2069
0
  if (unicode)
2070
0
    pcsc_cchReaderLen *= 2;
2071
2072
0
  if (pcchReaderLen)
2073
0
  {
2074
0
    if (*pcchReaderLen == SCARD_AUTOALLOCATE)
2075
0
      allocateReader = TRUE;
2076
0
    else if (mszReaderNames && (*pcchReaderLen < pcsc_cchReaderLen))
2077
0
      return SCARD_E_INSUFFICIENT_BUFFER;
2078
0
    else
2079
0
      pcsc_cchReaderLen = *pcchReaderLen;
2080
0
  }
2081
2082
0
  if (pcbAtrLen)
2083
0
  {
2084
0
    if (*pcbAtrLen == SCARD_AUTOALLOCATE)
2085
0
      allocateAtr = TRUE;
2086
0
    else if (pbAtr && (*pcbAtrLen < pcsc_cbAtrLen))
2087
0
      return SCARD_E_INSUFFICIENT_BUFFER;
2088
0
    else
2089
0
      pcsc_cbAtrLen = *pcbAtrLen;
2090
0
  }
2091
2092
0
  if (allocateReader && pcsc_cchReaderLen > 0 && mszReaderNames)
2093
0
  {
2094
#ifdef __MACOSX__
2095
2096
    /**
2097
     * Workaround for SCardStatus Bug in MAC OS X Yosemite
2098
     */
2099
    if (OSXVersion == 0x10100000)
2100
      pcsc_cchReaderLen++;
2101
2102
#endif
2103
0
    tReader = calloc(sizeof(CHAR), pcsc_cchReaderLen + 1);
2104
2105
0
    if (!tReader)
2106
0
    {
2107
0
      status = ERROR_NOT_ENOUGH_MEMORY;
2108
0
      goto out_fail;
2109
0
    }
2110
2111
0
    readerNames = tReader;
2112
0
  }
2113
2114
0
  if (allocateAtr && pcsc_cbAtrLen > 0 && pbAtr)
2115
0
  {
2116
0
    tATR = calloc(1, pcsc_cbAtrLen);
2117
2118
0
    if (!tATR)
2119
0
    {
2120
0
      status = ERROR_NOT_ENOUGH_MEMORY;
2121
0
      goto out_fail;
2122
0
    }
2123
2124
0
    atr = tATR;
2125
0
  }
2126
2127
0
  status = g_PCSC.pfnSCardStatus(hCard, readerNames, &pcsc_cchReaderLen, &pcsc_dwState,
2128
0
                                 &pcsc_dwProtocol, atr, &pcsc_cbAtrLen);
2129
2130
0
  if (status != STATUS_SUCCESS)
2131
0
    goto out_fail;
2132
2133
0
  if (tATR)
2134
0
  {
2135
0
    PCSC_AddMemoryBlock(hContext, tATR);
2136
0
    *(BYTE**)pbAtr = tATR;
2137
0
  }
2138
2139
0
  if (tReader)
2140
0
  {
2141
0
    if (unicode)
2142
0
    {
2143
0
      size_t size = 0;
2144
0
      WCHAR* tmp = ConvertMszUtf8NToWCharAlloc(tReader, pcsc_cchReaderLen + 1, &size);
2145
2146
0
      if (tmp == nullptr)
2147
0
      {
2148
0
        status = ERROR_NOT_ENOUGH_MEMORY;
2149
0
        goto out_fail;
2150
0
      }
2151
2152
0
      free(tReader);
2153
2154
0
      PCSC_AddMemoryBlock(hContext, tmp);
2155
0
      *(WCHAR**)mszReaderNames = tmp;
2156
0
    }
2157
0
    else
2158
0
    {
2159
0
      tReader[pcsc_cchReaderLen - 1] = '\0';
2160
0
      PCSC_AddMemoryBlock(hContext, tReader);
2161
0
      *(char**)mszReaderNames = tReader;
2162
0
    }
2163
0
  }
2164
2165
0
  pcsc_dwState &= 0xFFFF;
2166
2167
0
  if (pdwState)
2168
0
    *pdwState = PCSC_ConvertCardStateToWinSCard((DWORD)pcsc_dwState, status);
2169
2170
0
  if (pdwProtocol)
2171
0
    *pdwProtocol = PCSC_ConvertProtocolsToWinSCard((DWORD)pcsc_dwProtocol);
2172
2173
0
  if (pcbAtrLen)
2174
0
    *pcbAtrLen = (DWORD)pcsc_cbAtrLen;
2175
2176
0
  if (pcchReaderLen)
2177
0
  {
2178
0
    WINPR_ASSERT(pcsc_cchReaderLen < UINT32_MAX);
2179
0
    *pcchReaderLen = (DWORD)pcsc_cchReaderLen + 1u;
2180
0
  }
2181
2182
0
  return (LONG)status;
2183
0
out_fail:
2184
0
  free(tReader);
2185
0
  free(tATR);
2186
0
  return (LONG)status;
2187
0
}
2188
2189
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardState(SCARDHANDLE hCard, LPDWORD pdwState,
2190
                                                        LPDWORD pdwProtocol, LPBYTE pbAtr,
2191
                                                        LPDWORD pcbAtrLen)
2192
0
{
2193
0
  DWORD cchReaderLen = 0;
2194
0
  SCARDCONTEXT hContext = 0;
2195
0
  LPSTR mszReaderNames = nullptr;
2196
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2197
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2198
0
  DWORD pcsc_dwState = 0;
2199
0
  DWORD pcsc_dwProtocol = 0;
2200
0
  DWORD pcsc_cbAtrLen = 0;
2201
2202
0
  if (pcbAtrLen)
2203
0
    pcsc_cbAtrLen = (DWORD)*pcbAtrLen;
2204
2205
0
  if (!g_PCSC.pfnSCardStatus)
2206
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardStatus");
2207
2208
0
  pCard = PCSC_GetCardHandleData(hCard);
2209
2210
0
  if (!pCard)
2211
0
    return SCARD_E_INVALID_VALUE;
2212
2213
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2214
0
  hContext = PCSC_GetCardContextFromHandle(hCard);
2215
2216
0
  if (!hContext)
2217
0
    return SCARD_E_INVALID_VALUE;
2218
2219
0
  cchReaderLen = SCARD_AUTOALLOCATE;
2220
0
  status = PCSC_SCardStatus_Internal(hCard, (LPSTR)&mszReaderNames, &cchReaderLen, &pcsc_dwState,
2221
0
                                     &pcsc_dwProtocol, pbAtr, &pcsc_cbAtrLen, FALSE);
2222
2223
0
  if (mszReaderNames)
2224
0
    PCSC_SCardFreeMemory_Internal(hContext, mszReaderNames);
2225
2226
0
  *pdwState = pcsc_dwState;
2227
0
  *pdwProtocol = PCSC_ConvertProtocolsToWinSCard(pcsc_dwProtocol);
2228
0
  if (pcbAtrLen)
2229
0
    *pcbAtrLen = pcsc_cbAtrLen;
2230
0
  return PCSC_MapErrorCodeToWinSCard(status);
2231
0
}
2232
2233
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardStatusA(SCARDHANDLE hCard, LPSTR mszReaderNames,
2234
                                                          LPDWORD pcchReaderLen, LPDWORD pdwState,
2235
                                                          LPDWORD pdwProtocol, LPBYTE pbAtr,
2236
                                                          LPDWORD pcbAtrLen)
2237
0
{
2238
2239
0
  return PCSC_SCardStatus_Internal(hCard, mszReaderNames, pcchReaderLen, pdwState, pdwProtocol,
2240
0
                                   pbAtr, pcbAtrLen, FALSE);
2241
0
}
2242
2243
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardStatusW(SCARDHANDLE hCard, LPWSTR mszReaderNames,
2244
                                                          LPDWORD pcchReaderLen, LPDWORD pdwState,
2245
                                                          LPDWORD pdwProtocol, LPBYTE pbAtr,
2246
                                                          LPDWORD pcbAtrLen)
2247
0
{
2248
2249
0
  return PCSC_SCardStatus_Internal(hCard, (LPSTR)mszReaderNames, pcchReaderLen, pdwState,
2250
0
                                   pdwProtocol, pbAtr, pcbAtrLen, TRUE);
2251
0
}
2252
2253
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardTransmit(
2254
    SCARDHANDLE hCard, LPCSCARD_IO_REQUEST pioSendPci, LPCBYTE pbSendBuffer, DWORD cbSendLength,
2255
    LPSCARD_IO_REQUEST pioRecvPci, LPBYTE pbRecvBuffer, LPDWORD pcbRecvLength)
2256
0
{
2257
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2258
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2259
0
  PCSC_DWORD cbExtraBytes = 0;
2260
0
  BYTE* pbExtraBytes = nullptr;
2261
0
  BYTE* pcsc_pbExtraBytes = nullptr;
2262
0
  PCSC_DWORD pcsc_cbSendLength = (PCSC_DWORD)cbSendLength;
2263
0
  PCSC_DWORD pcsc_cbRecvLength = 0;
2264
0
  union
2265
0
  {
2266
0
    const PCSC_SCARD_IO_REQUEST* pcs;
2267
0
    PCSC_SCARD_IO_REQUEST* ps;
2268
0
    LPSCARD_IO_REQUEST lps;
2269
0
    LPCSCARD_IO_REQUEST lpcs;
2270
0
    BYTE* pb;
2271
0
  } sendPci, recvPci, inRecvPci, inSendPci;
2272
2273
0
  sendPci.ps = nullptr;
2274
0
  recvPci.ps = nullptr;
2275
0
  inRecvPci.lps = pioRecvPci;
2276
0
  inSendPci.lpcs = pioSendPci;
2277
2278
0
  if (!g_PCSC.pfnSCardTransmit)
2279
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardTransmit");
2280
2281
0
  pCard = PCSC_GetCardHandleData(hCard);
2282
2283
0
  if (!pCard)
2284
0
    return SCARD_E_INVALID_VALUE;
2285
2286
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2287
2288
0
  if (!pcbRecvLength)
2289
0
    return SCARD_E_INVALID_PARAMETER;
2290
2291
0
  if (*pcbRecvLength == SCARD_AUTOALLOCATE)
2292
0
    return SCARD_E_INVALID_PARAMETER;
2293
2294
0
  pcsc_cbRecvLength = (PCSC_DWORD)*pcbRecvLength;
2295
2296
0
  if (!inSendPci.lpcs)
2297
0
  {
2298
0
    PCSC_DWORD dwState = 0;
2299
0
    PCSC_DWORD cbAtrLen = 0;
2300
0
    PCSC_DWORD dwProtocol = 0;
2301
0
    PCSC_DWORD cchReaderLen = 0;
2302
    /**
2303
     * pcsc-lite cannot have a null pioSendPci parameter, unlike WinSCard.
2304
     * Query the current protocol and use default SCARD_IO_REQUEST for it.
2305
     */
2306
0
    status = g_PCSC.pfnSCardStatus(hCard, nullptr, &cchReaderLen, &dwState, &dwProtocol,
2307
0
                                   nullptr, &cbAtrLen);
2308
2309
0
    if (status == SCARD_S_SUCCESS)
2310
0
    {
2311
0
      if (dwProtocol == SCARD_PROTOCOL_T0)
2312
0
        sendPci.pcs = PCSC_SCARD_PCI_T0;
2313
0
      else if (dwProtocol == SCARD_PROTOCOL_T1)
2314
0
        sendPci.pcs = PCSC_SCARD_PCI_T1;
2315
0
      else if (dwProtocol == PCSC_SCARD_PROTOCOL_RAW)
2316
0
        sendPci.pcs = PCSC_SCARD_PCI_RAW;
2317
0
    }
2318
0
  }
2319
0
  else
2320
0
  {
2321
0
    cbExtraBytes = inSendPci.lpcs->cbPciLength - sizeof(SCARD_IO_REQUEST);
2322
0
    sendPci.ps = (PCSC_SCARD_IO_REQUEST*)malloc(sizeof(PCSC_SCARD_IO_REQUEST) + cbExtraBytes);
2323
2324
0
    if (!sendPci.ps)
2325
0
      return SCARD_E_NO_MEMORY;
2326
2327
0
    sendPci.ps->dwProtocol = (PCSC_DWORD)inSendPci.lpcs->dwProtocol;
2328
0
    sendPci.ps->cbPciLength = sizeof(PCSC_SCARD_IO_REQUEST) + cbExtraBytes;
2329
0
    pbExtraBytes = &(inSendPci.pb)[sizeof(SCARD_IO_REQUEST)];
2330
0
    pcsc_pbExtraBytes = &(sendPci.pb)[sizeof(PCSC_SCARD_IO_REQUEST)];
2331
0
    CopyMemory(pcsc_pbExtraBytes, pbExtraBytes, cbExtraBytes);
2332
0
  }
2333
2334
0
  if (inRecvPci.lps)
2335
0
  {
2336
0
    cbExtraBytes = inRecvPci.lps->cbPciLength - sizeof(SCARD_IO_REQUEST);
2337
0
    recvPci.ps = (PCSC_SCARD_IO_REQUEST*)malloc(sizeof(PCSC_SCARD_IO_REQUEST) + cbExtraBytes);
2338
2339
0
    if (!recvPci.ps)
2340
0
    {
2341
0
      if (inSendPci.lpcs)
2342
0
        free(sendPci.ps);
2343
2344
0
      return SCARD_E_NO_MEMORY;
2345
0
    }
2346
2347
0
    recvPci.ps->dwProtocol = (PCSC_DWORD)inRecvPci.lps->dwProtocol;
2348
0
    recvPci.ps->cbPciLength = sizeof(PCSC_SCARD_IO_REQUEST) + cbExtraBytes;
2349
0
    pbExtraBytes = &(inRecvPci.pb)[sizeof(SCARD_IO_REQUEST)];
2350
0
    pcsc_pbExtraBytes = &(recvPci.pb)[sizeof(PCSC_SCARD_IO_REQUEST)];
2351
0
    CopyMemory(pcsc_pbExtraBytes, pbExtraBytes, cbExtraBytes);
2352
0
  }
2353
2354
0
  status = g_PCSC.pfnSCardTransmit(hCard, sendPci.ps, pbSendBuffer, pcsc_cbSendLength, recvPci.ps,
2355
0
                                   pbRecvBuffer, &pcsc_cbRecvLength);
2356
2357
0
  *pcbRecvLength = (DWORD)pcsc_cbRecvLength;
2358
2359
0
  if (inSendPci.lpcs)
2360
0
    free(sendPci.ps); /* pcsc_pioSendPci is dynamically allocated only when pioSendPci is
2361
                              non null */
2362
2363
0
  if (inRecvPci.lps)
2364
0
  {
2365
0
    cbExtraBytes = inRecvPci.lps->cbPciLength - sizeof(SCARD_IO_REQUEST);
2366
0
    pbExtraBytes = &(inRecvPci.pb)[sizeof(SCARD_IO_REQUEST)];
2367
0
    pcsc_pbExtraBytes = &(recvPci.pb)[sizeof(PCSC_SCARD_IO_REQUEST)];
2368
0
    CopyMemory(pbExtraBytes, pcsc_pbExtraBytes, cbExtraBytes); /* copy extra bytes */
2369
0
    free(recvPci.ps); /* pcsc_pioRecvPci is dynamically allocated only when pioRecvPci is
2370
                              non null */
2371
0
  }
2372
2373
0
  return PCSC_MapErrorCodeToWinSCard(status);
2374
0
}
2375
2376
WINPR_ATTR_NODISCARD static LONG WINAPI
2377
PCSC_SCardGetTransmitCount(SCARDHANDLE hCard, WINPR_ATTR_UNUSED LPDWORD pcTransmitCount)
2378
0
{
2379
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2380
2381
0
  pCard = PCSC_GetCardHandleData(hCard);
2382
2383
0
  if (!pCard)
2384
0
    return SCARD_E_INVALID_VALUE;
2385
2386
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2387
0
  return SCARD_S_SUCCESS;
2388
0
}
2389
2390
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardControl(SCARDHANDLE hCard, DWORD dwControlCode,
2391
                                                          LPCVOID lpInBuffer, DWORD cbInBufferSize,
2392
                                                          LPVOID lpOutBuffer, DWORD cbOutBufferSize,
2393
                                                          LPDWORD lpBytesReturned)
2394
0
{
2395
0
  DWORD IoCtlFunction = 0;
2396
0
  DWORD IoCtlDeviceType = 0;
2397
0
  BOOL getFeatureRequest = FALSE;
2398
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2399
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2400
0
  PCSC_DWORD pcsc_dwControlCode = 0;
2401
0
  PCSC_DWORD pcsc_cbInBufferSize = (PCSC_DWORD)cbInBufferSize;
2402
0
  PCSC_DWORD pcsc_cbOutBufferSize = (PCSC_DWORD)cbOutBufferSize;
2403
0
  PCSC_DWORD pcsc_BytesReturned = 0;
2404
2405
0
  if (!g_PCSC.pfnSCardControl)
2406
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardControl");
2407
2408
0
  pCard = PCSC_GetCardHandleData(hCard);
2409
2410
0
  if (!pCard)
2411
0
    return SCARD_E_INVALID_VALUE;
2412
2413
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2414
  /**
2415
   * PCSCv2 Part 10:
2416
   * http://www.pcscworkgroup.com/specifications/files/pcsc10_v2.02.09.pdf
2417
   *
2418
   * Smart Card Driver IOCTLs:
2419
   * http://msdn.microsoft.com/en-us/library/windows/hardware/ff548988/
2420
   *
2421
   * Converting Windows Feature Request IOCTL code to the pcsc-lite control code:
2422
   * http://musclecard.996296.n3.nabble.com/Converting-Windows-Feature-Request-IOCTL-code-to-the-pcsc-lite-control-code-td4906.html
2423
   */
2424
0
  IoCtlFunction = FUNCTION_FROM_CTL_CODE(dwControlCode);
2425
0
  IoCtlDeviceType = DEVICE_TYPE_FROM_CTL_CODE(dwControlCode);
2426
2427
0
  if (dwControlCode == IOCTL_SMARTCARD_GET_FEATURE_REQUEST)
2428
0
    getFeatureRequest = TRUE;
2429
2430
0
  if (IoCtlDeviceType == FILE_DEVICE_SMARTCARD)
2431
0
    dwControlCode = PCSC_SCARD_CTL_CODE(IoCtlFunction);
2432
2433
0
  pcsc_dwControlCode = (PCSC_DWORD)dwControlCode;
2434
0
  status = g_PCSC.pfnSCardControl(hCard, pcsc_dwControlCode, lpInBuffer, pcsc_cbInBufferSize,
2435
0
                                  lpOutBuffer, pcsc_cbOutBufferSize, &pcsc_BytesReturned);
2436
2437
0
  *lpBytesReturned = (DWORD)pcsc_BytesReturned;
2438
2439
0
  if (getFeatureRequest)
2440
0
  {
2441
0
    UINT32 count = 0;
2442
0
    PCSC_TLV_STRUCTURE* tlv = (PCSC_TLV_STRUCTURE*)lpOutBuffer;
2443
2444
0
    if ((*lpBytesReturned % sizeof(PCSC_TLV_STRUCTURE)) != 0)
2445
0
      return SCARD_E_UNEXPECTED;
2446
2447
0
    count = *lpBytesReturned / sizeof(PCSC_TLV_STRUCTURE);
2448
2449
0
    for (DWORD index = 0; index < count; index++)
2450
0
    {
2451
0
      if (tlv[index].length != 4)
2452
0
        return SCARD_E_UNEXPECTED;
2453
0
    }
2454
0
  }
2455
2456
0
  return PCSC_MapErrorCodeToWinSCard(status);
2457
0
}
2458
2459
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetAttrib_Internal(SCARDHANDLE hCard,
2460
                                                                     DWORD dwAttrId, LPBYTE pbAttr,
2461
                                                                     LPDWORD pcbAttrLen)
2462
0
{
2463
0
  SCARDCONTEXT hContext = 0;
2464
0
  BOOL pcbAttrLenAlloc = FALSE;
2465
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2466
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2467
0
  PCSC_DWORD pcsc_dwAttrId = (PCSC_DWORD)dwAttrId;
2468
0
  PCSC_DWORD pcsc_cbAttrLen = 0;
2469
2470
0
  if (!g_PCSC.pfnSCardGetAttrib)
2471
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardGetAttrib");
2472
2473
0
  pCard = PCSC_GetCardHandleData(hCard);
2474
2475
0
  if (!pCard)
2476
0
    return SCARD_E_INVALID_VALUE;
2477
2478
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2479
0
  hContext = PCSC_GetCardContextFromHandle(hCard);
2480
2481
0
  if (!hContext)
2482
0
    return SCARD_E_INVALID_HANDLE;
2483
2484
0
  if (!pcbAttrLen)
2485
0
    return SCARD_E_INVALID_PARAMETER;
2486
2487
0
  if (*pcbAttrLen == SCARD_AUTOALLOCATE)
2488
0
  {
2489
0
    if (!pbAttr)
2490
0
      return SCARD_E_INVALID_PARAMETER;
2491
0
    pcbAttrLenAlloc = TRUE;
2492
0
  }
2493
2494
0
  pcsc_cbAttrLen = pcbAttrLenAlloc ? PCSC_SCARD_AUTOALLOCATE : (PCSC_DWORD)*pcbAttrLen;
2495
2496
0
  if (pcbAttrLenAlloc && !g_SCardAutoAllocate)
2497
0
  {
2498
0
    pcsc_cbAttrLen = 0;
2499
0
    status = g_PCSC.pfnSCardGetAttrib(hCard, pcsc_dwAttrId, nullptr, &pcsc_cbAttrLen);
2500
2501
0
    if (status == SCARD_S_SUCCESS)
2502
0
    {
2503
0
      BYTE* tmp = (BYTE*)calloc(1, pcsc_cbAttrLen);
2504
2505
0
      if (!tmp)
2506
0
        return SCARD_E_NO_MEMORY;
2507
2508
0
      status = g_PCSC.pfnSCardGetAttrib(hCard, pcsc_dwAttrId, tmp, &pcsc_cbAttrLen);
2509
2510
0
      if (status != SCARD_S_SUCCESS)
2511
0
      {
2512
0
        free(tmp);
2513
0
        tmp = nullptr;
2514
0
      }
2515
0
      else
2516
0
        PCSC_AddMemoryBlock(hContext, tmp);
2517
0
      *(BYTE**)pbAttr = tmp;
2518
0
    }
2519
0
  }
2520
0
  else
2521
0
  {
2522
0
    status = g_PCSC.pfnSCardGetAttrib(hCard, pcsc_dwAttrId, pbAttr, &pcsc_cbAttrLen);
2523
0
  }
2524
2525
0
  if (status == SCARD_S_SUCCESS)
2526
0
    *pcbAttrLen = (DWORD)pcsc_cbAttrLen;
2527
0
  return PCSC_MapErrorCodeToWinSCard(status);
2528
0
}
2529
2530
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetAttrib_FriendlyName(SCARDHANDLE hCard,
2531
                                                                         DWORD dwAttrId,
2532
                                                                         LPBYTE pbAttr,
2533
                                                                         LPDWORD pcbAttrLen)
2534
0
{
2535
0
  char* namePCSC = nullptr;
2536
0
  char* pbAttrA = nullptr;
2537
2538
0
  SCARDCONTEXT hContext = PCSC_GetCardContextFromHandle(hCard);
2539
2540
0
  if (!hContext)
2541
0
    return SCARD_E_INVALID_HANDLE;
2542
2543
0
  if (!pcbAttrLen)
2544
0
    return SCARD_E_INVALID_PARAMETER;
2545
0
  const DWORD cbAttrLen = *pcbAttrLen;
2546
0
  *pcbAttrLen = SCARD_AUTOALLOCATE;
2547
0
  LONG status = PCSC_SCardGetAttrib_Internal(hCard, SCARD_ATTR_DEVICE_FRIENDLY_NAME_A,
2548
0
                                             (LPBYTE)&pbAttrA, pcbAttrLen);
2549
2550
0
  if (status != SCARD_S_SUCCESS)
2551
0
  {
2552
0
    WCHAR* pbAttrW = nullptr;
2553
2554
0
    *pcbAttrLen = SCARD_AUTOALLOCATE;
2555
0
    status = PCSC_SCardGetAttrib_Internal(hCard, SCARD_ATTR_DEVICE_FRIENDLY_NAME_W,
2556
0
                                          (LPBYTE)&pbAttrW, pcbAttrLen);
2557
2558
0
    if (status != SCARD_S_SUCCESS)
2559
0
      return status;
2560
2561
0
    namePCSC = ConvertMszWCharNToUtf8Alloc(pbAttrW, *pcbAttrLen, nullptr);
2562
0
    PCSC_SCardFreeMemory_Internal(hContext, pbAttrW);
2563
0
  }
2564
0
  else
2565
0
  {
2566
0
    namePCSC = strndup(pbAttrA, *pcbAttrLen);
2567
2568
0
    if (!namePCSC)
2569
0
      return SCARD_E_NO_MEMORY;
2570
2571
0
    PCSC_SCardFreeMemory_Internal(hContext, pbAttrA);
2572
0
  }
2573
2574
0
  size_t length = strnlen(namePCSC, *pcbAttrLen);
2575
2576
0
  if (dwAttrId == SCARD_ATTR_DEVICE_FRIENDLY_NAME_W)
2577
0
  {
2578
0
    size_t size = 0;
2579
0
    WCHAR* friendlyNameW = ConvertUtf8NToWCharAlloc(namePCSC, length, &size);
2580
    /* length here includes null terminator */
2581
2582
0
    if (!friendlyNameW)
2583
0
      status = SCARD_E_NO_MEMORY;
2584
0
    else
2585
0
    {
2586
0
      length = size + 1;
2587
2588
0
      if (cbAttrLen == SCARD_AUTOALLOCATE)
2589
0
      {
2590
0
        WINPR_ASSERT(length <= UINT32_MAX / sizeof(WCHAR));
2591
0
        *(WCHAR**)pbAttr = friendlyNameW;
2592
0
        *pcbAttrLen = (UINT32)length * sizeof(WCHAR);
2593
0
        PCSC_AddMemoryBlock(hContext, friendlyNameW);
2594
0
      }
2595
0
      else
2596
0
      {
2597
0
        const size_t wlen = length * sizeof(WCHAR);
2598
0
        *pcbAttrLen = WINPR_ASSERTING_INT_CAST(uint32_t, wlen);
2599
0
        if ((wlen > cbAttrLen) && pbAttr)
2600
0
          status = SCARD_E_INSUFFICIENT_BUFFER;
2601
0
        else if (pbAttr)
2602
0
          CopyMemory(pbAttr, friendlyNameW, (length * sizeof(WCHAR)));
2603
2604
0
        free(friendlyNameW);
2605
0
      }
2606
0
    }
2607
0
    free(namePCSC);
2608
0
  }
2609
0
  else
2610
0
  {
2611
0
    length++; /* Include '\0' in length */
2612
0
    if (cbAttrLen == SCARD_AUTOALLOCATE)
2613
0
    {
2614
0
      *(CHAR**)pbAttr = namePCSC;
2615
0
      WINPR_ASSERT(length <= UINT32_MAX);
2616
0
      *pcbAttrLen = (UINT32)length;
2617
0
      PCSC_AddMemoryBlock(hContext, namePCSC);
2618
0
    }
2619
0
    else
2620
0
    {
2621
0
      *pcbAttrLen = WINPR_ASSERTING_INT_CAST(uint32_t, length);
2622
0
      if ((length > cbAttrLen) && pbAttr)
2623
0
        status = SCARD_E_INSUFFICIENT_BUFFER;
2624
0
      else if (pbAttr)
2625
0
        CopyMemory(pbAttr, namePCSC, length);
2626
2627
0
      free(namePCSC);
2628
0
    }
2629
0
  }
2630
2631
0
  return status;
2632
0
}
2633
2634
static LONG PCSC_ReadDeviceSystemNameGet(WINPR_ATTR_UNUSED SCARDCONTEXT hContext, SCARDHANDLE hCard,
2635
                                         DWORD dwAttrId, LPBYTE* pbAttr, LPDWORD pcbAttrLen)
2636
0
{
2637
0
  PCSC_DWORD cchReader = 0;
2638
0
  PCSC_DWORD cbAtr = 0;
2639
0
  PCSC_DWORD dwState = 0;
2640
0
  PCSC_DWORD dwProtocol = 0;
2641
2642
0
  const PCSC_LONG rc =
2643
0
      g_PCSC.pfnSCardStatus(hCard, nullptr, &cchReader, &dwState, &dwProtocol, nullptr, &cbAtr);
2644
0
  if (rc != SCARD_S_SUCCESS)
2645
0
    return (LONG)rc;
2646
2647
0
  void* tmp = calloc(cchReader + 1, sizeof(CHAR));
2648
0
  if (!tmp)
2649
0
    return SCARD_E_NO_MEMORY;
2650
0
  const PCSC_LONG rc2 =
2651
0
      g_PCSC.pfnSCardStatus(hCard, tmp, &cchReader, &dwState, &dwProtocol, nullptr, &cbAtr);
2652
0
  if (rc2 != SCARD_S_SUCCESS)
2653
0
  {
2654
0
    free(tmp);
2655
0
    return (LONG)rc2;
2656
0
  }
2657
2658
0
  if (dwAttrId == SCARD_ATTR_DEVICE_SYSTEM_NAME_W)
2659
0
  {
2660
0
    size_t len = 0;
2661
0
    void* data = ConvertMszUtf8NToWCharAlloc(tmp, cchReader, &len);
2662
0
    if (!data)
2663
0
    {
2664
0
      free(tmp);
2665
0
      return SCARD_E_NO_MEMORY;
2666
0
    }
2667
0
    len *= sizeof(WCHAR);
2668
2669
0
    cchReader = WINPR_ASSERTING_INT_CAST(PCSC_DWORD, len);
2670
0
    free(tmp);
2671
0
    tmp = data;
2672
0
  }
2673
2674
0
  *pbAttr = tmp;
2675
0
  *pcbAttrLen = WINPR_ASSERTING_INT_CAST(DWORD, cchReader);
2676
0
  return SCARD_S_SUCCESS;
2677
0
}
2678
2679
static LONG PCSC_ReadDeviceSystemName(WINPR_ATTR_UNUSED SCARDCONTEXT hContext, SCARDHANDLE hCard,
2680
                                      DWORD dwAttrId, LPBYTE pbAttr, LPDWORD pcbAttrLen)
2681
0
{
2682
0
  BYTE* tmp = nullptr;
2683
0
  DWORD cbAttrLen = 0;
2684
0
  const LONG rc = PCSC_ReadDeviceSystemNameGet(hContext, hCard, dwAttrId, &tmp, &cbAttrLen);
2685
0
  if (rc != SCARD_S_SUCCESS)
2686
0
    return rc;
2687
2688
0
  if (*pcbAttrLen == SCARD_AUTOALLOCATE)
2689
0
  {
2690
0
    if (!PCSC_AddMemoryBlock(hContext, tmp))
2691
0
    {
2692
0
      free(tmp);
2693
0
      return SCARD_E_NO_MEMORY;
2694
0
    }
2695
2696
0
    *pcbAttrLen = cbAttrLen;
2697
0
    *(BYTE**)pbAttr = tmp;
2698
0
    return SCARD_S_SUCCESS;
2699
0
  }
2700
2701
0
  if (pbAttr)
2702
0
    memcpy(pbAttr, tmp, MIN(cbAttrLen, *pcbAttrLen));
2703
0
  free(tmp);
2704
2705
0
  if (pbAttr)
2706
0
  {
2707
0
    if (cbAttrLen > *pcbAttrLen)
2708
0
      return SCARD_E_INSUFFICIENT_BUFFER;
2709
0
  }
2710
2711
0
  *pcbAttrLen = cbAttrLen;
2712
2713
0
  return SCARD_S_SUCCESS;
2714
0
}
2715
2716
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetAttrib(SCARDHANDLE hCard, DWORD dwAttrId,
2717
                                                            LPBYTE pbAttr, LPDWORD pcbAttrLen)
2718
0
{
2719
0
  DWORD cbAttrLen = 0;
2720
0
  SCARDCONTEXT hContext = 0;
2721
0
  BOOL pcbAttrLenAlloc = FALSE;
2722
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2723
2724
0
  if (nullptr == pcbAttrLen)
2725
0
    return SCARD_E_INVALID_PARAMETER;
2726
2727
0
  cbAttrLen = *pcbAttrLen;
2728
2729
0
  if (*pcbAttrLen == SCARD_AUTOALLOCATE)
2730
0
  {
2731
0
    if (nullptr == pbAttr)
2732
0
      return SCARD_E_INVALID_PARAMETER;
2733
2734
0
    pcbAttrLenAlloc = TRUE;
2735
0
    *(BYTE**)pbAttr = nullptr;
2736
0
  }
2737
0
  else
2738
0
  {
2739
    /**
2740
     * pcsc-lite returns SCARD_E_INSUFFICIENT_BUFFER if the given
2741
     * buffer size is larger than PCSC_MAX_BUFFER_SIZE (264)
2742
     */
2743
0
    if (*pcbAttrLen > PCSC_MAX_BUFFER_SIZE)
2744
0
      *pcbAttrLen = PCSC_MAX_BUFFER_SIZE;
2745
0
  }
2746
2747
0
  hContext = PCSC_GetCardContextFromHandle(hCard);
2748
2749
0
  if (!hContext)
2750
0
    return SCARD_E_INVALID_HANDLE;
2751
2752
0
  if ((dwAttrId == SCARD_ATTR_DEVICE_FRIENDLY_NAME_A) ||
2753
0
      (dwAttrId == SCARD_ATTR_DEVICE_FRIENDLY_NAME_W))
2754
0
  {
2755
0
    return PCSC_SCardGetAttrib_FriendlyName(hCard, dwAttrId, pbAttr, pcbAttrLen);
2756
0
  }
2757
2758
0
  status = PCSC_SCardGetAttrib_Internal(hCard, dwAttrId, pbAttr, pcbAttrLen);
2759
2760
0
  if (status == SCARD_S_SUCCESS)
2761
0
  {
2762
0
    if (dwAttrId == SCARD_ATTR_VENDOR_NAME)
2763
0
    {
2764
0
      if (pbAttr)
2765
0
      {
2766
0
        const char* vendorName = nullptr;
2767
2768
        /**
2769
         * pcsc-lite adds a null terminator to the vendor name,
2770
         * while WinSCard doesn't. Strip the null terminator.
2771
         */
2772
2773
0
        if (pcbAttrLenAlloc)
2774
0
          vendorName = (char*)*(BYTE**)pbAttr;
2775
0
        else
2776
0
          vendorName = (char*)pbAttr;
2777
2778
0
        if (vendorName)
2779
0
        {
2780
0
          size_t len = strnlen(vendorName, *pcbAttrLen);
2781
0
          WINPR_ASSERT(len < UINT32_MAX);
2782
0
          *pcbAttrLen = (DWORD)len + 1;
2783
0
        }
2784
0
        else
2785
0
          *pcbAttrLen = 0;
2786
0
      }
2787
0
    }
2788
0
  }
2789
0
  else
2790
0
  {
2791
0
    if (dwAttrId == SCARD_ATTR_CURRENT_PROTOCOL_TYPE)
2792
0
    {
2793
0
      if (!pcbAttrLenAlloc)
2794
0
      {
2795
0
        PCSC_DWORD dwState = 0;
2796
0
        PCSC_DWORD cbAtrLen = 0;
2797
0
        PCSC_DWORD dwProtocol = 0;
2798
0
        PCSC_DWORD cchReaderLen = 0;
2799
0
        status = g_PCSC.pfnSCardStatus(hCard, nullptr, &cchReaderLen, &dwState, &dwProtocol,
2800
0
                                       nullptr, &cbAtrLen);
2801
2802
0
        if (status == SCARD_S_SUCCESS)
2803
0
        {
2804
0
          if ((cbAttrLen < sizeof(DWORD)) && pbAttr)
2805
0
            return SCARD_E_INSUFFICIENT_BUFFER;
2806
2807
0
          if (pbAttr)
2808
0
            *(DWORD*)pbAttr = PCSC_ConvertProtocolsToWinSCard(dwProtocol);
2809
0
          *pcbAttrLen = sizeof(DWORD);
2810
0
        }
2811
0
      }
2812
0
    }
2813
0
    else if (dwAttrId == SCARD_ATTR_CHANNEL_ID)
2814
0
    {
2815
0
      if (!pcbAttrLenAlloc)
2816
0
      {
2817
0
        UINT32 channelType = 0x20; /* USB */
2818
0
        UINT32 channelNumber = 0;
2819
2820
0
        if ((cbAttrLen < sizeof(DWORD)) && pbAttr)
2821
0
          return SCARD_E_INSUFFICIENT_BUFFER;
2822
2823
0
        status = SCARD_S_SUCCESS;
2824
0
        if (pbAttr)
2825
0
          *(DWORD*)pbAttr = (channelType << 16u) | channelNumber;
2826
0
        *pcbAttrLen = sizeof(DWORD);
2827
0
      }
2828
0
    }
2829
0
    else if (dwAttrId == SCARD_ATTR_VENDOR_IFD_TYPE)
2830
0
    {
2831
0
    }
2832
0
    else if (dwAttrId == SCARD_ATTR_DEFAULT_CLK)
2833
0
    {
2834
0
    }
2835
0
    else if (dwAttrId == SCARD_ATTR_DEFAULT_DATA_RATE)
2836
0
    {
2837
0
    }
2838
0
    else if (dwAttrId == SCARD_ATTR_MAX_CLK)
2839
0
    {
2840
0
    }
2841
0
    else if (dwAttrId == SCARD_ATTR_MAX_DATA_RATE)
2842
0
    {
2843
0
    }
2844
0
    else if (dwAttrId == SCARD_ATTR_MAX_IFSD)
2845
0
    {
2846
0
    }
2847
0
    else if (dwAttrId == SCARD_ATTR_CHARACTERISTICS)
2848
0
    {
2849
0
    }
2850
0
    else if (dwAttrId == SCARD_ATTR_DEVICE_SYSTEM_NAME_A)
2851
0
    {
2852
0
      status = PCSC_ReadDeviceSystemName(hContext, hCard, dwAttrId, pbAttr, pcbAttrLen);
2853
0
    }
2854
0
    else if (dwAttrId == SCARD_ATTR_DEVICE_SYSTEM_NAME_W)
2855
0
    {
2856
0
      status = PCSC_ReadDeviceSystemName(hContext, hCard, dwAttrId, pbAttr, pcbAttrLen);
2857
0
    }
2858
0
    else if (dwAttrId == SCARD_ATTR_DEVICE_UNIT)
2859
0
    {
2860
0
    }
2861
0
    else if (dwAttrId == SCARD_ATTR_POWER_MGMT_SUPPORT)
2862
0
    {
2863
0
    }
2864
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_CLK)
2865
0
    {
2866
0
    }
2867
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_F)
2868
0
    {
2869
0
    }
2870
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_D)
2871
0
    {
2872
0
    }
2873
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_N)
2874
0
    {
2875
0
    }
2876
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_CWT)
2877
0
    {
2878
0
    }
2879
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_BWT)
2880
0
    {
2881
0
    }
2882
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_IFSC)
2883
0
    {
2884
0
    }
2885
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_EBC_ENCODING)
2886
0
    {
2887
0
    }
2888
0
    else if (dwAttrId == SCARD_ATTR_CURRENT_IFSD)
2889
0
    {
2890
0
    }
2891
0
    else if (dwAttrId == SCARD_ATTR_ICC_TYPE_PER_ATR)
2892
0
    {
2893
0
    }
2894
0
  }
2895
2896
0
  return WINPR_ASSERTING_INT_CAST(LONG, status);
2897
0
}
2898
2899
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardSetAttrib(SCARDHANDLE hCard, DWORD dwAttrId,
2900
                                                            LPCBYTE pbAttr, DWORD cbAttrLen)
2901
0
{
2902
0
  PCSC_LONG status = SCARD_S_SUCCESS;
2903
0
  PCSC_SCARDHANDLE* pCard = nullptr;
2904
0
  PCSC_DWORD pcsc_dwAttrId = (PCSC_DWORD)dwAttrId;
2905
0
  PCSC_DWORD pcsc_cbAttrLen = (PCSC_DWORD)cbAttrLen;
2906
2907
0
  if (!g_PCSC.pfnSCardSetAttrib)
2908
0
    return PCSC_SCard_LogError("g_PCSC.pfnSCardSetAttrib");
2909
2910
0
  pCard = PCSC_GetCardHandleData(hCard);
2911
2912
0
  if (!pCard)
2913
0
    return SCARD_E_INVALID_VALUE;
2914
2915
0
  PCSC_WaitForCardAccess(0, hCard, pCard->shared);
2916
0
  status = g_PCSC.pfnSCardSetAttrib(hCard, pcsc_dwAttrId, pbAttr, pcsc_cbAttrLen);
2917
0
  return PCSC_MapErrorCodeToWinSCard(status);
2918
0
}
2919
2920
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardUIDlgSelectCardA(LPOPENCARDNAMEA_EX pDlgStruc)
2921
0
{
2922
0
  WINPR_UNUSED(pDlgStruc);
2923
2924
0
  return SCARD_E_UNSUPPORTED_FEATURE;
2925
0
}
2926
2927
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardUIDlgSelectCardW(LPOPENCARDNAMEW_EX pDlgStruc)
2928
0
{
2929
0
  WINPR_UNUSED(pDlgStruc);
2930
0
  return SCARD_E_UNSUPPORTED_FEATURE;
2931
0
}
2932
2933
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_GetOpenCardNameA(LPOPENCARDNAMEA pDlgStruc)
2934
0
{
2935
0
  WINPR_UNUSED(pDlgStruc);
2936
0
  return SCARD_E_UNSUPPORTED_FEATURE;
2937
0
}
2938
2939
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_GetOpenCardNameW(LPOPENCARDNAMEW pDlgStruc)
2940
0
{
2941
0
  WINPR_UNUSED(pDlgStruc);
2942
0
  return SCARD_E_UNSUPPORTED_FEATURE;
2943
0
}
2944
2945
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardDlgExtendedError(void)
2946
0
{
2947
2948
0
  return SCARD_E_UNSUPPORTED_FEATURE;
2949
0
}
2950
2951
WINPR_ATTR_NODISCARD
2952
static char* card_id_and_name_a(const UUID* CardIdentifier, LPCSTR LookupName)
2953
0
{
2954
0
  WINPR_ASSERT(CardIdentifier);
2955
0
  WINPR_ASSERT(LookupName);
2956
2957
0
  size_t len = strlen(LookupName) + 34;
2958
0
  char* id = malloc(len);
2959
0
  if (!id)
2960
0
    return nullptr;
2961
2962
0
  (void)snprintf(id, len, "%08X%04X%04X%02X%02X%02X%02X%02X%02X%02X%02X\\%s",
2963
0
                 CardIdentifier->Data1, CardIdentifier->Data2, CardIdentifier->Data3,
2964
0
                 CardIdentifier->Data4[0], CardIdentifier->Data4[1], CardIdentifier->Data4[2],
2965
0
                 CardIdentifier->Data4[3], CardIdentifier->Data4[4], CardIdentifier->Data4[5],
2966
0
                 CardIdentifier->Data4[6], CardIdentifier->Data4[7], LookupName);
2967
0
  return id;
2968
0
}
2969
2970
WINPR_ATTR_NODISCARD
2971
static char* card_id_and_name_w(const UUID* CardIdentifier, LPCWSTR LookupName)
2972
0
{
2973
0
  char* res = nullptr;
2974
0
  char* tmp = ConvertWCharToUtf8Alloc(LookupName, nullptr);
2975
0
  if (!tmp)
2976
0
    return nullptr;
2977
0
  res = card_id_and_name_a(CardIdentifier, tmp);
2978
0
  free(tmp);
2979
0
  return res;
2980
0
}
2981
2982
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardReadCacheA(SCARDCONTEXT hContext,
2983
                                                             UUID* CardIdentifier,
2984
                                                             DWORD FreshnessCounter,
2985
                                                             LPSTR LookupName, PBYTE Data,
2986
                                                             DWORD* DataLen)
2987
0
{
2988
0
  PCSC_CACHE_ITEM* data = nullptr;
2989
0
  PCSC_SCARDCONTEXT* ctx = PCSC_GetCardContextData(hContext);
2990
0
  if (!ctx)
2991
0
    return SCARD_E_INVALID_HANDLE;
2992
2993
0
  char* id = card_id_and_name_a(CardIdentifier, LookupName);
2994
2995
0
  data = HashTable_GetItemValue(ctx->cache, id);
2996
0
  free(id);
2997
0
  if (!data)
2998
0
  {
2999
0
    *DataLen = 0;
3000
0
    return SCARD_W_CACHE_ITEM_NOT_FOUND;
3001
0
  }
3002
3003
0
  if (FreshnessCounter != data->freshness)
3004
0
  {
3005
0
    *DataLen = 0;
3006
0
    return SCARD_W_CACHE_ITEM_STALE;
3007
0
  }
3008
3009
0
  if (*DataLen == SCARD_AUTOALLOCATE)
3010
0
  {
3011
0
    BYTE* mem = calloc(1, data->len);
3012
0
    if (!mem)
3013
0
      return SCARD_E_NO_MEMORY;
3014
3015
0
    if (!PCSC_AddMemoryBlock(hContext, mem))
3016
0
    {
3017
0
      free(mem);
3018
0
      return SCARD_E_NO_MEMORY;
3019
0
    }
3020
3021
0
    memcpy(mem, data->data, data->len);
3022
0
    *(BYTE**)Data = mem;
3023
0
  }
3024
0
  else
3025
0
  {
3026
0
    if (data->len > *DataLen)
3027
0
      return SCARD_E_INSUFFICIENT_BUFFER;
3028
0
    memcpy(Data, data->data, data->len);
3029
0
  }
3030
3031
0
  *DataLen = data->len;
3032
0
  return SCARD_S_SUCCESS;
3033
0
}
3034
3035
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardReadCacheW(SCARDCONTEXT hContext,
3036
                                                             UUID* CardIdentifier,
3037
                                                             DWORD FreshnessCounter,
3038
                                                             LPWSTR LookupName, PBYTE Data,
3039
                                                             DWORD* DataLen)
3040
0
{
3041
0
  PCSC_CACHE_ITEM* data = nullptr;
3042
0
  PCSC_SCARDCONTEXT* ctx = PCSC_GetCardContextData(hContext);
3043
0
  if (!ctx)
3044
0
    return SCARD_E_INVALID_HANDLE;
3045
3046
0
  char* id = card_id_and_name_w(CardIdentifier, LookupName);
3047
3048
0
  data = HashTable_GetItemValue(ctx->cache, id);
3049
0
  free(id);
3050
3051
0
  if (!data)
3052
0
  {
3053
0
    *DataLen = 0;
3054
0
    return SCARD_W_CACHE_ITEM_NOT_FOUND;
3055
0
  }
3056
3057
0
  if (FreshnessCounter != data->freshness)
3058
0
  {
3059
0
    *DataLen = 0;
3060
0
    return SCARD_W_CACHE_ITEM_STALE;
3061
0
  }
3062
3063
0
  if (*DataLen == SCARD_AUTOALLOCATE)
3064
0
  {
3065
0
    BYTE* mem = calloc(1, data->len);
3066
0
    if (!mem)
3067
0
      return SCARD_E_NO_MEMORY;
3068
3069
0
    if (!PCSC_AddMemoryBlock(hContext, mem))
3070
0
    {
3071
0
      free(mem);
3072
0
      return SCARD_E_NO_MEMORY;
3073
0
    }
3074
3075
0
    memcpy(mem, data->data, data->len);
3076
0
    *(BYTE**)Data = mem;
3077
0
  }
3078
0
  else
3079
0
  {
3080
0
    if (data->len > *DataLen)
3081
0
      return SCARD_E_INSUFFICIENT_BUFFER;
3082
0
    memcpy(Data, data->data, data->len);
3083
0
  }
3084
0
  *DataLen = data->len;
3085
0
  return SCARD_S_SUCCESS;
3086
0
}
3087
3088
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardWriteCacheA(SCARDCONTEXT hContext,
3089
                                                              UUID* CardIdentifier,
3090
                                                              DWORD FreshnessCounter,
3091
                                                              LPSTR LookupName, PBYTE Data,
3092
                                                              DWORD DataLen)
3093
0
{
3094
0
  PCSC_CACHE_ITEM* data = nullptr;
3095
0
  PCSC_SCARDCONTEXT* ctx = PCSC_GetCardContextData(hContext);
3096
0
  char* id = nullptr;
3097
3098
0
  if (!ctx)
3099
0
    return SCARD_E_FILE_NOT_FOUND;
3100
3101
0
  id = card_id_and_name_a(CardIdentifier, LookupName);
3102
3103
0
  if (!id)
3104
0
    return SCARD_E_NO_MEMORY;
3105
3106
0
  data = malloc(sizeof(PCSC_CACHE_ITEM));
3107
0
  if (!data)
3108
0
  {
3109
0
    free(id);
3110
0
    return SCARD_E_NO_MEMORY;
3111
0
  }
3112
0
  data->data = calloc(DataLen, 1);
3113
0
  if (!data->data)
3114
0
  {
3115
0
    free(id);
3116
0
    free(data);
3117
0
    return SCARD_E_NO_MEMORY;
3118
0
  }
3119
0
  data->len = DataLen;
3120
0
  data->freshness = FreshnessCounter;
3121
0
  memcpy(data->data, Data, data->len);
3122
3123
0
  HashTable_Remove(ctx->cache, id);
3124
0
  const BOOL rc = HashTable_Insert(ctx->cache, id, data);
3125
0
  free(id);
3126
3127
0
  if (!rc)
3128
0
  {
3129
0
    pcsc_cache_item_free(data);
3130
0
    return SCARD_E_NO_MEMORY;
3131
0
  }
3132
3133
  // NOLINTNEXTLINE(clang-analyzer-unix.Malloc): HashTable_Insert owns data
3134
0
  return SCARD_S_SUCCESS;
3135
0
}
3136
3137
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardWriteCacheW(SCARDCONTEXT hContext,
3138
                                                              UUID* CardIdentifier,
3139
                                                              DWORD FreshnessCounter,
3140
                                                              LPWSTR LookupName, PBYTE Data,
3141
                                                              DWORD DataLen)
3142
0
{
3143
0
  PCSC_CACHE_ITEM* data = nullptr;
3144
0
  PCSC_SCARDCONTEXT* ctx = PCSC_GetCardContextData(hContext);
3145
0
  char* id = nullptr;
3146
0
  if (!ctx)
3147
0
    return SCARD_E_FILE_NOT_FOUND;
3148
3149
0
  id = card_id_and_name_w(CardIdentifier, LookupName);
3150
3151
0
  if (!id)
3152
0
    return SCARD_E_NO_MEMORY;
3153
3154
0
  data = malloc(sizeof(PCSC_CACHE_ITEM));
3155
0
  if (!data)
3156
0
  {
3157
0
    free(id);
3158
0
    return SCARD_E_NO_MEMORY;
3159
0
  }
3160
0
  data->data = malloc(DataLen);
3161
0
  if (!data->data)
3162
0
  {
3163
0
    free(id);
3164
0
    free(data);
3165
0
    return SCARD_E_NO_MEMORY;
3166
0
  }
3167
0
  data->len = DataLen;
3168
0
  data->freshness = FreshnessCounter;
3169
0
  memcpy(data->data, Data, data->len);
3170
3171
0
  HashTable_Remove(ctx->cache, id);
3172
0
  const BOOL rc = HashTable_Insert(ctx->cache, id, data);
3173
0
  free(id);
3174
3175
0
  if (!rc)
3176
0
  {
3177
0
    pcsc_cache_item_free(data);
3178
0
    return SCARD_E_NO_MEMORY;
3179
0
  }
3180
3181
  // NOLINTNEXTLINE(clang-analyzer-unix.Malloc): HashTable_Insert owns data
3182
0
  return SCARD_S_SUCCESS;
3183
0
}
3184
3185
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetReaderIconA(
3186
    SCARDCONTEXT hContext, LPCSTR szReaderName,
3187
    LPBYTE pbIcon /* NOLINT(readability-non-const-parameter) */, LPDWORD pcbIcon)
3188
0
{
3189
0
  WINPR_UNUSED(hContext);
3190
0
  WINPR_UNUSED(szReaderName);
3191
0
  WINPR_UNUSED(pbIcon);
3192
0
  WINPR_ASSERT(pcbIcon);
3193
0
  *pcbIcon = 0;
3194
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3195
0
}
3196
3197
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetReaderIconW(
3198
    SCARDCONTEXT hContext, LPCWSTR szReaderName,
3199
    LPBYTE pbIcon /* NOLINT(readability-non-const-parameter) */, LPDWORD pcbIcon)
3200
0
{
3201
0
  WINPR_UNUSED(hContext);
3202
0
  WINPR_UNUSED(szReaderName);
3203
0
  WINPR_UNUSED(pbIcon);
3204
0
  WINPR_ASSERT(pcbIcon);
3205
0
  *pcbIcon = 0;
3206
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3207
0
}
3208
3209
WINPR_ATTR_NODISCARD static LONG WINAPI PCSC_SCardGetDeviceTypeIdA(SCARDCONTEXT hContext,
3210
                                                                   LPCSTR szReaderName,
3211
                                                                   LPDWORD pdwDeviceTypeId)
3212
0
{
3213
0
  WINPR_UNUSED(hContext);
3214
0
  WINPR_UNUSED(szReaderName);
3215
0
  WINPR_UNUSED(pdwDeviceTypeId);
3216
0
  if (pdwDeviceTypeId)
3217
0
    *pdwDeviceTypeId = SCARD_READER_TYPE_USB;
3218
0
  return SCARD_S_SUCCESS;
3219
0
}
3220
3221
WINPR_ATTR_NODISCARD
3222
static LONG WINAPI PCSC_SCardGetDeviceTypeIdW(SCARDCONTEXT hContext, LPCWSTR szReaderName,
3223
                                              LPDWORD pdwDeviceTypeId)
3224
0
{
3225
0
  WINPR_UNUSED(hContext);
3226
0
  WINPR_UNUSED(szReaderName);
3227
0
  if (pdwDeviceTypeId)
3228
0
    *pdwDeviceTypeId = SCARD_READER_TYPE_USB;
3229
0
  return SCARD_S_SUCCESS;
3230
0
}
3231
3232
WINPR_ATTR_NODISCARD
3233
static LONG WINAPI PCSC_SCardGetReaderDeviceInstanceIdA(
3234
    SCARDCONTEXT hContext, LPCSTR szReaderName,
3235
    LPSTR szDeviceInstanceId /* NOLINT(readability-non-const-parameter) */,
3236
    LPDWORD pcchDeviceInstanceId /* NOLINT(readability-non-const-parameter) */)
3237
0
{
3238
0
  WINPR_UNUSED(hContext);
3239
0
  WINPR_UNUSED(szReaderName);
3240
0
  WINPR_UNUSED(szDeviceInstanceId);
3241
0
  WINPR_UNUSED(pcchDeviceInstanceId);
3242
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3243
0
}
3244
3245
WINPR_ATTR_NODISCARD
3246
static LONG WINAPI PCSC_SCardGetReaderDeviceInstanceIdW(
3247
    SCARDCONTEXT hContext, LPCWSTR szReaderName,
3248
    LPWSTR szDeviceInstanceId /* NOLINT(readability-non-const-parameter) */,
3249
    LPDWORD pcchDeviceInstanceId /* NOLINT(readability-non-const-parameter) */)
3250
0
{
3251
0
  WINPR_UNUSED(hContext);
3252
0
  WINPR_UNUSED(szReaderName);
3253
0
  WINPR_UNUSED(szDeviceInstanceId);
3254
0
  WINPR_UNUSED(pcchDeviceInstanceId);
3255
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3256
0
}
3257
3258
WINPR_ATTR_NODISCARD
3259
static LONG WINAPI PCSC_SCardListReadersWithDeviceInstanceIdA(
3260
    SCARDCONTEXT hContext, LPCSTR szDeviceInstanceId,
3261
    LPSTR mszReaders /* NOLINT(readability-non-const-parameter) */,
3262
    LPDWORD pcchReaders /* NOLINT(readability-non-const-parameter) */)
3263
0
{
3264
0
  WINPR_UNUSED(hContext);
3265
0
  WINPR_UNUSED(szDeviceInstanceId);
3266
0
  WINPR_UNUSED(mszReaders);
3267
0
  WINPR_UNUSED(pcchReaders);
3268
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3269
0
}
3270
3271
WINPR_ATTR_NODISCARD
3272
static LONG WINAPI PCSC_SCardListReadersWithDeviceInstanceIdW(
3273
    SCARDCONTEXT hContext, LPCWSTR szDeviceInstanceId,
3274
    LPWSTR mszReaders /* NOLINT(readability-non-const-parameter) */,
3275
    LPDWORD pcchReaders /* NOLINT(readability-non-const-parameter) */)
3276
0
{
3277
0
  WINPR_UNUSED(hContext);
3278
0
  WINPR_UNUSED(szDeviceInstanceId);
3279
0
  WINPR_UNUSED(mszReaders);
3280
0
  WINPR_UNUSED(pcchReaders);
3281
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3282
0
}
3283
3284
WINPR_ATTR_NODISCARD
3285
static LONG WINAPI PCSC_SCardAudit(SCARDCONTEXT hContext, DWORD dwEvent)
3286
0
{
3287
3288
0
  WINPR_UNUSED(hContext);
3289
0
  WINPR_UNUSED(dwEvent);
3290
0
  return SCARD_E_UNSUPPORTED_FEATURE;
3291
0
}
3292
3293
#ifdef __MACOSX__
3294
WINPR_ATTR_NODISCARD
3295
unsigned int determineMacOSXVersion(void)
3296
{
3297
  int mib[2];
3298
  size_t len = 0;
3299
  char* kernelVersion = nullptr;
3300
  char* tok = nullptr;
3301
  unsigned int version = 0;
3302
  long majorVersion = 0;
3303
  long minorVersion = 0;
3304
  long patchVersion = 0;
3305
  int count = 0;
3306
  char* context = nullptr;
3307
  mib[0] = CTL_KERN;
3308
  mib[1] = KERN_OSRELEASE;
3309
3310
  if (sysctl(mib, 2, nullptr, &len, nullptr, 0) != 0)
3311
    return 0;
3312
3313
  kernelVersion = calloc(len, sizeof(char));
3314
3315
  if (!kernelVersion)
3316
    return 0;
3317
3318
  if (sysctl(mib, 2, kernelVersion, &len, nullptr, 0) != 0)
3319
  {
3320
    free(kernelVersion);
3321
    return 0;
3322
  }
3323
3324
  tok = strtok_s(kernelVersion, ".", &context);
3325
  errno = 0;
3326
3327
  while (tok)
3328
  {
3329
    switch (count)
3330
    {
3331
      case 0:
3332
        majorVersion = strtol(tok, nullptr, 0);
3333
3334
        if (errno != 0)
3335
          goto fail;
3336
3337
        break;
3338
3339
      case 1:
3340
        minorVersion = strtol(tok, nullptr, 0);
3341
3342
        if (errno != 0)
3343
          goto fail;
3344
3345
        break;
3346
3347
      case 2:
3348
        patchVersion = strtol(tok, nullptr, 0);
3349
3350
        if (errno != 0)
3351
          goto fail;
3352
3353
        break;
3354
    }
3355
3356
    tok = strtok_s(nullptr, ".", &context);
3357
    count++;
3358
  }
3359
3360
  /**
3361
   * Source : http://en.wikipedia.org/wiki/Darwin_(operating_system)
3362
   **/
3363
  if (majorVersion < 5)
3364
  {
3365
    if (minorVersion < 4)
3366
      version = 0x10000000;
3367
    else
3368
      version = 0x10010000;
3369
  }
3370
  else
3371
  {
3372
    switch (majorVersion)
3373
    {
3374
      case 5:
3375
        version = 0x10010000;
3376
        break;
3377
3378
      case 6:
3379
        version = 0x10020000;
3380
        break;
3381
3382
      case 7:
3383
        version = 0x10030000;
3384
        break;
3385
3386
      case 8:
3387
        version = 0x10040000;
3388
        break;
3389
3390
      case 9:
3391
        version = 0x10050000;
3392
        break;
3393
3394
      case 10:
3395
        version = 0x10060000;
3396
        break;
3397
3398
      case 11:
3399
        version = 0x10070000;
3400
        break;
3401
3402
      case 12:
3403
        version = 0x10080000;
3404
        break;
3405
3406
      case 13:
3407
        version = 0x10090000;
3408
        break;
3409
3410
      default:
3411
        version = 0x10100000;
3412
        break;
3413
    }
3414
3415
    version |= (minorVersion << 8) | (patchVersion);
3416
  }
3417
3418
fail:
3419
  free(kernelVersion);
3420
  return version;
3421
}
3422
#endif
3423
3424
static const SCardApiFunctionTable PCSC_SCardApiFunctionTable = {
3425
  0, /* dwVersion */
3426
  0, /* dwFlags */
3427
3428
  PCSC_SCardEstablishContext,                 /* SCardEstablishContext */
3429
  PCSC_SCardReleaseContext,                   /* SCardReleaseContext */
3430
  PCSC_SCardIsValidContext,                   /* SCardIsValidContext */
3431
  PCSC_SCardListReaderGroupsA,                /* SCardListReaderGroupsA */
3432
  PCSC_SCardListReaderGroupsW,                /* SCardListReaderGroupsW */
3433
  PCSC_SCardListReadersA,                     /* SCardListReadersA */
3434
  PCSC_SCardListReadersW,                     /* SCardListReadersW */
3435
  PCSC_SCardListCardsA,                       /* SCardListCardsA */
3436
  PCSC_SCardListCardsW,                       /* SCardListCardsW */
3437
  PCSC_SCardListInterfacesA,                  /* SCardListInterfacesA */
3438
  PCSC_SCardListInterfacesW,                  /* SCardListInterfacesW */
3439
  PCSC_SCardGetProviderIdA,                   /* SCardGetProviderIdA */
3440
  PCSC_SCardGetProviderIdW,                   /* SCardGetProviderIdW */
3441
  PCSC_SCardGetCardTypeProviderNameA,         /* SCardGetCardTypeProviderNameA */
3442
  PCSC_SCardGetCardTypeProviderNameW,         /* SCardGetCardTypeProviderNameW */
3443
  PCSC_SCardIntroduceReaderGroupA,            /* SCardIntroduceReaderGroupA */
3444
  PCSC_SCardIntroduceReaderGroupW,            /* SCardIntroduceReaderGroupW */
3445
  PCSC_SCardForgetReaderGroupA,               /* SCardForgetReaderGroupA */
3446
  PCSC_SCardForgetReaderGroupW,               /* SCardForgetReaderGroupW */
3447
  PCSC_SCardIntroduceReaderA,                 /* SCardIntroduceReaderA */
3448
  PCSC_SCardIntroduceReaderW,                 /* SCardIntroduceReaderW */
3449
  PCSC_SCardForgetReaderA,                    /* SCardForgetReaderA */
3450
  PCSC_SCardForgetReaderW,                    /* SCardForgetReaderW */
3451
  PCSC_SCardAddReaderToGroupA,                /* SCardAddReaderToGroupA */
3452
  PCSC_SCardAddReaderToGroupW,                /* SCardAddReaderToGroupW */
3453
  PCSC_SCardRemoveReaderFromGroupA,           /* SCardRemoveReaderFromGroupA */
3454
  PCSC_SCardRemoveReaderFromGroupW,           /* SCardRemoveReaderFromGroupW */
3455
  PCSC_SCardIntroduceCardTypeA,               /* SCardIntroduceCardTypeA */
3456
  PCSC_SCardIntroduceCardTypeW,               /* SCardIntroduceCardTypeW */
3457
  PCSC_SCardSetCardTypeProviderNameA,         /* SCardSetCardTypeProviderNameA */
3458
  PCSC_SCardSetCardTypeProviderNameW,         /* SCardSetCardTypeProviderNameW */
3459
  PCSC_SCardForgetCardTypeA,                  /* SCardForgetCardTypeA */
3460
  PCSC_SCardForgetCardTypeW,                  /* SCardForgetCardTypeW */
3461
  PCSC_SCardFreeMemory,                       /* SCardFreeMemory */
3462
  PCSC_SCardAccessStartedEvent,               /* SCardAccessStartedEvent */
3463
  PCSC_SCardReleaseStartedEvent,              /* SCardReleaseStartedEvent */
3464
  PCSC_SCardLocateCardsA,                     /* SCardLocateCardsA */
3465
  PCSC_SCardLocateCardsW,                     /* SCardLocateCardsW */
3466
  PCSC_SCardLocateCardsByATRA,                /* SCardLocateCardsByATRA */
3467
  PCSC_SCardLocateCardsByATRW,                /* SCardLocateCardsByATRW */
3468
  PCSC_SCardGetStatusChangeA,                 /* SCardGetStatusChangeA */
3469
  PCSC_SCardGetStatusChangeW,                 /* SCardGetStatusChangeW */
3470
  PCSC_SCardCancel,                           /* SCardCancel */
3471
  PCSC_SCardConnectA,                         /* SCardConnectA */
3472
  PCSC_SCardConnectW,                         /* SCardConnectW */
3473
  PCSC_SCardReconnect,                        /* SCardReconnect */
3474
  PCSC_SCardDisconnect,                       /* SCardDisconnect */
3475
  PCSC_SCardBeginTransaction,                 /* SCardBeginTransaction */
3476
  PCSC_SCardEndTransaction,                   /* SCardEndTransaction */
3477
  PCSC_SCardCancelTransaction,                /* SCardCancelTransaction */
3478
  PCSC_SCardState,                            /* SCardState */
3479
  PCSC_SCardStatusA,                          /* SCardStatusA */
3480
  PCSC_SCardStatusW,                          /* SCardStatusW */
3481
  PCSC_SCardTransmit,                         /* SCardTransmit */
3482
  PCSC_SCardGetTransmitCount,                 /* SCardGetTransmitCount */
3483
  PCSC_SCardControl,                          /* SCardControl */
3484
  PCSC_SCardGetAttrib,                        /* SCardGetAttrib */
3485
  PCSC_SCardSetAttrib,                        /* SCardSetAttrib */
3486
  PCSC_SCardUIDlgSelectCardA,                 /* SCardUIDlgSelectCardA */
3487
  PCSC_SCardUIDlgSelectCardW,                 /* SCardUIDlgSelectCardW */
3488
  PCSC_GetOpenCardNameA,                      /* GetOpenCardNameA */
3489
  PCSC_GetOpenCardNameW,                      /* GetOpenCardNameW */
3490
  PCSC_SCardDlgExtendedError,                 /* SCardDlgExtendedError */
3491
  PCSC_SCardReadCacheA,                       /* SCardReadCacheA */
3492
  PCSC_SCardReadCacheW,                       /* SCardReadCacheW */
3493
  PCSC_SCardWriteCacheA,                      /* SCardWriteCacheA */
3494
  PCSC_SCardWriteCacheW,                      /* SCardWriteCacheW */
3495
  PCSC_SCardGetReaderIconA,                   /* SCardGetReaderIconA */
3496
  PCSC_SCardGetReaderIconW,                   /* SCardGetReaderIconW */
3497
  PCSC_SCardGetDeviceTypeIdA,                 /* SCardGetDeviceTypeIdA */
3498
  PCSC_SCardGetDeviceTypeIdW,                 /* SCardGetDeviceTypeIdW */
3499
  PCSC_SCardGetReaderDeviceInstanceIdA,       /* SCardGetReaderDeviceInstanceIdA */
3500
  PCSC_SCardGetReaderDeviceInstanceIdW,       /* SCardGetReaderDeviceInstanceIdW */
3501
  PCSC_SCardListReadersWithDeviceInstanceIdA, /* SCardListReadersWithDeviceInstanceIdA */
3502
  PCSC_SCardListReadersWithDeviceInstanceIdW, /* SCardListReadersWithDeviceInstanceIdW */
3503
  PCSC_SCardAudit                             /* SCardAudit */
3504
};
3505
3506
const SCardApiFunctionTable* PCSC_GetSCardApiFunctionTable(void)
3507
0
{
3508
0
  return &PCSC_SCardApiFunctionTable;
3509
0
}
3510
3511
int PCSC_InitializeSCardApi(void)
3512
0
{
3513
  /* Disable pcsc-lite's (poor) blocking so we can handle it ourselves */
3514
0
  SetEnvironmentVariableA("PCSCLITE_NO_BLOCKING", "1");
3515
#ifdef __MACOSX__
3516
  g_PCSCModule = LoadLibraryX("/System/Library/Frameworks/PCSC.framework/PCSC");
3517
  OSXVersion = determineMacOSXVersion();
3518
3519
  if (OSXVersion == 0)
3520
    return -1;
3521
3522
#else
3523
0
  g_PCSCModule = LoadLibraryA("libpcsclite.so.1");
3524
3525
0
  if (!g_PCSCModule)
3526
0
    g_PCSCModule = LoadLibraryA("libpcsclite.so");
3527
3528
0
#endif
3529
3530
0
  if (!g_PCSCModule)
3531
0
    return -1;
3532
3533
    /* symbols defined in winpr/smartcard.h, might pose an issue with the GetProcAddress macro
3534
     * below. therefore undefine them here */
3535
0
#undef SCardListReaderGroups
3536
0
#undef SCardListReaders
3537
0
#undef SCardListCards
3538
0
#undef SCardListInterfaces
3539
0
#undef SCardGetProviderId
3540
0
#undef SCardGetCardTypeProviderName
3541
0
#undef SCardIntroduceReaderGroup
3542
0
#undef SCardForgetReaderGroup
3543
0
#undef SCardIntroduceReader
3544
0
#undef SCardForgetReader
3545
0
#undef SCardAddReaderToGroup
3546
0
#undef SCardRemoveReaderFromGroup
3547
0
#undef SCardIntroduceCardType
3548
0
#undef SCardSetCardTypeProviderName
3549
0
#undef SCardForgetCardType
3550
0
#undef SCardLocateCards
3551
0
#undef SCardLocateCardsByATR
3552
0
#undef SCardGetStatusChange
3553
0
#undef SCardConnect
3554
0
#undef SCardStatus
3555
0
#undef SCardUIDlgSelectCard
3556
0
#undef GetOpenCardName
3557
0
#undef SCardReadCache
3558
0
#undef SCardWriteCache
3559
0
#undef SCardGetReaderIcon
3560
0
#undef SCardGetDeviceTypeId
3561
0
#undef SCardGetReaderDeviceInstanceId
3562
0
#undef SCardListReadersWithDeviceInstanceId
3563
3564
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardEstablishContext);
3565
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardReleaseContext);
3566
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardIsValidContext);
3567
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardConnect);
3568
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardReconnect);
3569
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardDisconnect);
3570
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardBeginTransaction);
3571
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardEndTransaction);
3572
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardStatus);
3573
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardGetStatusChange);
3574
3575
#ifdef __MACOSX__
3576
3577
  if (OSXVersion >= 0x10050600)
3578
  {
3579
    WINSCARD_LOAD_PROC_EX(g_PCSCModule, g_PCSC, SCardControl, SCardControl132);
3580
  }
3581
  else
3582
  {
3583
    WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardControl);
3584
  }
3585
#else
3586
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardControl);
3587
0
#endif
3588
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardTransmit);
3589
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardListReaderGroups);
3590
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardListReaders);
3591
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardCancel);
3592
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardGetAttrib);
3593
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardSetAttrib);
3594
0
  g_PCSC.pfnSCardFreeMemory = nullptr;
3595
0
#ifndef __APPLE__
3596
0
  WINSCARD_LOAD_PROC(g_PCSCModule, g_PCSC, SCardFreeMemory);
3597
0
#endif
3598
3599
0
  if (g_PCSC.pfnSCardFreeMemory)
3600
0
    g_SCardAutoAllocate = TRUE;
3601
3602
#ifdef DISABLE_PCSC_SCARD_AUTOALLOCATE
3603
  g_PCSC.pfnSCardFreeMemory = nullptr;
3604
  g_SCardAutoAllocate = FALSE;
3605
#endif
3606
#ifdef __APPLE__
3607
  g_PnP_Notification = FALSE;
3608
#endif
3609
0
  return 1;
3610
0
}
3611
3612
#endif