Coverage Report

Created: 2026-04-12 07:03

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/libfreerdp/utils/rdpdr_utils.c
Line
Count
Source
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * SCard utility functions
4
 *
5
 * Copyright 2021 Armin Novak <armin.novak@thincast.com>
6
 * Copyright 2021 Thincast Technologies GmbH
7
 *
8
 * Licensed under the Apache License, Version 2.0 (the "License");
9
 * you may not use this file except in compliance with the License.
10
 * You may obtain a copy of the License at
11
 *
12
 *     http://www.apache.org/licenses/LICENSE-2.0
13
 *
14
 * Unless required by applicable law or agreed to in writing, software
15
 * distributed under the License is distributed on an "AS IS" BASIS,
16
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17
 * See the License for the specific language governing permissions and
18
 * limitations under the License.
19
 */
20
21
#include <winpr/wlog.h>
22
#include <winpr/print.h>
23
#include <winpr/smartcard.h>
24
25
#include <freerdp/utils/rdpdr_utils.h>
26
#include <freerdp/channels/scard.h>
27
#include <freerdp/channels/rdpdr.h>
28
29
#include <freerdp/log.h>
30
31
LONG scard_log_status_error(const char* tag, const char* what, LONG status)
32
0
{
33
0
  wLog* log = WLog_Get(tag);
34
0
  return scard_log_status_error_wlog(log, "%s", status, what);
35
0
}
36
37
LONG scard_log_status_error_wlog(wLog* log, const char* what, LONG status, ...)
38
0
{
39
0
  if (status != SCARD_S_SUCCESS)
40
0
  {
41
0
    DWORD level = WLOG_ERROR;
42
0
    switch (status)
43
0
    {
44
0
      case SCARD_W_RESET_CARD:
45
0
      case SCARD_E_NOT_TRANSACTED:
46
0
      case SCARD_E_CANCELLED:
47
0
      case SCARD_E_UNSUPPORTED_FEATURE:
48
0
      case SCARD_E_TIMEOUT:
49
0
        level = WLOG_DEBUG;
50
0
        break;
51
0
      case SCARD_E_NO_READERS_AVAILABLE:
52
0
        level = WLOG_INFO;
53
0
        break;
54
0
      default:
55
0
        break;
56
0
    }
57
58
0
    char* str = nullptr;
59
0
    size_t slen = 0;
60
0
    va_list ap = WINPR_C_ARRAY_INIT;
61
0
    va_start(ap, status);
62
0
    winpr_vasprintf(&str, &slen, what, ap);
63
0
    va_end(ap);
64
0
    WLog_Print(log, level, "%s failed with error %s [%" PRId32 "]", str,
65
0
               SCardGetErrorString(status), status);
66
0
    free(str);
67
0
  }
68
0
  return status;
69
0
}
70
71
const char* scard_get_ioctl_string(UINT32 ioControlCode, BOOL funcName)
72
0
{
73
0
  switch (ioControlCode)
74
0
  {
75
0
    case SCARD_IOCTL_ESTABLISHCONTEXT:
76
0
      return funcName ? "SCardEstablishContext" : "SCARD_IOCTL_ESTABLISHCONTEXT";
77
78
0
    case SCARD_IOCTL_RELEASECONTEXT:
79
0
      return funcName ? "SCardReleaseContext" : "SCARD_IOCTL_RELEASECONTEXT";
80
81
0
    case SCARD_IOCTL_ISVALIDCONTEXT:
82
0
      return funcName ? "SCardIsValidContext" : "SCARD_IOCTL_ISVALIDCONTEXT";
83
84
0
    case SCARD_IOCTL_LISTREADERGROUPSA:
85
0
      return funcName ? "SCardListReaderGroupsA" : "SCARD_IOCTL_LISTREADERGROUPSA";
86
87
0
    case SCARD_IOCTL_LISTREADERGROUPSW:
88
0
      return funcName ? "SCardListReaderGroupsW" : "SCARD_IOCTL_LISTREADERGROUPSW";
89
90
0
    case SCARD_IOCTL_LISTREADERSA:
91
0
      return funcName ? "SCardListReadersA" : "SCARD_IOCTL_LISTREADERSA";
92
93
0
    case SCARD_IOCTL_LISTREADERSW:
94
0
      return funcName ? "SCardListReadersW" : "SCARD_IOCTL_LISTREADERSW";
95
96
0
    case SCARD_IOCTL_INTRODUCEREADERGROUPA:
97
0
      return funcName ? "SCardIntroduceReaderGroupA" : "SCARD_IOCTL_INTRODUCEREADERGROUPA";
98
99
0
    case SCARD_IOCTL_INTRODUCEREADERGROUPW:
100
0
      return funcName ? "SCardIntroduceReaderGroupW" : "SCARD_IOCTL_INTRODUCEREADERGROUPW";
101
102
0
    case SCARD_IOCTL_FORGETREADERGROUPA:
103
0
      return funcName ? "SCardForgetReaderGroupA" : "SCARD_IOCTL_FORGETREADERGROUPA";
104
105
0
    case SCARD_IOCTL_FORGETREADERGROUPW:
106
0
      return funcName ? "SCardForgetReaderGroupW" : "SCARD_IOCTL_FORGETREADERGROUPW";
107
108
0
    case SCARD_IOCTL_INTRODUCEREADERA:
109
0
      return funcName ? "SCardIntroduceReaderA" : "SCARD_IOCTL_INTRODUCEREADERA";
110
111
0
    case SCARD_IOCTL_INTRODUCEREADERW:
112
0
      return funcName ? "SCardIntroduceReaderW" : "SCARD_IOCTL_INTRODUCEREADERW";
113
114
0
    case SCARD_IOCTL_FORGETREADERA:
115
0
      return funcName ? "SCardForgetReaderA" : "SCARD_IOCTL_FORGETREADERA";
116
117
0
    case SCARD_IOCTL_FORGETREADERW:
118
0
      return funcName ? "SCardForgetReaderW" : "SCARD_IOCTL_FORGETREADERW";
119
120
0
    case SCARD_IOCTL_ADDREADERTOGROUPA:
121
0
      return funcName ? "SCardAddReaderToGroupA" : "SCARD_IOCTL_ADDREADERTOGROUPA";
122
123
0
    case SCARD_IOCTL_ADDREADERTOGROUPW:
124
0
      return funcName ? "SCardAddReaderToGroupW" : "SCARD_IOCTL_ADDREADERTOGROUPW";
125
126
0
    case SCARD_IOCTL_REMOVEREADERFROMGROUPA:
127
0
      return funcName ? "SCardRemoveReaderFromGroupA" : "SCARD_IOCTL_REMOVEREADERFROMGROUPA";
128
129
0
    case SCARD_IOCTL_REMOVEREADERFROMGROUPW:
130
0
      return funcName ? "SCardRemoveReaderFromGroupW" : "SCARD_IOCTL_REMOVEREADERFROMGROUPW";
131
132
0
    case SCARD_IOCTL_LOCATECARDSA:
133
0
      return funcName ? "SCardLocateCardsA" : "SCARD_IOCTL_LOCATECARDSA";
134
135
0
    case SCARD_IOCTL_LOCATECARDSW:
136
0
      return funcName ? "SCardLocateCardsW" : "SCARD_IOCTL_LOCATECARDSW";
137
138
0
    case SCARD_IOCTL_GETSTATUSCHANGEA:
139
0
      return funcName ? "SCardGetStatusChangeA" : "SCARD_IOCTL_GETSTATUSCHANGEA";
140
141
0
    case SCARD_IOCTL_GETSTATUSCHANGEW:
142
0
      return funcName ? "SCardGetStatusChangeW" : "SCARD_IOCTL_GETSTATUSCHANGEW";
143
144
0
    case SCARD_IOCTL_CANCEL:
145
0
      return funcName ? "SCardCancel" : "SCARD_IOCTL_CANCEL";
146
147
0
    case SCARD_IOCTL_CONNECTA:
148
0
      return funcName ? "SCardConnectA" : "SCARD_IOCTL_CONNECTA";
149
150
0
    case SCARD_IOCTL_CONNECTW:
151
0
      return funcName ? "SCardConnectW" : "SCARD_IOCTL_CONNECTW";
152
153
0
    case SCARD_IOCTL_RECONNECT:
154
0
      return funcName ? "SCardReconnect" : "SCARD_IOCTL_RECONNECT";
155
156
0
    case SCARD_IOCTL_DISCONNECT:
157
0
      return funcName ? "SCardDisconnect" : "SCARD_IOCTL_DISCONNECT";
158
159
0
    case SCARD_IOCTL_BEGINTRANSACTION:
160
0
      return funcName ? "SCardBeginTransaction" : "SCARD_IOCTL_BEGINTRANSACTION";
161
162
0
    case SCARD_IOCTL_ENDTRANSACTION:
163
0
      return funcName ? "SCardEndTransaction" : "SCARD_IOCTL_ENDTRANSACTION";
164
165
0
    case SCARD_IOCTL_STATE:
166
0
      return funcName ? "SCardState" : "SCARD_IOCTL_STATE";
167
168
0
    case SCARD_IOCTL_STATUSA:
169
0
      return funcName ? "SCardStatusA" : "SCARD_IOCTL_STATUSA";
170
171
0
    case SCARD_IOCTL_STATUSW:
172
0
      return funcName ? "SCardStatusW" : "SCARD_IOCTL_STATUSW";
173
174
0
    case SCARD_IOCTL_TRANSMIT:
175
0
      return funcName ? "SCardTransmit" : "SCARD_IOCTL_TRANSMIT";
176
177
0
    case SCARD_IOCTL_CONTROL:
178
0
      return funcName ? "SCardControl" : "SCARD_IOCTL_CONTROL";
179
180
0
    case SCARD_IOCTL_GETATTRIB:
181
0
      return funcName ? "SCardGetAttrib" : "SCARD_IOCTL_GETATTRIB";
182
183
0
    case SCARD_IOCTL_SETATTRIB:
184
0
      return funcName ? "SCardSetAttrib" : "SCARD_IOCTL_SETATTRIB";
185
186
0
    case SCARD_IOCTL_ACCESSSTARTEDEVENT:
187
0
      return funcName ? "SCardAccessStartedEvent" : "SCARD_IOCTL_ACCESSSTARTEDEVENT";
188
189
0
    case SCARD_IOCTL_LOCATECARDSBYATRA:
190
0
      return funcName ? "SCardLocateCardsByATRA" : "SCARD_IOCTL_LOCATECARDSBYATRA";
191
192
0
    case SCARD_IOCTL_LOCATECARDSBYATRW:
193
0
      return funcName ? "SCardLocateCardsByATRB" : "SCARD_IOCTL_LOCATECARDSBYATRW";
194
195
0
    case SCARD_IOCTL_READCACHEA:
196
0
      return funcName ? "SCardReadCacheA" : "SCARD_IOCTL_READCACHEA";
197
198
0
    case SCARD_IOCTL_READCACHEW:
199
0
      return funcName ? "SCardReadCacheW" : "SCARD_IOCTL_READCACHEW";
200
201
0
    case SCARD_IOCTL_WRITECACHEA:
202
0
      return funcName ? "SCardWriteCacheA" : "SCARD_IOCTL_WRITECACHEA";
203
204
0
    case SCARD_IOCTL_WRITECACHEW:
205
0
      return funcName ? "SCardWriteCacheW" : "SCARD_IOCTL_WRITECACHEW";
206
207
0
    case SCARD_IOCTL_GETTRANSMITCOUNT:
208
0
      return funcName ? "SCardGetTransmitCount" : "SCARD_IOCTL_GETTRANSMITCOUNT";
209
210
0
    case SCARD_IOCTL_RELEASETARTEDEVENT:
211
0
      return funcName ? "SCardReleaseStartedEvent" : "SCARD_IOCTL_RELEASETARTEDEVENT";
212
213
0
    case SCARD_IOCTL_GETREADERICON:
214
0
      return funcName ? "SCardGetReaderIcon" : "SCARD_IOCTL_GETREADERICON";
215
216
0
    case SCARD_IOCTL_GETDEVICETYPEID:
217
0
      return funcName ? "SCardGetDeviceTypeId" : "SCARD_IOCTL_GETDEVICETYPEID";
218
219
0
    default:
220
0
      return funcName ? "SCardUnknown" : "SCARD_IOCTL_UNKNOWN";
221
0
  }
222
0
}
223
224
const char* rdpdr_component_string(UINT16 component)
225
0
{
226
0
  switch (component)
227
0
  {
228
0
    case RDPDR_CTYP_PRN:
229
0
      return "RDPDR_CTYP_PRN";
230
0
    case RDPDR_CTYP_CORE:
231
0
      return "RDPDR_CTYP_CORE";
232
0
    default:
233
0
      return "UNKNOWN";
234
0
  }
235
0
}
236
237
const char* rdpdr_packetid_string(UINT16 packetid)
238
0
{
239
0
  switch (packetid)
240
0
  {
241
0
    case PAKID_CORE_SERVER_ANNOUNCE:
242
0
      return "PAKID_CORE_SERVER_ANNOUNCE";
243
0
    case PAKID_CORE_CLIENTID_CONFIRM:
244
0
      return "PAKID_CORE_CLIENTID_CONFIRM";
245
0
    case PAKID_CORE_CLIENT_NAME:
246
0
      return "PAKID_CORE_CLIENT_NAME";
247
0
    case PAKID_CORE_DEVICELIST_ANNOUNCE:
248
0
      return "PAKID_CORE_DEVICELIST_ANNOUNCE";
249
0
    case PAKID_CORE_DEVICE_REPLY:
250
0
      return "PAKID_CORE_DEVICE_REPLY";
251
0
    case PAKID_CORE_DEVICE_IOREQUEST:
252
0
      return "PAKID_CORE_DEVICE_IOREQUEST";
253
0
    case PAKID_CORE_DEVICE_IOCOMPLETION:
254
0
      return "PAKID_CORE_DEVICE_IOCOMPLETION";
255
0
    case PAKID_CORE_SERVER_CAPABILITY:
256
0
      return "PAKID_CORE_SERVER_CAPABILITY";
257
0
    case PAKID_CORE_CLIENT_CAPABILITY:
258
0
      return "PAKID_CORE_CLIENT_CAPABILITY";
259
0
    case PAKID_CORE_DEVICELIST_REMOVE:
260
0
      return "PAKID_CORE_DEVICELIST_REMOVE";
261
0
    case PAKID_CORE_USER_LOGGEDON:
262
0
      return "PAKID_CORE_USER_LOGGEDON";
263
0
    case PAKID_PRN_CACHE_DATA:
264
0
      return "PAKID_PRN_CACHE_DATA";
265
0
    case PAKID_PRN_USING_XPS:
266
0
      return "PAKID_PRN_USING_XPS";
267
0
    default:
268
0
      return "UNKNOWN";
269
0
  }
270
0
}
271
272
BOOL rdpdr_write_iocompletion_header(wStream* out, UINT32 DeviceId, UINT32 CompletionId,
273
                                     NTSTATUS ioStatus)
