FreeRDP
Loading...
Searching...
No Matches
cliprdr_common.c
1
23#include <winpr/crt.h>
24#include <winpr/stream.h>
25#include <freerdp/channels/log.h>
26
27#define TAG CHANNELS_TAG("cliprdr.common")
28
29#include "cliprdr_common.h"
30
31static const char* CB_MSG_TYPE_STR(UINT32 type)
32{
33 switch (type)
34 {
35 case CB_TYPE_NONE:
36 return "CB_TYPE_NONE";
37 case CB_MONITOR_READY:
38 return "CB_MONITOR_READY";
39 case CB_FORMAT_LIST:
40 return "CB_FORMAT_LIST";
41 case CB_FORMAT_LIST_RESPONSE:
42 return "CB_FORMAT_LIST_RESPONSE";
43 case CB_FORMAT_DATA_REQUEST:
44 return "CB_FORMAT_DATA_REQUEST";
45 case CB_FORMAT_DATA_RESPONSE:
46 return "CB_FORMAT_DATA_RESPONSE";
47 case CB_TEMP_DIRECTORY:
48 return "CB_TEMP_DIRECTORY";
49 case CB_CLIP_CAPS:
50 return "CB_CLIP_CAPS";
51 case CB_FILECONTENTS_REQUEST:
52 return "CB_FILECONTENTS_REQUEST";
53 case CB_FILECONTENTS_RESPONSE:
54 return "CB_FILECONTENTS_RESPONSE";
55 case CB_LOCK_CLIPDATA:
56 return "CB_LOCK_CLIPDATA";
57 case CB_UNLOCK_CLIPDATA:
58 return "CB_UNLOCK_CLIPDATA";
59 default:
60 return "UNKNOWN";
61 }
62}
63
64const char* CB_MSG_TYPE_STRING(UINT16 type, char* buffer, size_t size)
65{
66 (void)_snprintf(buffer, size, "%s [0x%04" PRIx16 "]", CB_MSG_TYPE_STR(type), type);
67 return buffer;
68}
69
70const char* CB_MSG_FLAGS_STRING(UINT16 msgFlags, char* buffer, size_t size)
71{
72 if ((msgFlags & CB_RESPONSE_OK) != 0)
73 winpr_str_append("CB_RESPONSE_OK", buffer, size, "|");
74 if ((msgFlags & CB_RESPONSE_FAIL) != 0)
75 winpr_str_append("CB_RESPONSE_FAIL", buffer, size, "|");
76 if ((msgFlags & CB_ASCII_NAMES) != 0)
77 winpr_str_append("CB_ASCII_NAMES", buffer, size, "|");
78
79 const size_t len = strnlen(buffer, size);
80 if (!len)
81 winpr_str_append("NONE", buffer, size, "");
82
83 char val[32] = WINPR_C_ARRAY_INIT;
84 (void)_snprintf(val, sizeof(val), "[0x%04" PRIx16 "]", msgFlags);
85 winpr_str_append(val, buffer, size, "|");
86 return buffer;
87}
88
89static BOOL cliprdr_validate_file_contents_request(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
90{
91 /*
92 * [MS-RDPECLIP] 2.2.5.3 File Contents Request PDU (CLIPRDR_FILECONTENTS_REQUEST).
93 *
94 * A request for the size of the file identified by the lindex field. The size MUST be
95 * returned as a 64-bit, unsigned integer. The cbRequested field MUST be set to
96 * 0x00000008 and both the nPositionLow and nPositionHigh fields MUST be
97 * set to 0x00000000.
98 */
99
100 if (request->dwFlags & FILECONTENTS_SIZE)
101 {
102 if (request->cbRequested != sizeof(UINT64))
103 {
104 WLog_ERR(TAG, "cbRequested must be %" PRIuz ", got %" PRIu32 "", sizeof(UINT64),
105 request->cbRequested);
106 return FALSE;
107 }
108
109 if (request->nPositionHigh != 0 || request->nPositionLow != 0)
110 {
111 WLog_ERR(TAG, "nPositionHigh and nPositionLow must be set to 0");
112 return FALSE;
113 }
114 }
115
116 return TRUE;
117}
118
119wStream* cliprdr_packet_new(UINT16 msgType, UINT16 msgFlags, size_t dataLen)
120{
121 WINPR_ASSERT(dataLen < UINT32_MAX);
122 wStream* s = Stream_New(nullptr, dataLen + 8ULL);
123
124 if (!s)
125 {
126 WLog_ERR(TAG, "Stream_New failed!");
127 return nullptr;
128 }
129
130 Stream_Write_UINT16(s, msgType);
131 Stream_Write_UINT16(s, msgFlags);
132 /* Write actual length after the entire packet has been constructed. */
133 Stream_Write_UINT32(s, 0);
134 return s;
135}
136
137static void cliprdr_write_file_contents_request(wStream* s,
138 const CLIPRDR_FILE_CONTENTS_REQUEST* request)
139{
140 Stream_Write_UINT32(s, request->streamId); /* streamId (4 bytes) */
141 Stream_Write_UINT32(s, request->listIndex); /* listIndex (4 bytes) */
142 Stream_Write_UINT32(s, request->dwFlags); /* dwFlags (4 bytes) */
143 Stream_Write_UINT32(s, request->nPositionLow); /* nPositionLow (4 bytes) */
144 Stream_Write_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
145 Stream_Write_UINT32(s, request->cbRequested); /* cbRequested (4 bytes) */
146
147 if (request->haveClipDataId)
148 Stream_Write_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
149}
150
151static inline void cliprdr_write_lock_unlock_clipdata(wStream* s, UINT32 clipDataId)
152{
153 Stream_Write_UINT32(s, clipDataId);
154}
155
156static void cliprdr_write_lock_clipdata(wStream* s,
157 const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
158{
159 cliprdr_write_lock_unlock_clipdata(s, lockClipboardData->clipDataId);
160}
161
162static void cliprdr_write_unlock_clipdata(wStream* s,
163 const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
164{
165 cliprdr_write_lock_unlock_clipdata(s, unlockClipboardData->clipDataId);
166}
167
168static void cliprdr_write_file_contents_response(wStream* s,
169 const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
170{
171 Stream_Write_UINT32(s, response->streamId); /* streamId (4 bytes) */
172 Stream_Write(s, response->requestedData, response->cbRequested);
173}
174
175wStream* cliprdr_packet_lock_clipdata_new(const CLIPRDR_LOCK_CLIPBOARD_DATA* lockClipboardData)
176{
177 wStream* s = nullptr;
178
179 if (!lockClipboardData)
180 return nullptr;
181
182 s = cliprdr_packet_new(CB_LOCK_CLIPDATA, 0, 4);
183
184 if (!s)
185 return nullptr;
186
187 cliprdr_write_lock_clipdata(s, lockClipboardData);
188 return s;
189}
190
191wStream*
192cliprdr_packet_unlock_clipdata_new(const CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
193{
194 wStream* s = nullptr;
195
196 if (!unlockClipboardData)
197 return nullptr;
198
199 s = cliprdr_packet_new(CB_UNLOCK_CLIPDATA, 0, 4);
200
201 if (!s)
202 return nullptr;
203
204 cliprdr_write_unlock_clipdata(s, unlockClipboardData);
205 return s;
206}
207
208wStream* cliprdr_packet_file_contents_request_new(const CLIPRDR_FILE_CONTENTS_REQUEST* request)
209{
210 wStream* s = nullptr;
211
212 if (!request)
213 return nullptr;
214
215 s = cliprdr_packet_new(CB_FILECONTENTS_REQUEST, 0, 28);
216
217 if (!s)
218 return nullptr;
219
220 cliprdr_write_file_contents_request(s, request);
221 return s;
222}
223
224wStream* cliprdr_packet_file_contents_response_new(const CLIPRDR_FILE_CONTENTS_RESPONSE* response)
225{
226 wStream* s = nullptr;
227
228 if (!response)
229 return nullptr;
230
231 s = cliprdr_packet_new(CB_FILECONTENTS_RESPONSE, response->common.msgFlags,
232 4 + response->cbRequested);
233
234 if (!s)
235 return nullptr;
236
237 cliprdr_write_file_contents_response(s, response);
238 return s;
239}
240
241wStream* cliprdr_packet_format_list_new(const CLIPRDR_FORMAT_LIST* formatList,
242 BOOL useLongFormatNames, BOOL useAsciiNames)
243{
244 WINPR_ASSERT(formatList);
245
246 if (formatList->common.msgType != CB_FORMAT_LIST)
247 WLog_WARN(TAG, "called with invalid type %08" PRIx32, formatList->common.msgType);
248
249 if (useLongFormatNames && useAsciiNames)
250 WLog_WARN(TAG, "called with invalid arguments useLongFormatNames=true && "
251 "useAsciiNames=true. useAsciiNames requires "
252 "useLongFormatNames=false, ignoring argument.");
253
254 const UINT32 length = formatList->numFormats * 36;
255 const size_t formatNameCharSize =
256 (useLongFormatNames || !useAsciiNames) ? sizeof(WCHAR) : sizeof(CHAR);
257
258 wStream* s = cliprdr_packet_new(CB_FORMAT_LIST, 0, length);
259 if (!s)
260 {
261 WLog_ERR(TAG, "cliprdr_packet_new failed!");
262 return nullptr;
263 }
264
265 for (UINT32 index = 0; index < formatList->numFormats; index++)
266 {
267 const CLIPRDR_FORMAT* format = &(formatList->formats[index]);
268
269 const char* szFormatName = format->formatName;
270 size_t formatNameLength = 0;
271 if (szFormatName)
272 formatNameLength = strlen(szFormatName);
273
274 size_t formatNameMaxLength = formatNameLength + 1; /* Ensure '\0' termination in output */
275 if (!Stream_EnsureRemainingCapacity(s,
276 4 + MAX(32, formatNameMaxLength * formatNameCharSize)))
277 goto fail;
278
279 Stream_Write_UINT32(s, format->formatId); /* formatId (4 bytes) */
280
281 if (!useLongFormatNames)
282 {
283 formatNameMaxLength = useAsciiNames ? 32 : 16;
284 formatNameLength = MIN(formatNameMaxLength - 1, formatNameLength);
285 }
286
287 if (szFormatName && (formatNameLength > 0))
288 {
289 if (useAsciiNames)
290 {
291 Stream_Write(s, szFormatName, formatNameLength);
292 Stream_Zero(s, formatNameMaxLength - formatNameLength);
293 }
294 else
295 {
296 if (Stream_Write_UTF16_String_From_UTF8(s, formatNameMaxLength, szFormatName,
297 formatNameLength, TRUE) < 0)
298 goto fail;
299 }
300 }
301 else
302 Stream_Zero(s, formatNameMaxLength * formatNameCharSize);
303 }
304
305 return s;
306
307fail:
308 Stream_Free(s, TRUE);
309 return nullptr;
310}
311
312UINT cliprdr_read_unlock_clipdata(wStream* s, CLIPRDR_UNLOCK_CLIPBOARD_DATA* unlockClipboardData)
313{
314 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
315 return ERROR_INVALID_DATA;
316
317 Stream_Read_UINT32(s, unlockClipboardData->clipDataId); /* clipDataId (4 bytes) */
318 return CHANNEL_RC_OK;
319}
320
321UINT cliprdr_read_format_data_request(wStream* s, CLIPRDR_FORMAT_DATA_REQUEST* request)
322{
323 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
324 return ERROR_INVALID_DATA;
325
326 Stream_Read_UINT32(s, request->requestedFormatId); /* requestedFormatId (4 bytes) */
327 return CHANNEL_RC_OK;
328}
329
330UINT cliprdr_read_format_data_response(wStream* s, CLIPRDR_FORMAT_DATA_RESPONSE* response)
331{
332 response->requestedFormatData = nullptr;
333
334 if (!Stream_CheckAndLogRequiredLength(TAG, s, response->common.dataLen))
335 return ERROR_INVALID_DATA;
336
337 if (response->common.dataLen > 0)
338 {
339 response->requestedFormatData = Stream_ConstPointer(s);
340 if (!Stream_SafeSeek(s, response->common.dataLen))
341 return ERROR_INVALID_DATA;
342 }
343 return CHANNEL_RC_OK;
344}
345
346UINT cliprdr_read_file_contents_request(wStream* s, CLIPRDR_FILE_CONTENTS_REQUEST* request)
347{
348 if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
349 return ERROR_INVALID_DATA;
350
351 request->haveClipDataId = FALSE;
352 Stream_Read_UINT32(s, request->streamId); /* streamId (4 bytes) */
353 Stream_Read_UINT32(s, request->listIndex); /* listIndex (4 bytes) */
354 Stream_Read_UINT32(s, request->dwFlags); /* dwFlags (4 bytes) */
355 Stream_Read_UINT32(s, request->nPositionLow); /* nPositionLow (4 bytes) */
356 Stream_Read_UINT32(s, request->nPositionHigh); /* nPositionHigh (4 bytes) */
357 Stream_Read_UINT32(s, request->cbRequested); /* cbRequested (4 bytes) */
358
359 if (Stream_GetRemainingLength(s) >= 4)
360 {
361 Stream_Read_UINT32(s, request->clipDataId); /* clipDataId (4 bytes) */
362 request->haveClipDataId = TRUE;
363 }
364
365 if (!cliprdr_validate_file_contents_request(request))
366 return ERROR_BAD_ARGUMENTS;
367
368 return CHANNEL_RC_OK;
369}
370
371UINT cliprdr_read_file_contents_response(wStream* s, CLIPRDR_FILE_CONTENTS_RESPONSE* response)
372{
373 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
374 return ERROR_INVALID_DATA;
375
376 Stream_Read_UINT32(s, response->streamId); /* streamId (4 bytes) */
377 response->requestedData = Stream_ConstPointer(s); /* requestedFileContentsData */
378
379 if (response->common.dataLen < 4)
380 {
381 WLog_WARN(TAG, "dataLen=%" PRIu32 " but expected >= 4", response->common.dataLen);
382 return ERROR_INVALID_DATA;
383 }
384
385 response->cbRequested = response->common.dataLen - 4;
386 if (!Stream_CheckAndLogRequiredLength(TAG, s, response->cbRequested))
387 return ERROR_INVALID_DATA;
388 Stream_Seek(s, response->cbRequested);
389 return CHANNEL_RC_OK;
390}
391
392UINT cliprdr_read_format_list(wLog* log, wStream* s, CLIPRDR_FORMAT_LIST* formatList,
393 BOOL useLongFormatNames)
394{
395 UINT32 index = 0;
396 size_t formatNameLength = 0;
397 const char* szFormatName = nullptr;
398 const WCHAR* wszFormatName = nullptr;
399 wStream sub1buffer = WINPR_C_ARRAY_INIT;
400 CLIPRDR_FORMAT* formats = nullptr;
401 UINT error = ERROR_INTERNAL_ERROR;
402
403 const BOOL asciiNames = (formatList->common.msgFlags & CB_ASCII_NAMES) != 0;
404
405 index = 0;
406 /* empty format list */
407 formatList->formats = nullptr;
408 formatList->numFormats = 0;
409
410 wStream* sub1 =
411 Stream_StaticConstInit(&sub1buffer, Stream_ConstPointer(s), formatList->common.dataLen);
412 if (!Stream_SafeSeek(s, formatList->common.dataLen))
413 return ERROR_INVALID_DATA;
414
415 if (!formatList->common.dataLen)
416 {
417 }
418 else if (!useLongFormatNames)
419 {
420 const size_t cap = Stream_Capacity(sub1) / 36ULL;
421 if (cap > UINT32_MAX)
422 {
423 WLog_Print(log, WLOG_ERROR, "Invalid short format list length: %" PRIuz "", cap);
424 return ERROR_INTERNAL_ERROR;
425 }
426 formatList->numFormats = (UINT32)cap;
427
428 if (formatList->numFormats)
429 formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
430
431 if (!formats)
432 {
433 WLog_Print(log, WLOG_ERROR, "calloc failed!");
434 return CHANNEL_RC_NO_MEMORY;
435 }
436
437 formatList->formats = formats;
438
439 while (Stream_GetRemainingLength(sub1) >= 4)
440 {
441 if (index >= formatList->numFormats)
442 goto error_out;
443
444 CLIPRDR_FORMAT* format = &formats[index];
445
446 Stream_Read_UINT32(sub1, format->formatId); /* formatId (4 bytes) */
447
448 /* According to MS-RDPECLIP 2.2.3.1.1.1 formatName is "a 32-byte block containing
449 * the *null-terminated* name assigned to the Clipboard Format: (32 ASCII 8 characters
450 * or 16 Unicode characters)"
451 * However, both Windows RDSH and mstsc violate this specs as seen in the following
452 * example of a transferred short format name string: [R.i.c.h. .T.e.x.t. .F.o.r.m.a.t.]
453 * These are 16 unicode characters - *without* terminating null !
454 */
455
456 szFormatName = Stream_ConstPointer(sub1);
457 wszFormatName = Stream_ConstPointer(sub1);
458 if (!Stream_SafeSeek(sub1, 32))
459 goto error_out;
460
461 free(format->formatName);
462 format->formatName = nullptr;
463
464 if (asciiNames)
465 {
466 if (szFormatName[0])
467 {
468 /* ensure null termination */
469 format->formatName = strndup(szFormatName, 31);
470 if (!format->formatName)
471 {
472 WLog_Print(log, WLOG_ERROR, "malloc failed!");
473 error = CHANNEL_RC_NO_MEMORY;
474 goto error_out;
475 }
476 }
477 }
478 else
479 {
480 if (wszFormatName[0])
481 {
482 format->formatName = ConvertWCharNToUtf8Alloc(wszFormatName, 16, nullptr);
483 if (!format->formatName)
484 goto error_out;
485 }
486 }
487
488 index++;
489 }
490 }
491 else
492 {
493 wStream sub2buffer = sub1buffer;
494 wStream* sub2 = &sub2buffer;
495
496 while (Stream_GetRemainingLength(sub1) > 0)
497 {
498 size_t rest = 0;
499 if (!Stream_SafeSeek(sub1, 4)) /* formatId (4 bytes) */
500 goto error_out;
501
502 wszFormatName = Stream_ConstPointer(sub1);
503 rest = Stream_GetRemainingLength(sub1);
504 formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
505
506 if (!Stream_SafeSeek(sub1, (formatNameLength + 1) * sizeof(WCHAR)))
507 goto error_out;
508 formatList->numFormats++;
509 }
510
511 if (formatList->numFormats)
512 formats = (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT));
513
514 if (!formats)
515 {
516 WLog_Print(log, WLOG_ERROR, "calloc failed!");
517 return CHANNEL_RC_NO_MEMORY;
518 }
519
520 formatList->formats = formats;
521
522 while (Stream_GetRemainingLength(sub2) >= 4)
523 {
524 if (index >= formatList->numFormats)
525 goto error_out;
526
527 size_t rest = 0;
528 CLIPRDR_FORMAT* format = &formats[index];
529
530 Stream_Read_UINT32(sub2, format->formatId); /* formatId (4 bytes) */
531
532 free(format->formatName);
533 format->formatName = nullptr;
534
535 wszFormatName = Stream_ConstPointer(sub2);
536 rest = Stream_GetRemainingLength(sub2);
537 formatNameLength = _wcsnlen(wszFormatName, rest / sizeof(WCHAR));
538 if (!Stream_SafeSeek(sub2, (formatNameLength + 1) * sizeof(WCHAR)))
539 goto error_out;
540
541 if (formatNameLength)
542 {
543 format->formatName =
544 ConvertWCharNToUtf8Alloc(wszFormatName, formatNameLength, nullptr);
545 if (!format->formatName)
546 goto error_out;
547 }
548
549 index++;
550 }
551 }
552
553 return CHANNEL_RC_OK;
554
555error_out:
556 cliprdr_free_format_list(formatList);
557 return error;
558}
559
560void cliprdr_free_format_list(CLIPRDR_FORMAT_LIST* formatList)
561{
562 if (formatList == nullptr)
563 return;
564
565 if (formatList->formats)
566 {
567 for (UINT32 index = 0; index < formatList->numFormats; index++)
568 {
569 free(formatList->formats[index].formatName);
570 }
571
572 free(formatList->formats);
573 formatList->formats = nullptr;
574 formatList->numFormats = 0;
575 }
576}