24#include <winpr/stream.h>
25#include <freerdp/channels/log.h>
27#define TAG CHANNELS_TAG("cliprdr.common")
29#include "cliprdr_common.h"
31static const char* CB_MSG_TYPE_STR(UINT32 type)
36 return "CB_TYPE_NONE";
37 case CB_MONITOR_READY:
38 return "CB_MONITOR_READY";
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";
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";
64const char* CB_MSG_TYPE_STRING(UINT16 type,
char* buffer,
size_t size)
66 (void)_snprintf(buffer, size,
"%s [0x%04" PRIx16
"]", CB_MSG_TYPE_STR(type), type);
70const char* CB_MSG_FLAGS_STRING(UINT16 msgFlags,
char* buffer,
size_t size)
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,
"|");
79 const size_t len = strnlen(buffer, size);
81 winpr_str_append(
"NONE", buffer, size,
"");
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,
"|");
100 if (request->dwFlags & FILECONTENTS_SIZE)
102 if (request->cbRequested !=
sizeof(UINT64))
104 WLog_ERR(TAG,
"cbRequested must be %" PRIuz
", got %" PRIu32
"",
sizeof(UINT64),
105 request->cbRequested);
109 if (request->nPositionHigh != 0 || request->nPositionLow != 0)
111 WLog_ERR(TAG,
"nPositionHigh and nPositionLow must be set to 0");
119wStream* cliprdr_packet_new(UINT16 msgType, UINT16 msgFlags,
size_t dataLen)
121 WINPR_ASSERT(dataLen < UINT32_MAX);
122 wStream* s = Stream_New(
nullptr, dataLen + 8ULL);
126 WLog_ERR(TAG,
"Stream_New failed!");
130 Stream_Write_UINT16(s, msgType);
131 Stream_Write_UINT16(s, msgFlags);
133 Stream_Write_UINT32(s, 0);
137static void cliprdr_write_file_contents_request(
wStream* s,
140 Stream_Write_UINT32(s, request->streamId);
141 Stream_Write_UINT32(s, request->listIndex);
142 Stream_Write_UINT32(s, request->dwFlags);
143 Stream_Write_UINT32(s, request->nPositionLow);
144 Stream_Write_UINT32(s, request->nPositionHigh);
145 Stream_Write_UINT32(s, request->cbRequested);
147 if (request->haveClipDataId)
148 Stream_Write_UINT32(s, request->clipDataId);
151static inline void cliprdr_write_lock_unlock_clipdata(
wStream* s, UINT32 clipDataId)
153 Stream_Write_UINT32(s, clipDataId);
156static void cliprdr_write_lock_clipdata(
wStream* s,
159 cliprdr_write_lock_unlock_clipdata(s, lockClipboardData->clipDataId);
162static void cliprdr_write_unlock_clipdata(
wStream* s,
165 cliprdr_write_lock_unlock_clipdata(s, unlockClipboardData->clipDataId);
168static void cliprdr_write_file_contents_response(
wStream* s,
171 Stream_Write_UINT32(s, response->streamId);
172 Stream_Write(s, response->requestedData, response->cbRequested);
179 if (!lockClipboardData)
182 s = cliprdr_packet_new(CB_LOCK_CLIPDATA, 0, 4);
187 cliprdr_write_lock_clipdata(s, lockClipboardData);
196 if (!unlockClipboardData)
199 s = cliprdr_packet_new(CB_UNLOCK_CLIPDATA, 0, 4);
204 cliprdr_write_unlock_clipdata(s, unlockClipboardData);
215 s = cliprdr_packet_new(CB_FILECONTENTS_REQUEST, 0, 28);
220 cliprdr_write_file_contents_request(s, request);
231 s = cliprdr_packet_new(CB_FILECONTENTS_RESPONSE, response->common.msgFlags,
232 4 + response->cbRequested);
237 cliprdr_write_file_contents_response(s, response);
242 BOOL useLongFormatNames, BOOL useAsciiNames)
244 WINPR_ASSERT(formatList);
246 if (formatList->common.msgType != CB_FORMAT_LIST)
247 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, formatList->common.msgType);
249 if (useLongFormatNames && useAsciiNames)
250 WLog_WARN(TAG,
"called with invalid arguments useLongFormatNames=true && "
251 "useAsciiNames=true. useAsciiNames requires "
252 "useLongFormatNames=false, ignoring argument.");
254 const UINT32 length = formatList->numFormats * 36;
255 const size_t formatNameCharSize =
256 (useLongFormatNames || !useAsciiNames) ?
sizeof(WCHAR) :
sizeof(CHAR);
258 wStream* s = cliprdr_packet_new(CB_FORMAT_LIST, 0, length);
261 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
265 for (UINT32 index = 0; index < formatList->numFormats; index++)
269 const char* szFormatName = format->formatName;
270 size_t formatNameLength = 0;
272 formatNameLength = strlen(szFormatName);
274 size_t formatNameMaxLength = formatNameLength + 1;
275 if (!Stream_EnsureRemainingCapacity(s,
276 4 + MAX(32, formatNameMaxLength * formatNameCharSize)))
279 Stream_Write_UINT32(s, format->formatId);
281 if (!useLongFormatNames)
283 formatNameMaxLength = useAsciiNames ? 32 : 16;
284 formatNameLength = MIN(formatNameMaxLength - 1, formatNameLength);
287 if (szFormatName && (formatNameLength > 0))
291 Stream_Write(s, szFormatName, formatNameLength);
292 Stream_Zero(s, formatNameMaxLength - formatNameLength);
296 if (Stream_Write_UTF16_String_From_UTF8(s, formatNameMaxLength, szFormatName,
297 formatNameLength, TRUE) < 0)
302 Stream_Zero(s, formatNameMaxLength * formatNameCharSize);
308 Stream_Free(s, TRUE);
314 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
315 return ERROR_INVALID_DATA;
317 Stream_Read_UINT32(s, unlockClipboardData->clipDataId);
318 return CHANNEL_RC_OK;
323 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
324 return ERROR_INVALID_DATA;
326 Stream_Read_UINT32(s, request->requestedFormatId);
327 return CHANNEL_RC_OK;
332 response->requestedFormatData =
nullptr;
334 if (!Stream_CheckAndLogRequiredLength(TAG, s, response->common.dataLen))
335 return ERROR_INVALID_DATA;
337 if (response->common.dataLen > 0)
339 response->requestedFormatData = Stream_ConstPointer(s);
340 if (!Stream_SafeSeek(s, response->common.dataLen))
341 return ERROR_INVALID_DATA;
343 return CHANNEL_RC_OK;
348 if (!Stream_CheckAndLogRequiredLength(TAG, s, 24))
349 return ERROR_INVALID_DATA;
351 request->haveClipDataId = FALSE;
352 Stream_Read_UINT32(s, request->streamId);
353 Stream_Read_UINT32(s, request->listIndex);
354 Stream_Read_UINT32(s, request->dwFlags);
355 Stream_Read_UINT32(s, request->nPositionLow);
356 Stream_Read_UINT32(s, request->nPositionHigh);
357 Stream_Read_UINT32(s, request->cbRequested);
359 if (Stream_GetRemainingLength(s) >= 4)
361 Stream_Read_UINT32(s, request->clipDataId);
362 request->haveClipDataId = TRUE;
365 if (!cliprdr_validate_file_contents_request(request))
366 return ERROR_BAD_ARGUMENTS;
368 return CHANNEL_RC_OK;
373 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
374 return ERROR_INVALID_DATA;
376 Stream_Read_UINT32(s, response->streamId);
377 response->requestedData = Stream_ConstPointer(s);
379 if (response->common.dataLen < 4)
381 WLog_WARN(TAG,
"dataLen=%" PRIu32
" but expected >= 4", response->common.dataLen);
382 return ERROR_INVALID_DATA;
384 response->cbRequested = response->common.dataLen - 4;
385 return CHANNEL_RC_OK;
389 BOOL useLongFormatNames)
392 size_t formatNameLength = 0;
393 const char* szFormatName =
nullptr;
394 const WCHAR* wszFormatName =
nullptr;
395 wStream sub1buffer = WINPR_C_ARRAY_INIT;
397 UINT error = ERROR_INTERNAL_ERROR;
399 const BOOL asciiNames = (formatList->common.msgFlags & CB_ASCII_NAMES) != 0;
403 formatList->formats =
nullptr;
404 formatList->numFormats = 0;
407 Stream_StaticConstInit(&sub1buffer, Stream_ConstPointer(s), formatList->common.dataLen);
408 if (!Stream_SafeSeek(s, formatList->common.dataLen))
409 return ERROR_INVALID_DATA;
411 if (!formatList->common.dataLen)
414 else if (!useLongFormatNames)
416 const size_t cap = Stream_Capacity(sub1) / 36ULL;
417 if (cap > UINT32_MAX)
419 WLog_Print(log, WLOG_ERROR,
"Invalid short format list length: %" PRIuz
"", cap);
420 return ERROR_INTERNAL_ERROR;
422 formatList->numFormats = (UINT32)cap;
424 if (formatList->numFormats)
429 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
430 return CHANNEL_RC_NO_MEMORY;
433 formatList->formats = formats;
435 while (Stream_GetRemainingLength(sub1) >= 4)
437 if (index >= formatList->numFormats)
442 Stream_Read_UINT32(sub1, format->formatId);
452 szFormatName = Stream_ConstPointer(sub1);
453 wszFormatName = Stream_ConstPointer(sub1);
454 if (!Stream_SafeSeek(sub1, 32))
457 free(format->formatName);
458 format->formatName =
nullptr;
465 format->formatName = strndup(szFormatName, 31);
466 if (!format->formatName)
468 WLog_Print(log, WLOG_ERROR,
"malloc failed!");
469 error = CHANNEL_RC_NO_MEMORY;
476 if (wszFormatName[0])
478 format->formatName = ConvertWCharNToUtf8Alloc(wszFormatName, 16,
nullptr);
479 if (!format->formatName)
489 wStream sub2buffer = sub1buffer;
492 while (Stream_GetRemainingLength(sub1) > 0)
495 if (!Stream_SafeSeek(sub1, 4))
498 wszFormatName = Stream_ConstPointer(sub1);
499 rest = Stream_GetRemainingLength(sub1);
500 formatNameLength = _wcsnlen(wszFormatName, rest /
sizeof(WCHAR));
502 if (!Stream_SafeSeek(sub1, (formatNameLength + 1) *
sizeof(WCHAR)))
504 formatList->numFormats++;
507 if (formatList->numFormats)
512 WLog_Print(log, WLOG_ERROR,
"calloc failed!");
513 return CHANNEL_RC_NO_MEMORY;
516 formatList->formats = formats;
518 while (Stream_GetRemainingLength(sub2) >= 4)
520 if (index >= formatList->numFormats)
526 Stream_Read_UINT32(sub2, format->formatId);
528 free(format->formatName);
529 format->formatName =
nullptr;
531 wszFormatName = Stream_ConstPointer(sub2);
532 rest = Stream_GetRemainingLength(sub2);
533 formatNameLength = _wcsnlen(wszFormatName, rest /
sizeof(WCHAR));
534 if (!Stream_SafeSeek(sub2, (formatNameLength + 1) *
sizeof(WCHAR)))
537 if (formatNameLength)
540 ConvertWCharNToUtf8Alloc(wszFormatName, formatNameLength,
nullptr);
541 if (!format->formatName)
549 return CHANNEL_RC_OK;
552 cliprdr_free_format_list(formatList);
558 if (formatList ==
nullptr)
561 if (formatList->formats)
563 for (UINT32 index = 0; index < formatList->numFormats; index++)
565 free(formatList->formats[index].formatName);
568 free(formatList->formats);
569 formatList->formats =
nullptr;
570 formatList->numFormats = 0;