22#include <freerdp/config.h>
25#include <winpr/assert.h>
26#include <winpr/print.h>
27#include <winpr/stream.h>
29#include <freerdp/freerdp.h>
30#include <freerdp/channels/log.h>
31#include "cliprdr_main.h"
32#include "../cliprdr_common.h"
87 WINPR_ASSERT(cliprdr);
89 pos = Stream_GetPosition(s);
90 if ((pos < 8) || (pos > UINT32_MAX))
96 dataLen = (UINT32)(pos - 8);
97 Stream_SetPosition(s, 4);
98 Stream_Write_UINT32(s, dataLen);
100 WINPR_ASSERT(pos <= UINT32_MAX);
101 status = WTSVirtualChannelWrite(cliprdr->ChannelHandle, Stream_BufferAs(s,
char), (UINT32)pos,
103 rc = status ? CHANNEL_RC_OK : ERROR_INTERNAL_ERROR;
105 Stream_Free(s, TRUE);
114static UINT cliprdr_server_capabilities(CliprdrServerContext* context,
119 WINPR_ASSERT(context);
120 WINPR_ASSERT(capabilities);
124 if (capabilities->common.msgType != CB_CLIP_CAPS)
125 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, capabilities->common.msgType);
127 if (capabilities->cCapabilitiesSets > UINT16_MAX)
129 WLog_ERR(TAG,
"Invalid number of capability sets in clipboard caps");
130 return ERROR_INVALID_PARAMETER;
133 wStream* s = cliprdr_packet_new(CB_CLIP_CAPS, 0, 4 + CB_CAPSTYPE_GENERAL_LEN);
137 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
138 return ERROR_INTERNAL_ERROR;
141 Stream_Write_UINT16(s,
142 (UINT16)capabilities->cCapabilitiesSets);
143 Stream_Write_UINT16(s, 0);
144 for (UINT32 x = 0; x < capabilities->cCapabilitiesSets; x++)
148 offset += cap->capabilitySetLength;
150 switch (cap->capabilitySetType)
152 case CB_CAPSTYPE_GENERAL:
157 s, generalCapabilitySet->capabilitySetType);
159 s, generalCapabilitySet->capabilitySetLength);
160 Stream_Write_UINT32(s, generalCapabilitySet->version);
162 s, generalCapabilitySet->generalFlags);
167 WLog_WARN(TAG,
"Unknown capability set type %08" PRIx16, cap->capabilitySetType);
168 if (!Stream_SafeSeek(s, cap->capabilitySetLength))
170 WLog_ERR(TAG,
"short stream");
171 Stream_Free(s, TRUE);
172 return ERROR_NO_DATA;
177 WLog_DBG(TAG,
"ServerCapabilities");
178 return cliprdr_server_packet_send(cliprdr, s);
186static UINT cliprdr_server_monitor_ready(CliprdrServerContext* context,
192 WINPR_ASSERT(context);
193 WINPR_ASSERT(monitorReady);
197 if (monitorReady->common.msgType != CB_MONITOR_READY)
198 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, monitorReady->common.msgType);
200 s = cliprdr_packet_new(CB_MONITOR_READY, monitorReady->common.msgFlags,
201 monitorReady->common.dataLen);
205 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
206 return ERROR_INTERNAL_ERROR;
209 WLog_DBG(TAG,
"ServerMonitorReady");
210 return cliprdr_server_packet_send(cliprdr, s);
218static UINT cliprdr_server_format_list(CliprdrServerContext* context,
224 WINPR_ASSERT(context);
225 WINPR_ASSERT(formatList);
229 s = cliprdr_packet_format_list_new(formatList, context->useLongFormatNames, FALSE);
232 WLog_ERR(TAG,
"cliprdr_packet_format_list_new failed!");
233 return ERROR_INTERNAL_ERROR;
236 WLog_DBG(TAG,
"ServerFormatList: numFormats: %" PRIu32
"", formatList->numFormats);
237 return cliprdr_server_packet_send(cliprdr, s);
246cliprdr_server_format_list_response(CliprdrServerContext* context,
252 WINPR_ASSERT(context);
253 WINPR_ASSERT(formatListResponse);
256 if (formatListResponse->common.msgType != CB_FORMAT_LIST_RESPONSE)
257 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, formatListResponse->common.msgType);
259 s = cliprdr_packet_new(CB_FORMAT_LIST_RESPONSE, formatListResponse->common.msgFlags,
260 formatListResponse->common.dataLen);
264 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
265 return ERROR_INTERNAL_ERROR;
268 WLog_DBG(TAG,
"ServerFormatListResponse");
269 return cliprdr_server_packet_send(cliprdr, s);
277static UINT cliprdr_server_lock_clipboard_data(CliprdrServerContext* context,
283 WINPR_ASSERT(context);
284 WINPR_ASSERT(lockClipboardData);
287 if (lockClipboardData->common.msgType != CB_LOCK_CLIPDATA)
288 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, lockClipboardData->common.msgType);
290 s = cliprdr_packet_lock_clipdata_new(lockClipboardData);
293 WLog_ERR(TAG,
"cliprdr_packet_lock_clipdata_new failed!");
294 return ERROR_INTERNAL_ERROR;
297 WLog_DBG(TAG,
"ServerLockClipboardData: clipDataId: 0x%08" PRIX32
"",
298 lockClipboardData->clipDataId);
299 return cliprdr_server_packet_send(cliprdr, s);
308cliprdr_server_unlock_clipboard_data(CliprdrServerContext* context,
314 WINPR_ASSERT(context);
315 WINPR_ASSERT(unlockClipboardData);
318 if (unlockClipboardData->common.msgType != CB_UNLOCK_CLIPDATA)
319 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, unlockClipboardData->common.msgType);
321 s = cliprdr_packet_unlock_clipdata_new(unlockClipboardData);
325 WLog_ERR(TAG,
"cliprdr_packet_unlock_clipdata_new failed!");
326 return ERROR_INTERNAL_ERROR;
329 WLog_DBG(TAG,
"ServerUnlockClipboardData: clipDataId: 0x%08" PRIX32
"",
330 unlockClipboardData->clipDataId);
331 return cliprdr_server_packet_send(cliprdr, s);
339static UINT cliprdr_server_format_data_request(CliprdrServerContext* context,
345 WINPR_ASSERT(context);
346 WINPR_ASSERT(formatDataRequest);
349 if (formatDataRequest->common.msgType != CB_FORMAT_DATA_REQUEST)
350 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, formatDataRequest->common.msgType);
352 s = cliprdr_packet_new(CB_FORMAT_DATA_REQUEST, formatDataRequest->common.msgFlags,
353 formatDataRequest->common.dataLen);
357 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
358 return ERROR_INTERNAL_ERROR;
361 Stream_Write_UINT32(s, formatDataRequest->requestedFormatId);
362 WLog_DBG(TAG,
"ClientFormatDataRequest");
363 return cliprdr_server_packet_send(cliprdr, s);
372cliprdr_server_format_data_response(CliprdrServerContext* context,
378 WINPR_ASSERT(context);
379 WINPR_ASSERT(formatDataResponse);
383 if (formatDataResponse->common.msgType != CB_FORMAT_DATA_RESPONSE)
384 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, formatDataResponse->common.msgType);
386 s = cliprdr_packet_new(CB_FORMAT_DATA_RESPONSE, formatDataResponse->common.msgFlags,
387 formatDataResponse->common.dataLen);
391 WLog_ERR(TAG,
"cliprdr_packet_new failed!");
392 return ERROR_INTERNAL_ERROR;
395 Stream_Write(s, formatDataResponse->requestedFormatData, formatDataResponse->common.dataLen);
396 WLog_DBG(TAG,
"ServerFormatDataResponse");
397 return cliprdr_server_packet_send(cliprdr, s);
406cliprdr_server_file_contents_request(CliprdrServerContext* context,
412 WINPR_ASSERT(context);
413 WINPR_ASSERT(fileContentsRequest);
417 if (fileContentsRequest->common.msgType != CB_FILECONTENTS_REQUEST)
418 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, fileContentsRequest->common.msgType);
420 s = cliprdr_packet_file_contents_request_new(fileContentsRequest);
423 WLog_ERR(TAG,
"cliprdr_packet_file_contents_request_new failed!");
424 return ERROR_INTERNAL_ERROR;
427 WLog_DBG(TAG,
"ServerFileContentsRequest: streamId: 0x%08" PRIX32
"",
428 fileContentsRequest->streamId);
429 return cliprdr_server_packet_send(cliprdr, s);
438cliprdr_server_file_contents_response(CliprdrServerContext* context,
444 WINPR_ASSERT(context);
445 WINPR_ASSERT(fileContentsResponse);
449 if (fileContentsResponse->common.msgType != CB_FILECONTENTS_RESPONSE)
450 WLog_WARN(TAG,
"called with invalid type %08" PRIx32, fileContentsResponse->common.msgType);
452 s = cliprdr_packet_file_contents_response_new(fileContentsResponse);
455 WLog_ERR(TAG,
"cliprdr_packet_file_contents_response_new failed!");
456 return ERROR_INTERNAL_ERROR;
459 WLog_DBG(TAG,
"ServerFileContentsResponse: streamId: 0x%08" PRIX32
"",
460 fileContentsResponse->streamId);
461 return cliprdr_server_packet_send(cliprdr, s);
469static UINT cliprdr_server_receive_general_capability(CliprdrServerContext* context,
wStream* s,
472 WINPR_ASSERT(context);
473 WINPR_ASSERT(cap_set);
475 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
476 return ERROR_INVALID_DATA;
478 Stream_Read_UINT32(s, cap_set->version);
479 Stream_Read_UINT32(s, cap_set->generalFlags);
481 if (context->useLongFormatNames)
482 context->useLongFormatNames = (cap_set->generalFlags & CB_USE_LONG_FORMAT_NAMES) != 0;
484 if (context->streamFileClipEnabled)
485 context->streamFileClipEnabled = (cap_set->generalFlags & CB_STREAM_FILECLIP_ENABLED) != 0;
487 if (context->fileClipNoFilePaths)
488 context->fileClipNoFilePaths = (cap_set->generalFlags & CB_FILECLIP_NO_FILE_PATHS) != 0;
490 if (context->canLockClipData)
491 context->canLockClipData = (cap_set->generalFlags & CB_CAN_LOCK_CLIPDATA) != 0;
493 if (context->hasHugeFileSupport)
494 context->hasHugeFileSupport = (cap_set->generalFlags & CB_HUGE_FILE_SUPPORT_ENABLED) != 0;
496 return CHANNEL_RC_OK;
504static UINT cliprdr_server_receive_capabilities(CliprdrServerContext* context,
wStream* s,
507 UINT16 capabilitySetType = 0;
508 UINT16 capabilitySetLength = 0;
509 UINT error = ERROR_INVALID_DATA;
510 size_t cap_sets_size = 0;
514 WINPR_ASSERT(context);
515 WINPR_UNUSED(header);
517 WLog_DBG(TAG,
"CliprdrClientCapabilities");
518 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
519 return ERROR_INVALID_DATA;
521 Stream_Read_UINT16(s, capabilities.cCapabilitiesSets);
522 Stream_Seek_UINT16(s);
524 for (
size_t index = 0; index < capabilities.cCapabilitiesSets; index++)
527 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
529 Stream_Read_UINT16(s, capabilitySetType);
530 Stream_Read_UINT16(s, capabilitySetLength);
532 cap_sets_size += capabilitySetLength;
534 if (cap_sets_size > 0)
535 tmp = realloc(capabilities.capabilitySets, cap_sets_size);
538 WLog_ERR(TAG,
"capabilities.capabilitySets realloc failed!");
539 free(capabilities.capabilitySets);
540 return CHANNEL_RC_NO_MEMORY;
545 capSet = &(capabilities.capabilitySets[index]);
547 capSet->capabilitySetType = capabilitySetType;
548 capSet->capabilitySetLength = capabilitySetLength;
550 switch (capSet->capabilitySetType)
552 case CB_CAPSTYPE_GENERAL:
553 error = cliprdr_server_receive_general_capability(
558 "cliprdr_server_receive_general_capability failed with error %" PRIu32
566 WLog_ERR(TAG,
"unknown cliprdr capability set: %" PRIu16
"",
567 capSet->capabilitySetType);
572 error = CHANNEL_RC_OK;
573 IFCALLRET(context->ClientCapabilities, error, context, &capabilities);
575 free(capabilities.capabilitySets);
584static UINT cliprdr_server_receive_temporary_directory(CliprdrServerContext* context,
wStream* s,
590 UINT error = CHANNEL_RC_OK;
592 WINPR_ASSERT(context);
593 WINPR_UNUSED(header);
596 WINPR_ASSERT(cliprdr);
598 if (!Stream_CheckAndLogRequiredLength(TAG, s,
599 ARRAYSIZE(cliprdr->temporaryDirectory) *
sizeof(WCHAR)))
600 return CHANNEL_RC_NO_MEMORY;
602 const WCHAR* wszTempDir = Stream_ConstPointer(s);
604 if (wszTempDir[ARRAYSIZE(cliprdr->temporaryDirectory) - 1] != 0)
606 WLog_ERR(TAG,
"wszTempDir[259] was not 0");
607 return ERROR_INVALID_DATA;
610 if (ConvertWCharNToUtf8(wszTempDir, ARRAYSIZE(cliprdr->temporaryDirectory),
611 cliprdr->temporaryDirectory,
612 ARRAYSIZE(cliprdr->temporaryDirectory)) < 0)
614 WLog_ERR(TAG,
"failed to convert temporary directory name");
615 return ERROR_INVALID_DATA;
618 length = strnlen(cliprdr->temporaryDirectory, ARRAYSIZE(cliprdr->temporaryDirectory));
620 if (length >= ARRAYSIZE(cliprdr->temporaryDirectory))
621 length = ARRAYSIZE(cliprdr->temporaryDirectory) - 1;
623 CopyMemory(tempDirectory.szTempDir, cliprdr->temporaryDirectory, length);
624 tempDirectory.szTempDir[length] =
'\0';
625 WLog_DBG(TAG,
"CliprdrTemporaryDirectory: %s", cliprdr->temporaryDirectory);
626 IFCALLRET(context->TempDirectory, error, context, &tempDirectory);
629 WLog_ERR(TAG,
"TempDirectory failed with error %" PRIu32
"!", error);
639static UINT cliprdr_server_receive_format_list(CliprdrServerContext* context,
wStream* s,
643 UINT error = CHANNEL_RC_OK;
645 WINPR_ASSERT(context);
646 WINPR_ASSERT(header);
648 formatList.common.msgType = CB_FORMAT_LIST;
649 formatList.common.msgFlags = header->msgFlags;
650 formatList.common.dataLen = header->dataLen;
652 wLog* log = WLog_Get(TAG);
653 if ((error = cliprdr_read_format_list(log, s, &formatList, context->useLongFormatNames)))
656 WLog_Print(log, WLOG_DEBUG,
"ClientFormatList: numFormats: %" PRIu32
"", formatList.numFormats);
657 IFCALLRET(context->ClientFormatList, error, context, &formatList);
660 WLog_Print(log, WLOG_ERROR,
"ClientFormatList failed with error %" PRIu32
"!", error);
663 cliprdr_free_format_list(&formatList);
672static UINT cliprdr_server_receive_format_list_response(CliprdrServerContext* context,
wStream* s,
676 UINT error = CHANNEL_RC_OK;
678 WINPR_ASSERT(context);
679 WINPR_ASSERT(header);
682 WLog_DBG(TAG,
"CliprdrClientFormatListResponse");
683 formatListResponse.common.msgType = CB_FORMAT_LIST_RESPONSE;
684 formatListResponse.common.msgFlags = header->msgFlags;
685 formatListResponse.common.dataLen = header->dataLen;
686 IFCALLRET(context->ClientFormatListResponse, error, context, &formatListResponse);
689 WLog_ERR(TAG,
"ClientFormatListResponse failed with error %" PRIu32
"!", error);
699static UINT cliprdr_server_receive_lock_clipdata(CliprdrServerContext* context,
wStream* s,
703 UINT error = CHANNEL_RC_OK;
705 WINPR_ASSERT(context);
706 WINPR_ASSERT(header);
708 WLog_DBG(TAG,
"CliprdrClientLockClipData");
710 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
711 return ERROR_INVALID_DATA;
713 lockClipboardData.common.msgType = CB_LOCK_CLIPDATA;
714 lockClipboardData.common.msgFlags = header->msgFlags;
715 lockClipboardData.common.dataLen = header->dataLen;
716 Stream_Read_UINT32(s, lockClipboardData.clipDataId);
717 IFCALLRET(context->ClientLockClipboardData, error, context, &lockClipboardData);
720 WLog_ERR(TAG,
"ClientLockClipboardData failed with error %" PRIu32
"!", error);
730static UINT cliprdr_server_receive_unlock_clipdata(CliprdrServerContext* context,
wStream* s,
734 UINT error = CHANNEL_RC_OK;
736 WINPR_ASSERT(context);
737 WINPR_ASSERT(header);
739 WLog_DBG(TAG,
"CliprdrClientUnlockClipData");
741 unlockClipboardData.common.msgType = CB_UNLOCK_CLIPDATA;
742 unlockClipboardData.common.msgFlags = header->msgFlags;
743 unlockClipboardData.common.dataLen = header->dataLen;
745 if ((error = cliprdr_read_unlock_clipdata(s, &unlockClipboardData)))
748 IFCALLRET(context->ClientUnlockClipboardData, error, context, &unlockClipboardData);
751 WLog_ERR(TAG,
"ClientUnlockClipboardData failed with error %" PRIu32
"!", error);
761static UINT cliprdr_server_receive_format_data_request(CliprdrServerContext* context,
wStream* s,
765 UINT error = CHANNEL_RC_OK;
767 WINPR_ASSERT(context);
768 WINPR_ASSERT(header);
770 WLog_DBG(TAG,
"CliprdrClientFormatDataRequest");
771 formatDataRequest.common.msgType = CB_FORMAT_DATA_REQUEST;
772 formatDataRequest.common.msgFlags = header->msgFlags;
773 formatDataRequest.common.dataLen = header->dataLen;
775 if ((error = cliprdr_read_format_data_request(s, &formatDataRequest)))
778 context->lastRequestedFormatId = formatDataRequest.requestedFormatId;
779 IFCALLRET(context->ClientFormatDataRequest, error, context, &formatDataRequest);
782 WLog_ERR(TAG,
"ClientFormatDataRequest failed with error %" PRIu32
"!", error);
792static UINT cliprdr_server_receive_format_data_response(CliprdrServerContext* context,
wStream* s,
796 UINT error = CHANNEL_RC_OK;
798 WINPR_ASSERT(context);
799 WINPR_ASSERT(header);
801 WLog_DBG(TAG,
"CliprdrClientFormatDataResponse");
802 formatDataResponse.common.msgType = CB_FORMAT_DATA_RESPONSE;
803 formatDataResponse.common.msgFlags = header->msgFlags;
804 formatDataResponse.common.dataLen = header->dataLen;
806 if ((error = cliprdr_read_format_data_response(s, &formatDataResponse)))
809 IFCALLRET(context->ClientFormatDataResponse, error, context, &formatDataResponse);
812 WLog_ERR(TAG,
"ClientFormatDataResponse failed with error %" PRIu32
"!", error);
822static UINT cliprdr_server_receive_filecontents_request(CliprdrServerContext* context,
wStream* s,
826 UINT error = CHANNEL_RC_OK;
828 WINPR_ASSERT(context);
829 WINPR_ASSERT(header);
831 WLog_DBG(TAG,
"CliprdrClientFileContentsRequest");
832 request.common.msgType = CB_FILECONTENTS_REQUEST;
833 request.common.msgFlags = header->msgFlags;
834 request.common.dataLen = header->dataLen;
836 if ((error = cliprdr_read_file_contents_request(s, &request)))
839 if (!context->hasHugeFileSupport)
841 if (request.nPositionHigh > 0)
842 return ERROR_INVALID_DATA;
843 if ((UINT64)request.nPositionLow + request.cbRequested > UINT32_MAX)
844 return ERROR_INVALID_DATA;
846 IFCALLRET(context->ClientFileContentsRequest, error, context, &request);
849 WLog_ERR(TAG,
"ClientFileContentsRequest failed with error %" PRIu32
"!", error);
859static UINT cliprdr_server_receive_filecontents_response(CliprdrServerContext* context,
wStream* s,
863 UINT error = CHANNEL_RC_OK;
865 WINPR_ASSERT(context);
866 WINPR_ASSERT(header);
868 WLog_DBG(TAG,
"CliprdrClientFileContentsResponse");
870 response.common.msgType = CB_FILECONTENTS_RESPONSE;
871 response.common.msgFlags = header->msgFlags;
872 response.common.dataLen = header->dataLen;
874 if ((error = cliprdr_read_file_contents_response(s, &response)))
877 IFCALLRET(context->ClientFileContentsResponse, error, context, &response);
880 WLog_ERR(TAG,
"ClientFileContentsResponse failed with error %" PRIu32
"!", error);
890static UINT cliprdr_server_receive_pdu(CliprdrServerContext* context,
wStream* s,
895 WINPR_ASSERT(context);
896 WINPR_ASSERT(header);
898 char buffer1[64] = WINPR_C_ARRAY_INIT;
899 char buffer2[64] = WINPR_C_ARRAY_INIT;
900 WLog_DBG(TAG,
"CliprdrServerReceivePdu: msgType: %s, msgFlags: %s dataLen: %" PRIu32
"",
901 CB_MSG_TYPE_STRING(header->msgType, buffer1,
sizeof(buffer1)),
902 CB_MSG_FLAGS_STRING(header->msgFlags, buffer2,
sizeof(buffer2)), header->dataLen);
904 switch (header->msgType)
907 if ((error = cliprdr_server_receive_capabilities(context, s, header)))
908 WLog_ERR(TAG,
"cliprdr_server_receive_capabilities failed with error %" PRIu32
"!",
913 case CB_TEMP_DIRECTORY:
914 if ((error = cliprdr_server_receive_temporary_directory(context, s, header)))
916 "cliprdr_server_receive_temporary_directory failed with error %" PRIu32
923 if ((error = cliprdr_server_receive_format_list(context, s, header)))
924 WLog_ERR(TAG,
"cliprdr_server_receive_format_list failed with error %" PRIu32
"!",
929 case CB_FORMAT_LIST_RESPONSE:
930 if ((error = cliprdr_server_receive_format_list_response(context, s, header)))
932 "cliprdr_server_receive_format_list_response failed with error %" PRIu32
938 case CB_LOCK_CLIPDATA:
939 if ((error = cliprdr_server_receive_lock_clipdata(context, s, header)))
940 WLog_ERR(TAG,
"cliprdr_server_receive_lock_clipdata failed with error %" PRIu32
"!",
945 case CB_UNLOCK_CLIPDATA:
946 if ((error = cliprdr_server_receive_unlock_clipdata(context, s, header)))
948 "cliprdr_server_receive_unlock_clipdata failed with error %" PRIu32
"!",
953 case CB_FORMAT_DATA_REQUEST:
954 if ((error = cliprdr_server_receive_format_data_request(context, s, header)))
956 "cliprdr_server_receive_format_data_request failed with error %" PRIu32
962 case CB_FORMAT_DATA_RESPONSE:
963 if ((error = cliprdr_server_receive_format_data_response(context, s, header)))
965 "cliprdr_server_receive_format_data_response failed with error %" PRIu32
971 case CB_FILECONTENTS_REQUEST:
972 if ((error = cliprdr_server_receive_filecontents_request(context, s, header)))
974 "cliprdr_server_receive_filecontents_request failed with error %" PRIu32
980 case CB_FILECONTENTS_RESPONSE:
981 if ((error = cliprdr_server_receive_filecontents_response(context, s, header)))
983 "cliprdr_server_receive_filecontents_response failed with error %" PRIu32
990 error = ERROR_INVALID_DATA;
991 WLog_ERR(TAG,
"Unexpected clipboard PDU type: %" PRIu16
"", header->msgType);
1003static UINT cliprdr_server_init(CliprdrServerContext* context)
1005 UINT32 generalFlags = 0;
1011 WINPR_ASSERT(context);
1013 monitorReady.common.msgType = CB_MONITOR_READY;
1014 capabilities.common.msgType = CB_CLIP_CAPS;
1016 if (context->useLongFormatNames)
1017 generalFlags |= CB_USE_LONG_FORMAT_NAMES;
1019 if (context->streamFileClipEnabled)
1020 generalFlags |= CB_STREAM_FILECLIP_ENABLED;
1022 if (context->fileClipNoFilePaths)
1023 generalFlags |= CB_FILECLIP_NO_FILE_PATHS;
1025 if (context->canLockClipData)
1026 generalFlags |= CB_CAN_LOCK_CLIPDATA;
1028 if (context->hasHugeFileSupport)
1029 generalFlags |= CB_HUGE_FILE_SUPPORT_ENABLED;
1031 capabilities.common.msgType = CB_CLIP_CAPS;
1032 capabilities.common.msgFlags = 0;
1033 capabilities.common.dataLen = 4 + CB_CAPSTYPE_GENERAL_LEN;
1034 capabilities.cCapabilitiesSets = 1;
1036 generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
1037 generalCapabilitySet.capabilitySetLength = CB_CAPSTYPE_GENERAL_LEN;
1038 generalCapabilitySet.version = CB_CAPS_VERSION_2;
1039 generalCapabilitySet.generalFlags = generalFlags;
1041 if ((error = context->ServerCapabilities(context, &capabilities)))
1043 WLog_ERR(TAG,
"ServerCapabilities failed with error %" PRIu32
"!", error);
1047 if ((error = context->MonitorReady(context, &monitorReady)))
1049 WLog_ERR(TAG,
"MonitorReady failed with error %" PRIu32
"!", error);
1061static UINT cliprdr_server_read(CliprdrServerContext* context)
1064 size_t position = 0;
1065 DWORD BytesToRead = 0;
1066 DWORD BytesReturned = 0;
1072 WINPR_ASSERT(context);
1075 WINPR_ASSERT(cliprdr);
1079 if (Stream_GetPosition(s) < CLIPRDR_HEADER_LENGTH)
1082 BytesToRead = (UINT32)(CLIPRDR_HEADER_LENGTH - Stream_GetPosition(s));
1083 status = WaitForSingleObject(cliprdr->ChannelEvent, 0);
1085 if (status == WAIT_FAILED)
1087 error = GetLastError();
1088 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1092 if (status == WAIT_TIMEOUT)
1093 return CHANNEL_RC_OK;
1095 if (!WTSVirtualChannelRead(cliprdr->ChannelHandle, 0, Stream_Pointer(s), BytesToRead,
1098 WLog_ERR(TAG,
"WTSVirtualChannelRead failed!");
1099 return ERROR_INTERNAL_ERROR;
1102 Stream_Seek(s, BytesReturned);
1105 if (Stream_GetPosition(s) >= CLIPRDR_HEADER_LENGTH)
1107 position = Stream_GetPosition(s);
1108 Stream_SetPosition(s, 0);
1109 Stream_Read_UINT16(s, header.msgType);
1110 Stream_Read_UINT16(s, header.msgFlags);
1111 Stream_Read_UINT32(s, header.dataLen);
1113 if (!Stream_EnsureRemainingCapacity(s, header.dataLen))
1115 WLog_ERR(TAG,
"Stream_EnsureCapacity failed!");
1116 return CHANNEL_RC_NO_MEMORY;
1119 Stream_SetPosition(s, position);
1121 if (Stream_GetPosition(s) < (header.dataLen + CLIPRDR_HEADER_LENGTH))
1125 (UINT32)((header.dataLen + CLIPRDR_HEADER_LENGTH) - Stream_GetPosition(s));
1126 status = WaitForSingleObject(cliprdr->ChannelEvent, 0);
1128 if (status == WAIT_FAILED)
1130 error = GetLastError();
1131 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1135 if (status == WAIT_TIMEOUT)
1136 return CHANNEL_RC_OK;
1138 if (!WTSVirtualChannelRead(cliprdr->ChannelHandle, 0, Stream_Pointer(s), BytesToRead,
1141 WLog_ERR(TAG,
"WTSVirtualChannelRead failed!");
1142 return ERROR_INTERNAL_ERROR;
1145 Stream_Seek(s, BytesReturned);
1148 if (Stream_GetPosition(s) >= (header.dataLen + CLIPRDR_HEADER_LENGTH))
1150 Stream_SetPosition(s, (header.dataLen + CLIPRDR_HEADER_LENGTH));
1151 Stream_SealLength(s);
1152 Stream_SetPosition(s, CLIPRDR_HEADER_LENGTH);
1154 if ((error = cliprdr_server_receive_pdu(context, s, &header)))
1156 WLog_ERR(TAG,
"cliprdr_server_receive_pdu failed with error code %" PRIu32
"!",
1161 Stream_SetPosition(s, 0);
1163 status = WaitForSingleObject(cliprdr->ChannelEvent, 0);
1165 if (status == WAIT_FAILED)
1167 error = GetLastError();
1168 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1172 if (status == WAIT_TIMEOUT)
1173 return CHANNEL_RC_OK;
1178 if (!WTSVirtualChannelRead(cliprdr->ChannelHandle, 0, Stream_Pointer(s), BytesToRead,
1181 WLog_ERR(TAG,
"WTSVirtualChannelRead failed!");
1182 return ERROR_INTERNAL_ERROR;
1185 if (BytesReturned == 4)
1187 Stream_Read_UINT16(s, header.msgType);
1188 Stream_Read_UINT16(s, header.msgFlags);
1190 if (!header.msgType)
1193 Stream_SetPosition(s, 0);
1198 Stream_Seek(s, BytesReturned);
1203 return CHANNEL_RC_OK;
1206static DWORD WINAPI cliprdr_server_thread(LPVOID arg)
1210 HANDLE events[MAXIMUM_WAIT_OBJECTS] = WINPR_C_ARRAY_INIT;
1211 HANDLE ChannelEvent =
nullptr;
1212 CliprdrServerContext* context = (CliprdrServerContext*)arg;
1214 UINT error = CHANNEL_RC_OK;
1216 WINPR_ASSERT(context);
1219 WINPR_ASSERT(cliprdr);
1221 ChannelEvent = context->GetEventHandle(context);
1223 events[nCount++] = cliprdr->StopEvent;
1224 events[nCount++] = ChannelEvent;
1226 if (context->autoInitializationSequence)
1228 if ((error = cliprdr_server_init(context)))
1230 WLog_ERR(TAG,
"cliprdr_server_init failed with error %" PRIu32
"!", error);
1237 status = WaitForMultipleObjects(nCount, events, FALSE, INFINITE);
1239 if (status == WAIT_FAILED)
1241 error = GetLastError();
1242 WLog_ERR(TAG,
"WaitForMultipleObjects failed with error %" PRIu32
"", error);
1246 status = WaitForSingleObject(cliprdr->StopEvent, 0);
1248 if (status == WAIT_FAILED)
1250 error = GetLastError();
1251 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1255 if (status == WAIT_OBJECT_0)
1258 status = WaitForSingleObject(ChannelEvent, 0);
1260 if (status == WAIT_FAILED)
1262 error = GetLastError();
1263 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1267 if (status == WAIT_OBJECT_0)
1269 if ((error = context->CheckEventHandle(context)))
1271 WLog_ERR(TAG,
"CheckEventHandle failed with error %" PRIu32
"!", error);
1279 if (error && context->rdpcontext)
1280 setChannelError(context->rdpcontext, error,
"cliprdr_server_thread reported an error");
1291static UINT cliprdr_server_open(CliprdrServerContext* context)
1293 void* buffer =
nullptr;
1294 DWORD BytesReturned = 0;
1297 WINPR_ASSERT(context);
1300 WINPR_ASSERT(cliprdr);
1302 cliprdr->ChannelHandle =
1303 WTSVirtualChannelOpen(cliprdr->vcm, WTS_CURRENT_SESSION, CLIPRDR_SVC_CHANNEL_NAME);
1305 if (!cliprdr->ChannelHandle)
1307 WLog_ERR(TAG,
"WTSVirtualChannelOpen for cliprdr failed!");
1308 return ERROR_INTERNAL_ERROR;
1311 cliprdr->ChannelEvent =
nullptr;
1313 if (WTSVirtualChannelQuery(cliprdr->ChannelHandle, WTSVirtualEventHandle, &buffer,
1316 if (BytesReturned !=
sizeof(HANDLE))
1318 WLog_ERR(TAG,
"BytesReturned has not size of HANDLE!");
1319 return ERROR_INTERNAL_ERROR;
1322 cliprdr->ChannelEvent = *(HANDLE*)buffer;
1323 WTSFreeMemory(buffer);
1326 if (!cliprdr->ChannelEvent)
1328 WLog_ERR(TAG,
"WTSVirtualChannelQuery for cliprdr failed!");
1329 return ERROR_INTERNAL_ERROR;
1332 return CHANNEL_RC_OK;
1340static UINT cliprdr_server_close(CliprdrServerContext* context)
1344 WINPR_ASSERT(context);
1347 WINPR_ASSERT(cliprdr);
1349 if (cliprdr->ChannelHandle)
1351 (void)WTSVirtualChannelClose(cliprdr->ChannelHandle);
1352 cliprdr->ChannelHandle =
nullptr;
1355 return CHANNEL_RC_OK;
1363static UINT cliprdr_server_start(CliprdrServerContext* context)
1368 WINPR_ASSERT(context);
1371 WINPR_ASSERT(cliprdr);
1373 if (!cliprdr->ChannelHandle)
1375 if ((error = context->Open(context)))
1377 WLog_ERR(TAG,
"Open failed!");
1382 if (!(cliprdr->StopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
1384 WLog_ERR(TAG,
"CreateEvent failed!");
1385 return ERROR_INTERNAL_ERROR;
1388 if (!(cliprdr->Thread =
1389 CreateThread(
nullptr, 0, cliprdr_server_thread, (
void*)context, 0,
nullptr)))
1391 WLog_ERR(TAG,
"CreateThread failed!");
1392 (void)CloseHandle(cliprdr->StopEvent);
1393 cliprdr->StopEvent =
nullptr;
1394 return ERROR_INTERNAL_ERROR;
1397 return CHANNEL_RC_OK;
1405static UINT cliprdr_server_stop(CliprdrServerContext* context)
1407 UINT error = CHANNEL_RC_OK;
1410 WINPR_ASSERT(context);
1413 WINPR_ASSERT(cliprdr);
1415 if (cliprdr->StopEvent)
1417 (void)SetEvent(cliprdr->StopEvent);
1419 if (WaitForSingleObject(cliprdr->Thread, INFINITE) == WAIT_FAILED)
1421 error = GetLastError();
1422 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
1426 (void)CloseHandle(cliprdr->Thread);
1427 (void)CloseHandle(cliprdr->StopEvent);
1430 if (cliprdr->ChannelHandle)
1431 return context->Close(context);
1436static HANDLE cliprdr_server_get_event_handle(CliprdrServerContext* context)
1440 WINPR_ASSERT(context);
1443 WINPR_ASSERT(cliprdr);
1444 return cliprdr->ChannelEvent;
1452static UINT cliprdr_server_check_event_handle(CliprdrServerContext* context)
1454 return cliprdr_server_read(context);
1457CliprdrServerContext* cliprdr_server_context_new(HANDLE vcm)
1460 CliprdrServerContext* context = (CliprdrServerContext*)calloc(1,
sizeof(CliprdrServerContext));
1464 context->autoInitializationSequence = TRUE;
1465 context->Open = cliprdr_server_open;
1466 context->Close = cliprdr_server_close;
1467 context->Start = cliprdr_server_start;
1468 context->Stop = cliprdr_server_stop;
1469 context->GetEventHandle = cliprdr_server_get_event_handle;
1470 context->CheckEventHandle = cliprdr_server_check_event_handle;
1471 context->ServerCapabilities = cliprdr_server_capabilities;
1472 context->MonitorReady = cliprdr_server_monitor_ready;
1473 context->ServerFormatList = cliprdr_server_format_list;
1474 context->ServerFormatListResponse = cliprdr_server_format_list_response;
1475 context->ServerLockClipboardData = cliprdr_server_lock_clipboard_data;
1476 context->ServerUnlockClipboardData = cliprdr_server_unlock_clipboard_data;
1477 context->ServerFormatDataRequest = cliprdr_server_format_data_request;
1478 context->ServerFormatDataResponse = cliprdr_server_format_data_response;
1479 context->ServerFileContentsRequest = cliprdr_server_file_contents_request;
1480 context->ServerFileContentsResponse = cliprdr_server_file_contents_response;
1486 cliprdr->s = Stream_New(
nullptr, 4096);
1490 WLog_ERR(TAG,
"Stream_New failed!");
1491 free(context->handle);
1498 WLog_ERR(TAG,
"calloc failed!");
1507void cliprdr_server_context_free(CliprdrServerContext* context)
1518 Stream_Free(cliprdr->s, TRUE);
1521 free(context->handle);