274
0
{
275
0
  WINPR_ASSERT(out);
276
0
  Stream_ResetPosition(out);
277
0
  if (!Stream_EnsureRemainingCapacity(out, 16))
278
0
    return FALSE;
279
280
0
  Stream_Write_UINT16(out, RDPDR_CTYP_CORE);                /* Component (2 bytes) */
281
0
  Stream_Write_UINT16(out, PAKID_CORE_DEVICE_IOCOMPLETION); /* PacketId (2 bytes) */
282
0
  Stream_Write_UINT32(out, DeviceId);                       /* DeviceId (4 bytes) */
283
0
  Stream_Write_UINT32(out, CompletionId);                   /* CompletionId (4 bytes) */
284
0
  Stream_Write_INT32(out, ioStatus);                        /* IoStatus (4 bytes) */
285
286
0
  return TRUE;
287
0
}
288
289
static void rdpdr_dump_packet(wLog* log, DWORD lvl, wStream* s, const char* custom, BOOL send)
290
0
{
291
0
  if (!WLog_IsLevelActive(log, lvl))
292
0
    return;
293
294
0
  const size_t gpos = Stream_GetPosition(s);
295
0
  const size_t pos = send ? Stream_GetPosition(s) : Stream_Length(s);
296
297
0
  UINT16 component = 0;
298
0
  UINT16 packetid = 0;
299
300
0
  Stream_ResetPosition(s);
301
302
0
  if (pos >= 2)
303
0
    Stream_Read_UINT16(s, component);
304
0
  if (pos >= 4)
305
0
    Stream_Read_UINT16(s, packetid);
306
307
0
  switch (packetid)
308
0
  {
309
0
    case PAKID_CORE_SERVER_ANNOUNCE:
310
0
    case PAKID_CORE_CLIENTID_CONFIRM:
311
0
    {
312
0
      UINT16 versionMajor = 0;
313
0
      UINT16 versionMinor = 0;
314
0
      UINT32 clientID = 0;
315
316
0
      if (pos >= 6)
317
0
        Stream_Read_UINT16(s, versionMajor);
318
0
      if (pos >= 8)
319
0
        Stream_Read_UINT16(s, versionMinor);
320
0
      if (pos >= 12)
321
0
        Stream_Read_UINT32(s, clientID);
322
0
      WLog_Print(log, lvl,
323
0
                 "%s [%s | %s] [version:%" PRIu16 ".%" PRIu16 "][id:0x%08" PRIx32
324
0
                 "] -> %" PRIuz,
325
0
                 custom, rdpdr_component_string(component), rdpdr_packetid_string(packetid),
326
0
                 versionMajor, versionMinor, clientID, pos);
327
0
    }
328
0
    break;
329
0
    case PAKID_CORE_CLIENT_NAME:
330
0
    {
331
0
      char name[256] = WINPR_C_ARRAY_INIT;
332
0
      UINT32 unicodeFlag = 0;
333
0
      UINT32 codePage = 0;
334
0
      UINT32 computerNameLen = 0;
335
0
      if (pos >= 8)
336
0
        Stream_Read_UINT32(s, unicodeFlag);
337
0
      if (pos >= 12)
338
0
        Stream_Read_UINT32(s, codePage);
339
0
      if (pos >= 16)
340
0
        Stream_Read_UINT32(s, computerNameLen);
341
0
      if (pos >= 16 + computerNameLen)
342
0
      {
343
0
        if (unicodeFlag == 0)
344
0
          Stream_Read(s, name, MIN(sizeof(name), computerNameLen));
345
0
        else
346
0
          (void)ConvertWCharNToUtf8(Stream_ConstPointer(s),
347
0
                                    computerNameLen / sizeof(WCHAR), name, sizeof(name));
348
0
      }
349
0
      WLog_Print(log, lvl,
350
0
                 "%s [%s | %s] [ucs:%" PRIu32 "|cp:%" PRIu32 "][len:0x%08" PRIx32
351
0
                 "] '%s' -> %" PRIuz,
352
0
                 custom, rdpdr_component_string(component), rdpdr_packetid_string(packetid),
353
0
                 unicodeFlag, codePage, computerNameLen, name, pos);
354
0
    }
355
0
    break;
356
357
0
    case PAKID_CORE_DEVICE_IOREQUEST:
358
0
    {
359
0
      UINT32 CompletionId = 0;
360
0
      UINT32 deviceID = 0;
361
0
      UINT32 FileId = 0;
362
0
      UINT32 MajorFunction = 0;
363
0
      UINT32 MinorFunction = 0;
364
365
0
      if (pos >= 8)
366
0
        Stream_Read_UINT32(s, deviceID);
367
0
      if (pos >= 12)
368
0
        Stream_Read_UINT32(s, FileId);
369
0
      if (pos >= 16)
370
0
        Stream_Read_UINT32(s, CompletionId);
371
0
      if (pos >= 20)
372
0
        Stream_Read_UINT32(s, MajorFunction);
373
0
      if (pos >= 24)
374
0
        Stream_Read_UINT32(s, MinorFunction);
375
0
      WLog_Print(log, lvl,
376
0
                 "%s [%s | %s] [0x%08" PRIx32 "] FileId=0x%08" PRIx32
377
0
                 ", CompletionId=0x%08" PRIx32 ", MajorFunction=0x%08" PRIx32
378
0
                 ", MinorFunction=0x%08" PRIx32 " -> %" PRIuz,
379
0
                 custom, rdpdr_component_string(component), rdpdr_packetid_string(packetid),
380
0
                 deviceID, FileId, CompletionId, MajorFunction, MinorFunction, pos);
381
0
    }
382
0
    break;
383
0
    case PAKID_CORE_DEVICE_IOCOMPLETION:
384
0
    {
385
0
      UINT32 completionID = 0;
386
0
      UINT32 ioStatus = 0;
387
0
      UINT32 deviceID = 0;
388
0
      if (pos >= 8)
389
0
        Stream_Read_UINT32(s, deviceID);
390
0
      if (pos >= 12)
391
0
        Stream_Read_UINT32(s, completionID);
392
0
      if (pos >= 16)
393
0
        Stream_Read_UINT32(s, ioStatus);
394
395
0
      WLog_Print(log, lvl,
396
0
                 "%s [%s | %s] [0x%08" PRIx32 "] completionID=0x%08" PRIx32
397
0
                 ", ioStatus=0x%08" PRIx32 " -> %" PRIuz,
398
0
                 custom, rdpdr_component_string(component), rdpdr_packetid_string(packetid),
399
0
                 deviceID, completionID, ioStatus, pos);
400
0
    }
401
0
    break;
402
0
    case PAKID_CORE_DEVICE_REPLY:
403
0
    {
404
0
      UINT32 deviceID = 0;
405
0
      UINT32 status = 0;
406
407
0
      if (pos >= 8)
408
0
        Stream_Read_UINT32(s, deviceID);
409
0
      if (pos >= 12)
410
0
        Stream_Read_UINT32(s, status);
411
0
      WLog_Print(log, lvl,
412
0
                 "%s [%s | %s] [id:0x%08" PRIx32 ",status=0x%08" PRIx32 "] -> %" PRIuz,
413
0
                 custom, rdpdr_component_string(component), rdpdr_packetid_string(packetid),
414
0
                 deviceID, status, pos);
415
0
    }
416
0
    break;
417
0
    case PAKID_CORE_CLIENT_CAPABILITY:
418
0
    case PAKID_CORE_SERVER_CAPABILITY:
419
0
    {
420
0
      UINT16 numCapabilities = 0;
421
0
      if (pos >= 6)
422
0
        Stream_Read_UINT16(s, numCapabilities);
423
0
      if (pos >= 8)
424
0
        Stream_Seek_UINT16(s); /* padding */
425
0
      WLog_Print(log, lvl, "%s [%s | %s] [caps:%" PRIu16 "] -> %" PRIuz, custom,
426
0
                 rdpdr_component_string(component), rdpdr_packetid_string(packetid),
427
0
                 numCapabilities, pos);
428
0
      for (UINT16 x = 0; x < numCapabilities; x++)
429
0
      {
430
0
        RDPDR_CAPABILITY_HEADER header = WINPR_C_ARRAY_INIT;
431
0
        const UINT error = rdpdr_read_capset_header(log, s, &header);
432
0
        if (error == CHANNEL_RC_OK)
433
0
          Stream_Seek(s, header.CapabilityLength);
434
0
      }
435
0
    }
436
0
    break;
437
0
    case PAKID_CORE_DEVICELIST_ANNOUNCE:
438
0
    {
439
0
      size_t offset = 8;
440
0
      UINT32 count = 0;
441
442
0
      if (pos >= offset)
443
0
        Stream_Read_UINT32(s, count);
444
445
0
      WLog_Print(log, lvl, "%s [%s | %s] [%" PRIu32 "] -> %" PRIuz, custom,
446
0
                 rdpdr_component_string(component), rdpdr_packetid_string(packetid), count,
447
0
                 pos);
448
449
0
      for (UINT32 x = 0; x < count; x++)
450
0
      {
451
0
        RdpdrDevice device = WINPR_C_ARRAY_INIT;
452
453
0
        offset += 20;
454
0
        if (pos >= offset)
455
0
        {
456
0
          Stream_Read_UINT32(s, device.DeviceType);       /* DeviceType (4 bytes) */
457
0
          Stream_Read_UINT32(s, device.DeviceId);         /* DeviceId (4 bytes) */
458
0
          Stream_Read(s, device.PreferredDosName, 8);     /* PreferredDosName (8 bytes) */
459
0
          Stream_Read_UINT32(s, device.DeviceDataLength); /* DeviceDataLength (4 bytes) */
460
0
          device.DeviceData = Stream_Pointer(s);
461
0
        }
462
0
        offset += device.DeviceDataLength;
463
464
0
        WLog_Print(log, lvl,
465
0
                   "%s [announce][%" PRIu32 "] %s [0x%08" PRIx32
466
0
                   "] '%s' [DeviceDataLength=%" PRIu32 "]",
467
0
                   custom, x, freerdp_rdpdr_dtyp_string(device.DeviceType), device.DeviceId,
468
0
                   device.PreferredDosName, device.DeviceDataLength);
469
0
      }
470
0
    }
471
0
    break;
472
0
    case PAKID_CORE_DEVICELIST_REMOVE:
473
0
    {
474
0
      size_t offset = 8;
475
0
      UINT32 count = 0;
476
477
0
      if (pos >= offset)
478
0
        Stream_Read_UINT32(s, count);
479
480
0
      WLog_Print(log, lvl, "%s [%s | %s] [%" PRIu32 "] -> %" PRIuz, custom,
481
0
                 rdpdr_component_string(component), rdpdr_packetid_string(packetid), count,
482
0
                 pos);
483
484
0
      for (UINT32 x = 0; x < count; x++)
485
0
      {
486
0
        UINT32 id = 0;
487
488
0
        offset += 4;
489
0
        if (pos >= offset)
490
0
          Stream_Read_UINT32(s, id);
491
492
0
        WLog_Print(log, lvl, "%s [remove][%" PRIu32 "] id=%" PRIu32, custom, x, id);
493
0
      }
494
0
    }
495
0
    break;
496
0
    case PAKID_CORE_USER_LOGGEDON:
497
0
      WLog_Print(log, lvl, "%s [%s | %s] -> %" PRIuz, custom,
498
0
                 rdpdr_component_string(component), rdpdr_packetid_string(packetid), pos);
499
0
      break;
500
0
    default:
501
0
    {
502
0
      WLog_Print(log, lvl, "%s [%s | %s] -> %" PRIuz, custom,
503
0
                 rdpdr_component_string(component), rdpdr_packetid_string(packetid), pos);
504
0
    }
505
0
    break;
506
0
  }
507
508
  // winpr_HexLogDump(log, lvl, Stream_Buffer(s), pos);
509
0
  if (!Stream_SetPosition(s, gpos))
510
0
    WLog_Print(log, WLOG_ERROR, "Stream_SetPosition(%" PRIuz ") failed", gpos);
511
0
}
512
513
void rdpdr_dump_received_packet(wLog* log, DWORD lvl, wStream* out, const char* custom)
514
0
{
515
0
  rdpdr_dump_packet(log, lvl, out, custom, FALSE);
516
0
}
517
518
void rdpdr_dump_send_packet(wLog* log, DWORD lvl, wStream* out, const char* custom)
519
0
{
520
0
  rdpdr_dump_packet(log, lvl, out, custom, TRUE);
521
0
}
522
523
const char* rdpdr_irp_string(UINT32 major)
524
0
{
525
0
  switch (major)
526
0
  {
527
0
    case IRP_MJ_CREATE:
528
0
      return "IRP_MJ_CREATE";
529
0
    case IRP_MJ_CLOSE:
530
0
      return "IRP_MJ_CLOSE";
531
0
    case IRP_MJ_READ:
532
0
      return "IRP_MJ_READ";
533
0
    case IRP_MJ_WRITE:
534
0
      return "IRP_MJ_WRITE";
535
0
    case IRP_MJ_DEVICE_CONTROL:
536
0
      return "IRP_MJ_DEVICE_CONTROL";
537
0
    case IRP_MJ_QUERY_VOLUME_INFORMATION:
538
0
      return "IRP_MJ_QUERY_VOLUME_INFORMATION";
539
0
    case IRP_MJ_SET_VOLUME_INFORMATION:
540
0
      return "IRP_MJ_SET_VOLUME_INFORMATION";
541
0
    case IRP_MJ_QUERY_INFORMATION:
542
0
      return "IRP_MJ_QUERY_INFORMATION";
543
0
    case IRP_MJ_SET_INFORMATION:
544
0
      return "IRP_MJ_SET_INFORMATION";
545
0
    case IRP_MJ_DIRECTORY_CONTROL:
546
0
      return "IRP_MJ_DIRECTORY_CONTROL";
547
0
    case IRP_MJ_LOCK_CONTROL:
548
0
      return "IRP_MJ_LOCK_CONTROL";
549
0
    default:
550
0
      return "IRP_UNKNOWN";
551
0
  }
552
0
}
553
554
const char* rdpdr_cap_type_string(UINT16 capability)
555
0
{
556
0
  switch (capability)
557
0
  {
558
0
    case CAP_GENERAL_TYPE:
559
0
      return "CAP_GENERAL_TYPE";
560
0
    case CAP_PRINTER_TYPE:
561
0
      return "CAP_PRINTER_TYPE";
562
0
    case CAP_PORT_TYPE:
563
0
      return "CAP_PORT_TYPE";
564
0
    case CAP_DRIVE_TYPE:
565
0
      return "CAP_DRIVE_TYPE";
566
0
    case CAP_SMARTCARD_TYPE:
567
0
      return "CAP_SMARTCARD_TYPE";
568
0
    default:
569
0
      return "CAP_UNKNOWN";
570
0
  }
571
0
}
572
573
UINT rdpdr_read_capset_header(wLog* log, wStream* s, RDPDR_CAPABILITY_HEADER* header)
574
0
{
575
0
  WINPR_ASSERT(header);
576
0
  if (!Stream_CheckAndLogRequiredLengthWLog(log, s, 8))
577
0
    return ERROR_INVALID_DATA;
578
579
0
  Stream_Read_UINT16(s, header->CapabilityType);   /* CapabilityType (2 bytes) */
580
0
  Stream_Read_UINT16(s, header->CapabilityLength); /* CapabilityLength (2 bytes) */
581
0
  Stream_Read_UINT32(s, header->Version);          /* Version (4 bytes) */
582
583
0
  WLog_Print(log, WLOG_TRACE,
584
0
             "capability %s [0x%04" PRIx16 "] got version %" PRIu32 ", length %" PRIu16,
585
0
             rdpdr_cap_type_string(header->CapabilityType), header->CapabilityType,
586
0
             header->Version, header->CapabilityLength);
587
0
  if (header->CapabilityLength < 8)
588
0
  {
589
0
    WLog_Print(log, WLOG_ERROR, "capability %s got short length %" PRIu32,
590
0
               rdpdr_cap_type_string(header->CapabilityType), header->CapabilityLength);
591
0
    return ERROR_INVALID_DATA;
592
0
  }
593
0
  header->CapabilityLength -= 8;
594
0
  if (!Stream_CheckAndLogRequiredLengthWLog(log, s, header->CapabilityLength))
595
0
    return ERROR_INVALID_DATA;
596
0
  return CHANNEL_RC_OK;
597
0
}
598
599
UINT rdpdr_write_capset_header(wLog* log, wStream* s, const RDPDR_CAPABILITY_HEADER* header)
600
0
{
601
0
  WINPR_ASSERT(header);
602
0
  WINPR_ASSERT(header->CapabilityLength >= 8);
603
604
0
  if (!Stream_EnsureRemainingCapacity(s, header->CapabilityLength))
605
0
  {
606
0
    WLog_Print(log, WLOG_ERROR, "not enough data in stream!");
607
0
    return ERROR_INVALID_DATA;
608
0
  }
609
610
0
  WLog_Print(log, WLOG_TRACE, "writing capability %s version %" PRIu32 ", length %" PRIu16,
611
0
             rdpdr_cap_type_string(header->CapabilityType), header->Version,
612
0
             header->CapabilityLength);
613
0
  Stream_Write_UINT16(s, header->CapabilityType);   /* CapabilityType (2 bytes) */
614
0
  Stream_Write_UINT16(s, header->CapabilityLength); /* CapabilityLength (2 bytes) */
615
0
  Stream_Write_UINT32(s, header->Version);          /* Version (4 bytes) */
616
0
  return CHANNEL_RC_OK;
617
0
}
618
619
const char* rdpdr_irp_val2str(UINT32 ioCode1)
620
0
{
621
0
  switch (ioCode1)
622
0
  {
623
0
    case RDPDR_IRP_MJ_CREATE:
624
0
      return "MJ_CREATE";
625
0
    case RDPDR_IRP_MJ_CLEANUP:
626
0
      return "MJ_CLEANUP";
627
0
    case RDPDR_IRP_MJ_CLOSE:
628
0
      return "MJ_CLOSE";
629
0
    case RDPDR_IRP_MJ_READ:
630
0
      return "MJ_READ";
631
0
    case RDPDR_IRP_MJ_WRITE:
632
0
      return "MJ_WRITE";
633
0
    case RDPDR_IRP_MJ_FLUSH_BUFFERS:
634
0
      return "MJ_FLUSH_BUFFERS";
635
0
    case RDPDR_IRP_MJ_SHUTDOWN:
636
0
      return "MJ_SHUTDOWN";
637
0
    case RDPDR_IRP_MJ_DEVICE_CONTROL:
638
0
      return "MJ_DEVICE_CONTROL";
639
0
    case RDPDR_IRP_MJ_QUERY_VOLUME_INFORMATION:
640
0
      return "MJ_QUERY_VOLUME_INFORMATION";
641
0
    case RDPDR_IRP_MJ_SET_VOLUME_INFORMATION:
642
0
      return "MJ_SET_VOLUME_INFORMATION";
643
0
    case RDPDR_IRP_MJ_QUERY_INFORMATION:
644
0
      return "MJ_QUERY_INFORMATION";
645
0
    case RDPDR_IRP_MJ_SET_INFORMATION:
646
0
      return "MJ_SET_INFORMATION";
647
0
    case RDPDR_IRP_MJ_DIRECTORY_CONTROL:
648
0
      return "MJ_DIRECTORY_CONTROL";
649
0
    case RDPDR_IRP_MJ_LOCK_CONTROL:
650
0
      return "MJ_LOCK_CONTROL";
651
0
    case RDPDR_IRP_MJ_QUERY_SECURITY:
652
0
      return "MJ_QUERY_SECURITY";
653
0
    case RDPDR_IRP_MJ_SET_SECURITY:
654
0
      return "MJ_SET_SECURITY";
655
0
    default:
656
0
      return "IRP_MJ_UNKNOWN";
657
0
  }
658
0
}
659
660
const char* rdpdr_irp_mask2str(UINT32 ioCode1Mask, char* buffer, size_t len)
661
0
{
662
0
  if (len < 1)
663
0
    return nullptr;
664
665
0
  if (!winpr_str_append("{", buffer, len, nullptr))
666
0
    return nullptr;
667
668
0
  for (size_t x = 0; x < 32; x++)
669
0
  {
670
0
    const UINT32 mask = (1u << x);
671
0
    if (ioCode1Mask & mask)
672
0
    {
673
0
      if (!winpr_str_append(rdpdr_irp_val2str(mask), &buffer[1], len - 1, "|"))
674
0
        return nullptr;
675
0
    }
676
0
  }
677
678
0
  char number[16] = WINPR_C_ARRAY_INIT;
679
0
  (void)_snprintf(number, sizeof(number), "}[0x%08" PRIx32 "]", ioCode1Mask);
680
0
  if (!winpr_str_append(number, buffer, len, nullptr))
681
0
    return nullptr;
682
0
  return buffer;
683
0
}
684
685
const char* rdpdr_device_type_string(UINT32 type)
686
0
{
687
0
  switch (type)
688
0
  {
689
0
    case RDPDR_DTYP_SERIAL:
690
0
      return "serial";
691
0
    case RDPDR_DTYP_PRINT:
692
0
      return "printer";
693
0
    case RDPDR_DTYP_FILESYSTEM:
694
0
      return "drive";
695
0
    case RDPDR_DTYP_SMARTCARD:
696
0
      return "smartcard";
697
0
    case RDPDR_DTYP_PARALLEL:
698
0
      return "parallel";
699
0
    default:
700
0
      return "UNKNOWN";
701
0
  }
702
0
}