22 #include <freerdp/config.h>
28 #include <X11/Xatom.h>
31 #include <X11/extensions/Xfixes.h>
34 #include <winpr/crt.h>
35 #include <winpr/assert.h>
36 #include <winpr/image.h>
37 #include <winpr/stream.h>
38 #include <winpr/clipboard.h>
39 #include <winpr/path.h>
41 #include <freerdp/utils/signal.h>
42 #include <freerdp/log.h>
43 #include <freerdp/client/cliprdr.h>
44 #include <freerdp/channels/channels.h>
45 #include <freerdp/channels/cliprdr.h>
47 #include <freerdp/client/client_cliprdr_file.h>
49 #include "xf_cliprdr.h"
53 #define TAG CLIENT_TAG("x11.cliprdr")
55 #define MAX_CLIPBOARD_FORMATS 255
57 #ifdef WITH_DEBUG_CLIPRDR
58 #define DEBUG_CLIPRDR(...) WLog_DBG(TAG, __VA_ARGS__)
60 #define DEBUG_CLIPRDR(...) \
69 UINT32 formatToRequest;
83 UINT32 formatToRequest;
90 rdpChannels* channels;
91 CliprdrClientContext* context;
99 Atom timestamp_property_atom;
100 Time selection_ownership_timestamp;
102 Atom raw_transfer_atom;
103 Atom raw_format_list_atom;
105 UINT32 numClientFormats;
106 xfCliprdrFormat clientFormats[20];
108 UINT32 numServerFormats;
114 int requestedFormatId;
116 wHashTable* cachedData;
117 wHashTable* cachedRawData;
119 BOOL data_raw_format;
121 RequestedFormat* requestedFormat;
123 XSelectionEvent* respond;
132 size_t incr_data_length;
136 int xfixes_event_base;
137 int xfixes_error_base;
138 BOOL xfixes_supported;
142 UINT32 lastSentNumFormats;
143 CliprdrFileContext* file;
146 static const char mime_text_plain[] =
"text/plain";
147 static const char mime_uri_list[] =
"text/uri-list";
148 static const char mime_html[] =
"text/html";
149 static const char* mime_bitmap[] = {
"image/bmp",
"image/x-bmp",
"image/x-MS-bmp",
150 "image/x-win-bitmap" };
151 static const char mime_webp[] =
"image/webp";
152 static const char mime_png[] =
"image/png";
153 static const char mime_jpeg[] =
"image/jpeg";
154 static const char mime_tiff[] =
"image/tiff";
155 static const char* mime_images[] = { mime_webp, mime_png, mime_jpeg, mime_tiff };
157 static const char mime_gnome_copied_files[] =
"x-special/gnome-copied-files";
158 static const char mime_mate_copied_files[] =
"x-special/mate-copied-files";
160 static const char type_FileGroupDescriptorW[] =
"FileGroupDescriptorW";
161 static const char type_HtmlFormat[] =
"HTML Format";
163 static void xf_cliprdr_clear_cached_data(xfClipboard* clipboard);
164 static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard, BOOL force);
165 static void xf_cliprdr_set_selection_owner(xfContext* xfc, xfClipboard* clipboard, Time timestamp);
167 static void requested_format_free(RequestedFormat** ppRequestedFormat)
169 if (!ppRequestedFormat)
171 if (!(*ppRequestedFormat))
174 free((*ppRequestedFormat)->formatName);
175 free(*ppRequestedFormat);
176 *ppRequestedFormat = NULL;
179 static BOOL requested_format_replace(RequestedFormat** ppRequestedFormat, UINT32 formatId,
180 const char* formatName)
182 if (!ppRequestedFormat)
185 requested_format_free(ppRequestedFormat);
186 RequestedFormat* requested = calloc(1,
sizeof(RequestedFormat));
189 requested->localFormat = formatId;
190 requested->formatToRequest = formatId;
193 requested->formatName = _strdup(formatName);
194 if (!requested->formatName)
201 *ppRequestedFormat = requested;
205 static BOOL requested_format_matches(
const RequestedFormat* pRequestedFormat, UINT32 formatId,
206 const char* formatName)
208 if (!pRequestedFormat)
210 if (pRequestedFormat->formatToRequest != formatId)
212 if (formatName || pRequestedFormat->formatName)
214 if (!formatName || !pRequestedFormat->formatName)
216 if (strcmp(formatName, pRequestedFormat->formatName) != 0)
222 static void xf_cached_data_free(
void* ptr)
224 xfCachedData* cached_data = ptr;
228 free(cached_data->data);
232 static xfCachedData* xf_cached_data_new(BYTE* data,
size_t data_length)
234 if (data_length > UINT32_MAX)
237 xfCachedData* cached_data = calloc(1,
sizeof(xfCachedData));
241 cached_data->data = data;
242 cached_data->data_length = (UINT32)data_length;
247 static xfCachedData* xf_cached_data_new_copy(
const BYTE* data,
size_t data_length)
252 copy = calloc(data_length + 1,
sizeof(BYTE));
255 memcpy(copy, data, data_length);
258 xfCachedData* cache = xf_cached_data_new(copy, data_length);
264 static void xf_clipboard_free_server_formats(xfClipboard* clipboard)
266 WINPR_ASSERT(clipboard);
267 if (clipboard->serverFormats)
269 for (
size_t i = 0; i < clipboard->numServerFormats; i++)
272 free(format->formatName);
275 free(clipboard->serverFormats);
276 clipboard->serverFormats = NULL;
280 static BOOL xf_cliprdr_update_owner(xfClipboard* clipboard)
282 WINPR_ASSERT(clipboard);
284 xfContext* xfc = clipboard->xfc;
287 if (!clipboard->sync)
290 Window owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
291 if (clipboard->owner == owner)
294 clipboard->owner = owner;
298 static void xf_cliprdr_check_owner(xfClipboard* clipboard)
300 if (xf_cliprdr_update_owner(clipboard))
301 xf_cliprdr_send_client_format_list(clipboard, FALSE);
304 static BOOL xf_cliprdr_is_self_owned(xfClipboard* clipboard)
306 xfContext* xfc = NULL;
308 WINPR_ASSERT(clipboard);
310 xfc = clipboard->xfc;
312 return XGetSelectionOwner(xfc->display, clipboard->clipboard_atom) == xfc->drawable;
315 static void xf_cliprdr_set_raw_transfer_enabled(xfClipboard* clipboard, BOOL enabled)
317 UINT32 data = WINPR_ASSERTING_INT_CAST(uint32_t, enabled);
318 xfContext* xfc = NULL;
320 WINPR_ASSERT(clipboard);
322 xfc = clipboard->xfc;
324 LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_transfer_atom,
325 XA_INTEGER, 32, PropModeReplace, (BYTE*)&data, 1);
328 static BOOL xf_cliprdr_is_raw_transfer_available(xfClipboard* clipboard)
333 unsigned long length = 0;
334 unsigned long bytes_left = 0;
336 UINT32 is_enabled = 0;
338 xfContext* xfc = NULL;
340 WINPR_ASSERT(clipboard);
342 xfc = clipboard->xfc;
345 owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
349 result = LogTagAndXGetWindowProperty(TAG, xfc->display, owner, clipboard->raw_transfer_atom,
350 0, 4, 0, XA_INTEGER, &type, &format, &length,
351 &bytes_left, (BYTE**)&data);
360 if ((owner == None) || (owner == xfc->drawable))
363 if (result != Success)
366 return is_enabled ? TRUE : FALSE;
369 static BOOL xf_cliprdr_formats_equal(
const CLIPRDR_FORMAT* server,
const xfCliprdrFormat* client)
371 WINPR_ASSERT(server);
372 WINPR_ASSERT(client);
374 if (server->formatName && client->formatName)
377 return (0 == strncmp(server->formatName, client->formatName, strlen(server->formatName)));
380 if (!server->formatName && !client->formatName)
382 return (server->formatId == client->formatToRequest);
388 static const xfCliprdrFormat* xf_cliprdr_get_client_format_by_id(xfClipboard* clipboard,
391 WINPR_ASSERT(clipboard);
393 for (
size_t index = 0; index < clipboard->numClientFormats; index++)
395 const xfCliprdrFormat* format = &(clipboard->clientFormats[index]);
397 if (format->formatToRequest == formatId)
404 static const xfCliprdrFormat* xf_cliprdr_get_client_format_by_atom(xfClipboard* clipboard,
407 WINPR_ASSERT(clipboard);
409 for (UINT32 i = 0; i < clipboard->numClientFormats; i++)
411 const xfCliprdrFormat* format = &(clipboard->clientFormats[i]);
413 if (format->atom == atom)
420 static const CLIPRDR_FORMAT* xf_cliprdr_get_server_format_by_atom(xfClipboard* clipboard, Atom atom)
422 WINPR_ASSERT(clipboard);
424 for (
size_t i = 0; i < clipboard->numClientFormats; i++)
426 const xfCliprdrFormat* client_format = &(clipboard->clientFormats[i]);
428 if (client_format->atom == atom)
430 for (
size_t j = 0; j < clipboard->numServerFormats; j++)
432 const CLIPRDR_FORMAT* server_format = &(clipboard->serverFormats[j]);
434 if (xf_cliprdr_formats_equal(server_format, client_format))
435 return server_format;
448 static UINT xf_cliprdr_send_data_request(xfClipboard* clipboard, UINT32 formatId,
449 const xfCliprdrFormat* cformat)
452 request.requestedFormatId = formatId;
454 DEBUG_CLIPRDR(
"requesting format 0x%08" PRIx32
" [%s] {local 0x%08" PRIx32
"} [%s]", formatId,
455 ClipboardGetFormatIdString(formatId), cformat->localFormat, cformat->formatName);
457 WINPR_ASSERT(clipboard);
458 WINPR_ASSERT(clipboard->context);
459 WINPR_ASSERT(clipboard->context->ClientFormatDataRequest);
460 return clipboard->context->ClientFormatDataRequest(clipboard->context, &request);
468 static UINT xf_cliprdr_send_data_response(xfClipboard* clipboard,
const xfCliprdrFormat* format,
469 const BYTE* data,
size_t size)
473 WINPR_ASSERT(clipboard);
476 if (clipboard->requestedFormatId < 0)
477 return CHANNEL_RC_OK;
482 DEBUG_CLIPRDR(
"send CB_RESPONSE_FAIL response {format 0x%08" PRIx32
483 " [%s] {local 0x%08" PRIx32
"} [%s]",
484 format->formatToRequest,
485 ClipboardGetFormatIdString(format->formatToRequest), format->localFormat,
488 DEBUG_CLIPRDR(
"send CB_RESPONSE_FAIL response");
492 WINPR_ASSERT(format);
493 DEBUG_CLIPRDR(
"send response format 0x%08" PRIx32
" [%s] {local 0x%08" PRIx32
"} [%s]",
494 format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
495 format->localFormat, format->formatName);
498 clipboard->requestedFormatId = -1;
500 response.common.msgFlags = (data) ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
502 WINPR_ASSERT(size <= UINT32_MAX);
503 response.common.dataLen = (UINT32)size;
504 response.requestedFormatData = data;
506 WINPR_ASSERT(clipboard->context);
507 WINPR_ASSERT(clipboard->context->ClientFormatDataResponse);
508 return clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
511 static wStream* xf_cliprdr_serialize_server_format_list(xfClipboard* clipboard)
513 UINT32 formatCount = 0;
516 WINPR_ASSERT(clipboard);
519 if (!(s = Stream_New(NULL, 128)))
521 WLog_ERR(TAG,
"failed to allocate serialized format list");
526 formatCount = (clipboard->numServerFormats > 0) ? clipboard->numServerFormats - 1 : 0;
527 Stream_Write_UINT32(s, formatCount);
529 for (UINT32 i = 0; i < formatCount; i++)
532 size_t name_length = format->formatName ? strlen(format->formatName) : 0;
534 DEBUG_CLIPRDR(
"server announced 0x%08" PRIx32
" [%s][%s]", format->formatId,
535 ClipboardGetFormatIdString(format->formatId), format->formatName);
536 if (!Stream_EnsureRemainingCapacity(s,
sizeof(UINT32) + name_length + 1))
538 WLog_ERR(TAG,
"failed to expand serialized format list");
542 Stream_Write_UINT32(s, format->formatId);
544 if (format->formatName)
545 Stream_Write(s, format->formatName, name_length);
547 Stream_Write_UINT8(s,
'\0');
550 Stream_SealLength(s);
553 Stream_Free(s, TRUE);
557 static CLIPRDR_FORMAT* xf_cliprdr_parse_server_format_list(BYTE* data,
size_t length,
563 WINPR_ASSERT(data || (length == 0));
564 WINPR_ASSERT(numFormats);
566 if (!(s = Stream_New(data, length)))
568 WLog_ERR(TAG,
"failed to allocate stream for parsing serialized format list");
572 if (!Stream_CheckAndLogRequiredLength(TAG, s,
sizeof(UINT32)))
575 Stream_Read_UINT32(s, *numFormats);
577 if (*numFormats > MAX_CLIPBOARD_FORMATS)
579 WLog_ERR(TAG,
"unexpectedly large number of formats: %" PRIu32
"", *numFormats);
585 WLog_ERR(TAG,
"failed to allocate format list");
589 for (UINT32 i = 0; i < *numFormats; i++)
591 const char* formatName = NULL;
592 size_t formatNameLength = 0;
594 if (!Stream_CheckAndLogRequiredLength(TAG, s,
sizeof(UINT32)))
597 Stream_Read_UINT32(s, formats[i].formatId);
598 formatName = (
const char*)Stream_Pointer(s);
599 formatNameLength = strnlen(formatName, Stream_GetRemainingLength(s));
601 if (formatNameLength == Stream_GetRemainingLength(s))
603 WLog_ERR(TAG,
"missing terminating null byte, %" PRIuz
" bytes left to read",
608 formats[i].formatName = strndup(formatName, formatNameLength);
609 Stream_Seek(s, formatNameLength + 1);
612 Stream_Free(s, FALSE);
615 Stream_Free(s, FALSE);
621 static void xf_cliprdr_free_formats(
CLIPRDR_FORMAT* formats, UINT32 numFormats)
623 WINPR_ASSERT(formats || (numFormats == 0));
625 for (UINT32 i = 0; i < numFormats; i++)
627 free(formats[i].formatName);
633 static CLIPRDR_FORMAT* xf_cliprdr_get_raw_server_formats(xfClipboard* clipboard, UINT32* numFormats)
637 unsigned long length = 0;
638 unsigned long remaining = 0;
641 xfContext* xfc = NULL;
643 WINPR_ASSERT(clipboard);
644 WINPR_ASSERT(numFormats);
646 xfc = clipboard->xfc;
651 Window owner = XGetSelectionOwner(xfc->display, clipboard->clipboard_atom);
652 LogTagAndXGetWindowProperty(TAG, xfc->display, owner, clipboard->raw_format_list_atom, 0, 4096,
653 False, clipboard->raw_format_list_atom, &type, &format, &length,
656 if (data && length > 0 && format == 8 && type == clipboard->raw_format_list_atom)
658 formats = xf_cliprdr_parse_server_format_list(data, length, numFormats);
663 "failed to retrieve raw format list: data=%p, length=%lu, format=%d, type=%lu "
665 (
void*)data, length, format, (
unsigned long)type,
666 (
unsigned long)clipboard->raw_format_list_atom);
675 static BOOL xf_cliprdr_should_add_format(
const CLIPRDR_FORMAT* formats,
size_t count,
676 const xfCliprdrFormat* xformat)
678 WINPR_ASSERT(formats);
683 for (
size_t x = 0; x < count; x++)
686 if (format->formatId == xformat->formatToRequest)
692 static CLIPRDR_FORMAT* xf_cliprdr_get_formats_from_targets(xfClipboard* clipboard,
697 int format_property = 0;
698 unsigned long length = 0;
699 unsigned long bytes_left = 0;
702 WINPR_ASSERT(clipboard);
703 WINPR_ASSERT(numFormats);
705 xfContext* xfc = clipboard->xfc;
709 LogTagAndXGetWindowProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0, 200,
710 0, XA_ATOM, &atom, &format_property, &length, &bytes_left, &data);
716 WLog_ERR(TAG,
"XGetWindowProperty set length = %lu but data is NULL", length);
722 WLog_ERR(TAG,
"failed to allocate %lu CLIPRDR_FORMAT structs", length);
727 for (
unsigned long i = 0; i < length; i++)
729 Atom tatom = ((Atom*)data)[i];
730 const xfCliprdrFormat* format = xf_cliprdr_get_client_format_by_atom(clipboard, tatom);
732 if (xf_cliprdr_should_add_format(formats, *numFormats, format))
735 cformat->formatId = format->formatToRequest;
736 if (format->formatName)
738 cformat->formatName = _strdup(format->formatName);
739 WINPR_ASSERT(cformat->formatName);
742 cformat->formatName = NULL;
756 static CLIPRDR_FORMAT* xf_cliprdr_get_client_formats(xfClipboard* clipboard, UINT32* numFormats)
760 WINPR_ASSERT(clipboard);
761 WINPR_ASSERT(numFormats);
765 if (xf_cliprdr_is_raw_transfer_available(clipboard))
767 formats = xf_cliprdr_get_raw_server_formats(clipboard, numFormats);
770 if (*numFormats == 0)
772 xf_cliprdr_free_formats(formats, *numFormats);
773 formats = xf_cliprdr_get_formats_from_targets(clipboard, numFormats);
779 static void xf_cliprdr_provide_server_format_list(xfClipboard* clipboard)
782 xfContext* xfc = NULL;
784 WINPR_ASSERT(clipboard);
786 xfc = clipboard->xfc;
789 formats = xf_cliprdr_serialize_server_format_list(clipboard);
793 const size_t len = Stream_Length(formats);
794 WINPR_ASSERT(len <= INT32_MAX);
795 LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_format_list_atom,
796 clipboard->raw_format_list_atom, 8, PropModeReplace,
797 Stream_Buffer(formats), (
int)len);
801 LogTagAndXDeleteProperty(TAG, xfc->display, xfc->drawable, clipboard->raw_format_list_atom);
804 Stream_Free(formats, TRUE);
812 if (a->formatId != b->formatId)
814 if (!a->formatName && !b->formatName)
816 if (!a->formatName || !b->formatName)
818 return strcmp(a->formatName, b->formatName) == 0;
821 static BOOL xf_clipboard_changed(xfClipboard* clipboard,
const CLIPRDR_FORMAT* formats,
824 WINPR_ASSERT(clipboard);
825 WINPR_ASSERT(formats || (numFormats == 0));
827 if (clipboard->lastSentNumFormats != numFormats)
830 for (UINT32 x = 0; x < numFormats; x++)
833 BOOL contained = FALSE;
834 for (UINT32 y = 0; y < numFormats; y++)
836 if (xf_clipboard_format_equal(cur, &formats[y]))
849 static void xf_clipboard_formats_free(xfClipboard* clipboard)
851 WINPR_ASSERT(clipboard);
853 xf_cliprdr_free_formats(clipboard->lastSentFormats, clipboard->lastSentNumFormats);
854 clipboard->lastSentFormats = NULL;
855 clipboard->lastSentNumFormats = 0;
858 static BOOL xf_clipboard_copy_formats(xfClipboard* clipboard,
const CLIPRDR_FORMAT* formats,
861 WINPR_ASSERT(clipboard);
862 WINPR_ASSERT(formats || (numFormats == 0));
864 xf_clipboard_formats_free(clipboard);
866 clipboard->lastSentFormats = calloc(numFormats,
sizeof(
CLIPRDR_FORMAT));
867 if (!clipboard->lastSentFormats)
869 clipboard->lastSentNumFormats = numFormats;
870 for (UINT32 x = 0; x < numFormats; x++)
876 lcur->formatName = _strdup(cur->formatName);
881 static UINT xf_cliprdr_send_format_list(xfClipboard* clipboard,
const CLIPRDR_FORMAT* formats,
882 UINT32 numFormats, BOOL force)
888 } cnv = { .cpv = formats };
890 .numFormats = numFormats,
892 .common.msgType = CB_FORMAT_LIST };
895 WINPR_ASSERT(clipboard);
896 WINPR_ASSERT(formats || (numFormats == 0));
898 if (!force && !xf_clipboard_changed(clipboard, formats, numFormats))
899 return CHANNEL_RC_OK;
901 #if defined(WITH_DEBUG_CLIPRDR)
902 for (UINT32 x = 0; x < numFormats; x++)
905 DEBUG_CLIPRDR(
"announcing format 0x%08" PRIx32
" [%s] [%s]", format->formatId,
906 ClipboardGetFormatIdString(format->formatId), format->formatName);
910 xf_clipboard_copy_formats(clipboard, formats, numFormats);
912 xf_cliprdr_send_data_response(clipboard, NULL, NULL, 0);
914 xf_cliprdr_clear_cached_data(clipboard);
916 ret = cliprdr_file_context_notify_new_client_format_list(clipboard->file);
920 WINPR_ASSERT(clipboard->context);
921 WINPR_ASSERT(clipboard->context->ClientFormatList);
922 return clipboard->context->ClientFormatList(clipboard->context, &formatList);
925 static void xf_cliprdr_get_requested_targets(xfClipboard* clipboard)
927 UINT32 numFormats = 0;
928 CLIPRDR_FORMAT* formats = xf_cliprdr_get_client_formats(clipboard, &numFormats);
929 xf_cliprdr_send_format_list(clipboard, formats, numFormats, FALSE);
930 xf_cliprdr_free_formats(formats, numFormats);
933 static void xf_cliprdr_process_requested_data(xfClipboard* clipboard, BOOL hasData,
934 const BYTE* data,
size_t size)
939 INT64 srcFormatId = -1;
940 BYTE* pDstData = NULL;
941 const xfCliprdrFormat* format = NULL;
943 WINPR_ASSERT(clipboard);
945 if (clipboard->incr_starts && hasData)
950 clipboard->incr_data_length = 0;
952 format = xf_cliprdr_get_client_format_by_id(
953 clipboard, WINPR_ASSERTING_INT_CAST(uint32_t, clipboard->requestedFormatId));
955 if (!hasData || !data || !format)
957 xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
961 switch (format->formatToRequest)
964 srcFormatId = CF_RAW;
970 srcFormatId = format->localFormat;
974 srcFormatId = format->localFormat;
980 xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
984 ClipboardLock(clipboard->system);
985 SrcSize = (UINT32)size;
986 bSuccess = ClipboardSetData(clipboard->system, (UINT32)srcFormatId, data, SrcSize);
991 pDstData = (BYTE*)ClipboardGetData(clipboard->system, format->formatToRequest, &DstSize);
993 ClipboardUnlock(clipboard->system);
997 xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
1008 ClipboardLock(clipboard->system);
1009 if (format->formatToRequest &&
1010 (format->formatToRequest ==
1011 ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW)))
1013 UINT error = NO_ERROR;
1019 const UINT32 flags = cliprdr_file_context_remote_get_flags(clipboard->file);
1020 error = cliprdr_serialize_file_list_ex(flags, file_array, file_count, &pDstData, &DstSize);
1023 WLog_ERR(TAG,
"failed to serialize CLIPRDR_FILELIST: 0x%08X", error);
1026 UINT32 formatId = ClipboardGetFormatId(clipboard->system, mime_uri_list);
1027 UINT32 url_size = 0;
1029 char* url = ClipboardGetData(clipboard->system, formatId, &url_size);
1030 cliprdr_file_context_update_client_data(clipboard->file, url, url_size);
1036 ClipboardUnlock(clipboard->system);
1038 xf_cliprdr_send_data_response(clipboard, format, pDstData, DstSize);
1042 static BOOL xf_add_input_flags(xfClipboard* clipboard,
long mask)
1044 WINPR_ASSERT(clipboard);
1046 xfContext* xfc = clipboard->xfc;
1049 XWindowAttributes attr = { 0 };
1050 XGetWindowAttributes(xfc->display, xfc->drawable, &attr);
1051 if ((attr.all_event_masks & mask) == 0)
1052 clipboard->event_mask = attr.all_event_masks;
1054 XSelectInput(xfc->display, xfc->drawable, attr.all_event_masks | mask);
1058 static BOOL xf_restore_input_flags(xfClipboard* clipboard)
1060 WINPR_ASSERT(clipboard);
1062 xfContext* xfc = clipboard->xfc;
1065 if (clipboard->event_mask != 0)
1067 XSelectInput(xfc->display, xfc->drawable, clipboard->event_mask);
1068 clipboard->event_mask = 0;
1073 static BOOL append(xfClipboard* clipboard,
const void* sdata,
size_t length)
1075 WINPR_ASSERT(clipboard);
1077 const size_t size = length + clipboard->incr_data_length + 2;
1078 BYTE* data = realloc(clipboard->incr_data, size);
1081 clipboard->incr_data = data;
1082 memcpy(&data[clipboard->incr_data_length], sdata, length);
1083 clipboard->incr_data_length += length;
1084 clipboard->incr_data[clipboard->incr_data_length + 0] =
'\0';
1085 clipboard->incr_data[clipboard->incr_data_length + 1] =
'\0';
1089 static BOOL xf_cliprdr_stop_incr(xfClipboard* clipboard)
1091 clipboard->incr_starts = FALSE;
1092 clipboard->incr_data_length = 0;
1093 return xf_restore_input_flags(clipboard);
1096 static BOOL xf_cliprdr_get_requested_data(xfClipboard* clipboard, Atom target)
1098 WINPR_ASSERT(clipboard);
1100 xfContext* xfc = clipboard->xfc;
1103 const xfCliprdrFormat* format = xf_cliprdr_get_client_format_by_id(
1104 clipboard, WINPR_ASSERTING_INT_CAST(uint32_t, clipboard->requestedFormatId));
1106 if (!format || (format->atom != target))
1108 xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
1113 BOOL has_data = FALSE;
1114 int format_property = 0;
1115 unsigned long length = 0;
1116 unsigned long total_bytes = 0;
1117 BYTE* property_data = NULL;
1118 const int rc = LogTagAndXGetWindowProperty(
1119 TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0, 0, False, target, &type,
1120 &format_property, &length, &total_bytes, &property_data);
1123 xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
1130 if ((total_bytes <= 0) && !clipboard->incr_starts)
1132 xf_cliprdr_stop_incr(clipboard);
1135 else if (type == clipboard->incr_atom)
1137 xf_cliprdr_stop_incr(clipboard);
1138 clipboard->incr_starts = TRUE;
1143 BYTE* incremental_data = NULL;
1144 unsigned long incremental_len = 0;
1147 len = clipboard->incr_data_length;
1148 if (total_bytes <= 0)
1150 xf_cliprdr_stop_incr(clipboard);
1154 else if (LogTagAndXGetWindowProperty(
1155 TAG, xfc->display, xfc->drawable, clipboard->property_atom, 0,
1156 WINPR_ASSERTING_INT_CAST(int32_t, total_bytes), False, target, &type,
1157 &format_property, &incremental_len, &length, &incremental_data) == Success)
1159 has_data = append(clipboard, incremental_data, incremental_len);
1160 len = clipboard->incr_data_length;
1163 if (incremental_data)
1164 XFree(incremental_data);
1167 LogTagAndXDeleteProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom);
1168 xf_cliprdr_process_requested_data(clipboard, has_data, clipboard->incr_data, len);
1173 static void xf_cliprdr_append_target(xfClipboard* clipboard, Atom target)
1175 WINPR_ASSERT(clipboard);
1177 if (clipboard->numTargets >= ARRAYSIZE(clipboard->targets))
1180 for (
size_t i = 0; i < clipboard->numTargets; i++)
1182 if (clipboard->targets[i] == target)
1186 clipboard->targets[clipboard->numTargets++] = target;
1189 static void xf_cliprdr_provide_targets(xfClipboard* clipboard,
const XSelectionEvent* respond)
1191 xfContext* xfc = NULL;
1193 WINPR_ASSERT(clipboard);
1195 xfc = clipboard->xfc;
1198 if (respond->property != None)
1200 WINPR_ASSERT(clipboard->numTargets <= INT32_MAX);
1201 LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property, XA_ATOM,
1202 32, PropModeReplace, (BYTE*)clipboard->targets,
1203 (
int)clipboard->numTargets);
1207 static void xf_cliprdr_provide_timestamp(xfClipboard* clipboard,
const XSelectionEvent* respond)
1209 xfContext* xfc = NULL;
1211 WINPR_ASSERT(clipboard);
1213 xfc = clipboard->xfc;
1216 if (respond->property != None)
1218 LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property,
1219 XA_INTEGER, 32, PropModeReplace,
1220 (BYTE*)&clipboard->selection_ownership_timestamp, 1);
1224 static void xf_cliprdr_provide_data(xfClipboard* clipboard,
const XSelectionEvent* respond,
1225 const BYTE* data, UINT32 size)
1227 xfContext* xfc = NULL;
1229 WINPR_ASSERT(clipboard);
1231 xfc = clipboard->xfc;
1234 if (respond->property != None)
1236 LogTagAndXChangeProperty(TAG, xfc->display, respond->requestor, respond->property,
1237 respond->target, 8, PropModeReplace, data,
1238 WINPR_ASSERTING_INT_CAST(int32_t, size));
1242 static void log_selection_event(xfContext* xfc,
const XEvent* event)
1244 const DWORD level = WLOG_TRACE;
1245 static wLog* _log_cached_ptr = NULL;
1246 if (!_log_cached_ptr)
1247 _log_cached_ptr = WLog_Get(TAG);
1248 if (WLog_IsLevelActive(_log_cached_ptr, level))
1251 switch (event->type)
1253 case SelectionClear:
1255 const XSelectionClearEvent* xevent = &
event->xselectionclear;
1257 Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
1258 WLog_Print(_log_cached_ptr, level,
"got event %s [selection %s]",
1259 x11_event_string(event->type), selection);
1263 case SelectionNotify:
1265 const XSelectionEvent* xevent = &
event->xselection;
1267 Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
1268 char* target = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->target);
1269 char*
property = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->property);
1270 WLog_Print(_log_cached_ptr, level,
1271 "got event %s [selection %s, target %s, property %s]",
1272 x11_event_string(event->type), selection, target, property);
1278 case SelectionRequest:
1280 const XSelectionRequestEvent* xevent = &
event->xselectionrequest;
1282 Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->selection);
1283 char* target = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->target);
1284 char*
property = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->property);
1285 WLog_Print(_log_cached_ptr, level,
1286 "got event %s [selection %s, target %s, property %s]",
1287 x11_event_string(event->type), selection, target, property);
1293 case PropertyNotify:
1295 const XPropertyEvent* xevent = &
event->xproperty;
1296 char* atom = Safe_XGetAtomName(_log_cached_ptr, xfc->display, xevent->atom);
1297 WLog_Print(_log_cached_ptr, level,
"got event %s [atom %s]",
1298 x11_event_string(event->type), atom);
1308 static BOOL xf_cliprdr_process_selection_notify(xfClipboard* clipboard,
1309 const XSelectionEvent* xevent)
1311 WINPR_ASSERT(clipboard);
1312 WINPR_ASSERT(xevent);
1314 if (xevent->target == clipboard->targets[1])
1316 if (xevent->property == None)
1318 xf_cliprdr_send_client_format_list(clipboard, FALSE);
1322 xf_cliprdr_get_requested_targets(clipboard);
1329 return xf_cliprdr_get_requested_data(clipboard, xevent->target);
1333 void xf_cliprdr_clear_cached_data(xfClipboard* clipboard)
1335 WINPR_ASSERT(clipboard);
1337 ClipboardLock(clipboard->system);
1338 ClipboardEmpty(clipboard->system);
1340 HashTable_Clear(clipboard->cachedData);
1341 HashTable_Clear(clipboard->cachedRawData);
1343 cliprdr_file_context_clear(clipboard->file);
1345 xf_cliprdr_stop_incr(clipboard);
1346 ClipboardUnlock(clipboard->system);
1349 static void* format_to_cache_slot(UINT32 format)
1356 cnv.uptr = 0x100000000ULL + format;
1360 static UINT32 get_dst_format_id_for_local_request(xfClipboard* clipboard,
1361 const xfCliprdrFormat* format)
1363 UINT32 dstFormatId = 0;
1365 WINPR_ASSERT(format);
1367 if (!format->formatName)
1368 return format->localFormat;
1370 ClipboardLock(clipboard->system);
1371 if (strcmp(format->formatName, type_HtmlFormat) == 0)
1372 dstFormatId = ClipboardGetFormatId(clipboard->system, mime_html);
1373 ClipboardUnlock(clipboard->system);
1375 if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
1376 dstFormatId = format->localFormat;
1381 static void get_src_format_info_for_local_request(xfClipboard* clipboard,
1382 const xfCliprdrFormat* format,
1383 UINT32* srcFormatId, BOOL* nullTerminated)
1386 *nullTerminated = FALSE;
1388 if (format->formatName)
1390 ClipboardLock(clipboard->system);
1391 if (strcmp(format->formatName, type_HtmlFormat) == 0)
1393 *srcFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
1394 *nullTerminated = TRUE;
1396 else if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
1398 *srcFormatId = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
1399 *nullTerminated = TRUE;
1401 ClipboardUnlock(clipboard->system);
1405 *srcFormatId = format->formatToRequest;
1406 switch (format->formatToRequest)
1410 case CF_UNICODETEXT:
1411 *nullTerminated = TRUE;
1414 *srcFormatId = CF_DIB;
1417 *srcFormatId = CF_TIFF;
1425 static xfCachedData* convert_data_from_existing_raw_data(xfClipboard* clipboard,
1426 xfCachedData* cached_raw_data,
1427 UINT32 srcFormatId, BOOL nullTerminated,
1430 xfCachedData* cached_data = NULL;
1432 BYTE* dst_data = NULL;
1433 UINT32 dst_size = 0;
1435 WINPR_ASSERT(clipboard);
1436 WINPR_ASSERT(cached_raw_data);
1437 WINPR_ASSERT(cached_raw_data->data);
1439 ClipboardLock(clipboard->system);
1440 success = ClipboardSetData(clipboard->system, srcFormatId, cached_raw_data->data,
1441 cached_raw_data->data_length);
1444 WLog_WARN(TAG,
"Failed to set clipboard data (formatId: %u, data: %p, data_length: %u)",
1445 srcFormatId, cached_raw_data->data, cached_raw_data->data_length);
1446 ClipboardUnlock(clipboard->system);
1450 dst_data = ClipboardGetData(clipboard->system, dstFormatId, &dst_size);
1453 WLog_WARN(TAG,
"Failed to get converted clipboard data");
1454 ClipboardUnlock(clipboard->system);
1457 ClipboardUnlock(clipboard->system);
1461 BYTE* nullTerminator = memchr(dst_data,
'\0', dst_size);
1464 const intptr_t diff = nullTerminator - dst_data;
1465 WINPR_ASSERT(diff >= 0);
1466 WINPR_ASSERT(diff <= UINT32_MAX);
1467 dst_size = (UINT32)diff;
1471 cached_data = xf_cached_data_new(dst_data, dst_size);
1474 WLog_WARN(TAG,
"Failed to allocate cache entry");
1479 if (!HashTable_Insert(clipboard->cachedData, format_to_cache_slot(dstFormatId), cached_data))
1481 WLog_WARN(TAG,
"Failed to cache clipboard data");
1482 xf_cached_data_free(cached_data);
1489 static BOOL xf_cliprdr_process_selection_request(xfClipboard* clipboard,
1490 const XSelectionRequestEvent* xevent)
1494 UINT32 formatId = 0;
1495 XSelectionEvent* respond = NULL;
1497 BOOL delayRespond = 0;
1498 BOOL rawTransfer = 0;
1499 unsigned long length = 0;
1500 unsigned long bytes_left = 0;
1501 xfContext* xfc = NULL;
1503 WINPR_ASSERT(clipboard);
1504 WINPR_ASSERT(xevent);
1506 xfc = clipboard->xfc;
1509 if (xevent->owner != xfc->drawable)
1512 delayRespond = FALSE;
1514 if (!(respond = (XSelectionEvent*)calloc(1,
sizeof(XSelectionEvent))))
1516 WLog_ERR(TAG,
"failed to allocate XEvent data");
1520 respond->property = None;
1521 respond->type = SelectionNotify;
1522 respond->display = xevent->display;
1523 respond->requestor = xevent->requestor;
1524 respond->selection = xevent->selection;
1525 respond->target = xevent->target;
1526 respond->time = xevent->time;
1528 if (xevent->target == clipboard->targets[0])
1531 respond->property = xevent->property;
1532 xf_cliprdr_provide_timestamp(clipboard, respond);
1534 else if (xevent->target == clipboard->targets[1])
1537 respond->property = xevent->property;
1538 xf_cliprdr_provide_targets(clipboard, respond);
1543 xf_cliprdr_get_server_format_by_atom(clipboard, xevent->target);
1544 const xfCliprdrFormat* cformat =
1545 xf_cliprdr_get_client_format_by_atom(clipboard, xevent->target);
1547 if (format && (xevent->requestor != xfc->drawable))
1549 formatId = format->formatId;
1550 rawTransfer = FALSE;
1551 xfCachedData* cached_data = NULL;
1552 UINT32 dstFormatId = 0;
1554 if (formatId == CF_RAW)
1556 if (LogTagAndXGetWindowProperty(
1557 TAG, xfc->display, xevent->requestor, clipboard->property_atom, 0, 4, 0,
1558 XA_INTEGER, &type, &fmt, &length, &bytes_left, &data) != Success)
1565 CopyMemory(&formatId, data, 4);
1570 dstFormatId = get_dst_format_id_for_local_request(clipboard, cformat);
1571 DEBUG_CLIPRDR(
"formatId: %u, dstFormatId: %u", formatId, dstFormatId);
1574 cached_data = HashTable_GetItemValue(clipboard->cachedData,
1575 format_to_cache_slot(dstFormatId));
1577 cached_data = HashTable_GetItemValue(clipboard->cachedRawData,
1578 format_to_cache_slot(formatId));
1580 DEBUG_CLIPRDR(
"hasCachedData: %u, rawTransfer: %u", cached_data ? 1 : 0, rawTransfer);
1582 if (!cached_data && !rawTransfer)
1584 UINT32 srcFormatId = 0;
1585 BOOL nullTerminated = FALSE;
1586 xfCachedData* cached_raw_data = NULL;
1588 get_src_format_info_for_local_request(clipboard, cformat, &srcFormatId,
1591 HashTable_GetItemValue(clipboard->cachedRawData, (
void*)(UINT_PTR)srcFormatId);
1593 DEBUG_CLIPRDR(
"hasCachedRawData: %u, rawDataLength: %u", cached_raw_data ? 1 : 0,
1594 cached_raw_data ? cached_raw_data->data_length : 0);
1596 if (cached_raw_data && cached_raw_data->data_length != 0)
1597 cached_data = convert_data_from_existing_raw_data(
1598 clipboard, cached_raw_data, srcFormatId, nullTerminated, dstFormatId);
1601 DEBUG_CLIPRDR(
"hasCachedData: %u", cached_data ? 1 : 0);
1606 respond->property = xevent->property;
1609 xf_cliprdr_provide_data(clipboard, respond, cached_data->data,
1610 cached_data->data_length);
1612 else if (clipboard->respond)
1618 WINPR_ASSERT(cformat);
1624 respond->property = xevent->property;
1625 clipboard->respond = respond;
1626 requested_format_replace(&clipboard->requestedFormat, formatId,
1627 cformat->formatName);
1628 clipboard->data_raw_format = rawTransfer;
1629 delayRespond = TRUE;
1630 xf_cliprdr_send_data_request(clipboard, formatId, cformat);
1640 XSelectionEvent* sev;
1644 XSendEvent(xfc->display, xevent->requestor, 0, 0, conv.ev);
1645 XFlush(xfc->display);
1652 static BOOL xf_cliprdr_process_selection_clear(xfClipboard* clipboard,
1653 const XSelectionClearEvent* xevent)
1655 xfContext* xfc = NULL;
1657 WINPR_ASSERT(clipboard);
1658 WINPR_ASSERT(xevent);
1660 xfc = clipboard->xfc;
1663 WINPR_UNUSED(xevent);
1665 if (xf_cliprdr_is_self_owned(clipboard))
1668 LogTagAndXDeleteProperty(TAG, xfc->display, clipboard->root_window, clipboard->property_atom);
1672 static BOOL xf_cliprdr_process_property_notify(xfClipboard* clipboard,
const XPropertyEvent* xevent)
1674 const xfCliprdrFormat* format = NULL;
1675 xfContext* xfc = NULL;
1680 xfc = clipboard->xfc;
1682 WINPR_ASSERT(xevent);
1684 if (xevent->atom == clipboard->timestamp_property_atom)
1690 xf_cliprdr_set_selection_owner(xfc, clipboard, xevent->time);
1694 if (xevent->atom != clipboard->property_atom)
1697 if (xevent->window == clipboard->root_window)
1699 xf_cliprdr_send_client_format_list(clipboard, FALSE);
1701 else if ((xevent->window == xfc->drawable) && (xevent->state == PropertyNewValue) &&
1702 clipboard->incr_starts)
1704 format = xf_cliprdr_get_client_format_by_id(
1705 clipboard, WINPR_ASSERTING_INT_CAST(uint32_t, clipboard->requestedFormatId));
1708 xf_cliprdr_get_requested_data(clipboard, format->atom);
1714 void xf_cliprdr_handle_xevent(xfContext* xfc,
const XEvent* event)
1716 xfClipboard* clipboard = NULL;
1721 clipboard = xfc->clipboard;
1728 if (clipboard->xfixes_supported &&
1729 event->type == XFixesSelectionNotify + clipboard->xfixes_event_base)
1731 const XFixesSelectionNotifyEvent* se = (
const XFixesSelectionNotifyEvent*)event;
1733 if (se->subtype == XFixesSetSelectionOwnerNotify)
1735 if (se->selection != clipboard->clipboard_atom)
1738 if (XGetSelectionOwner(xfc->display, se->selection) == xfc->drawable)
1741 clipboard->owner = None;
1742 xf_cliprdr_check_owner(clipboard);
1750 switch (event->type)
1752 case SelectionNotify:
1753 log_selection_event(xfc, event);
1754 xf_cliprdr_process_selection_notify(clipboard, &event->xselection);
1757 case SelectionRequest:
1758 log_selection_event(xfc, event);
1759 xf_cliprdr_process_selection_request(clipboard, &event->xselectionrequest);
1762 case SelectionClear:
1763 log_selection_event(xfc, event);
1764 xf_cliprdr_process_selection_clear(clipboard, &event->xselectionclear);
1767 case PropertyNotify:
1768 log_selection_event(xfc, event);
1769 xf_cliprdr_process_property_notify(clipboard, &event->xproperty);
1773 if (!clipboard->xfixes_supported)
1775 xf_cliprdr_check_owner(clipboard);
1789 static UINT xf_cliprdr_send_client_capabilities(xfClipboard* clipboard)
1794 WINPR_ASSERT(clipboard);
1796 capabilities.cCapabilitiesSets = 1;
1798 generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
1799 generalCapabilitySet.capabilitySetLength = 12;
1800 generalCapabilitySet.version = CB_CAPS_VERSION_2;
1801 generalCapabilitySet.generalFlags = CB_USE_LONG_FORMAT_NAMES;
1803 WINPR_ASSERT(clipboard);
1804 generalCapabilitySet.generalFlags |= cliprdr_file_context_current_flags(clipboard->file);
1806 WINPR_ASSERT(clipboard->context);
1807 WINPR_ASSERT(clipboard->context->ClientCapabilities);
1808 return clipboard->context->ClientCapabilities(clipboard->context, &capabilities);
1816 static UINT xf_cliprdr_send_client_format_list(xfClipboard* clipboard, BOOL force)
1818 WINPR_ASSERT(clipboard);
1820 xfContext* xfc = clipboard->xfc;
1823 UINT32 numFormats = 0;
1824 CLIPRDR_FORMAT* formats = xf_cliprdr_get_client_formats(clipboard, &numFormats);
1826 const UINT ret = xf_cliprdr_send_format_list(clipboard, formats, numFormats, force);
1828 if (clipboard->owner && clipboard->owner != xfc->drawable)
1831 LogTagAndXConvertSelection(TAG, xfc->display, clipboard->clipboard_atom,
1832 clipboard->targets[1], clipboard->property_atom, xfc->drawable,
1836 xf_cliprdr_free_formats(formats, numFormats);
1846 static UINT xf_cliprdr_send_client_format_list_response(xfClipboard* clipboard, BOOL status)
1850 formatListResponse.common.msgType = CB_FORMAT_LIST_RESPONSE;
1851 formatListResponse.common.msgFlags = status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
1852 formatListResponse.common.dataLen = 0;
1854 WINPR_ASSERT(clipboard);
1855 WINPR_ASSERT(clipboard->context);
1856 WINPR_ASSERT(clipboard->context->ClientFormatListResponse);
1857 return clipboard->context->ClientFormatListResponse(clipboard->context, &formatListResponse);
1865 static UINT xf_cliprdr_monitor_ready(CliprdrClientContext* context,
1869 xfClipboard* clipboard = NULL;
1871 WINPR_ASSERT(context);
1872 WINPR_ASSERT(monitorReady);
1874 clipboard = cliprdr_file_context_get_context(context->custom);
1875 WINPR_ASSERT(clipboard);
1877 WINPR_UNUSED(monitorReady);
1879 if ((ret = xf_cliprdr_send_client_capabilities(clipboard)) != CHANNEL_RC_OK)
1882 xf_clipboard_formats_free(clipboard);
1884 if ((ret = xf_cliprdr_send_client_format_list(clipboard, TRUE)) != CHANNEL_RC_OK)
1887 clipboard->sync = TRUE;
1888 return CHANNEL_RC_OK;
1896 static UINT xf_cliprdr_server_capabilities(CliprdrClientContext* context,
1900 const BYTE* capsPtr = NULL;
1901 xfClipboard* clipboard = NULL;
1903 WINPR_ASSERT(context);
1904 WINPR_ASSERT(capabilities);
1906 clipboard = cliprdr_file_context_get_context(context->custom);
1907 WINPR_ASSERT(clipboard);
1909 capsPtr = (
const BYTE*)capabilities->capabilitySets;
1910 WINPR_ASSERT(capsPtr);
1912 cliprdr_file_context_remote_set_flags(clipboard->file, 0);
1914 for (UINT32 i = 0; i < capabilities->cCapabilitiesSets; i++)
1918 if (caps->capabilitySetType == CB_CAPSTYPE_GENERAL)
1922 cliprdr_file_context_remote_set_flags(clipboard->file, generalCaps->generalFlags);
1925 capsPtr += caps->capabilitySetLength;
1928 return CHANNEL_RC_OK;
1931 static void xf_cliprdr_prepare_to_set_selection_owner(xfContext* xfc, xfClipboard* clipboard)
1934 WINPR_ASSERT(clipboard);
1951 Atom value = clipboard->timestamp_property_atom;
1953 LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->timestamp_property_atom,
1954 XA_ATOM, 32, PropModeReplace, (BYTE*)&value, 1);
1955 XFlush(xfc->display);
1958 static void xf_cliprdr_set_selection_owner(xfContext* xfc, xfClipboard* clipboard, Time timestamp)
1961 WINPR_ASSERT(clipboard);
1967 clipboard->selection_ownership_timestamp = timestamp;
1968 XSetSelectionOwner(xfc->display, clipboard->clipboard_atom, xfc->drawable, timestamp);
1969 XFlush(xfc->display);
1977 static UINT xf_cliprdr_server_format_list(CliprdrClientContext* context,
1980 xfContext* xfc = NULL;
1982 xfClipboard* clipboard = NULL;
1984 WINPR_ASSERT(context);
1985 WINPR_ASSERT(formatList);
1987 clipboard = cliprdr_file_context_get_context(context->custom);
1988 WINPR_ASSERT(clipboard);
1990 xfc = clipboard->xfc;
1996 free(clipboard->respond);
1997 clipboard->respond = NULL;
1999 xf_clipboard_formats_free(clipboard);
2000 xf_cliprdr_clear_cached_data(clipboard);
2001 requested_format_free(&clipboard->requestedFormat);
2003 xf_clipboard_free_server_formats(clipboard);
2005 clipboard->numServerFormats = formatList->numFormats + 1;
2007 if (!(clipboard->serverFormats =
2010 WLog_ERR(TAG,
"failed to allocate %d CLIPRDR_FORMAT structs", clipboard->numServerFormats);
2011 ret = CHANNEL_RC_NO_MEMORY;
2015 for (
size_t i = 0; i < formatList->numFormats; i++)
2020 srvFormat->formatId = format->formatId;
2022 if (format->formatName)
2024 srvFormat->formatName = _strdup(format->formatName);
2026 if (!srvFormat->formatName)
2028 for (UINT32 k = 0; k < i; k++)
2029 free(clipboard->serverFormats[k].formatName);
2031 clipboard->numServerFormats = 0;
2032 free(clipboard->serverFormats);
2033 clipboard->serverFormats = NULL;
2034 ret = CHANNEL_RC_NO_MEMORY;
2040 ret = cliprdr_file_context_notify_new_server_format_list(clipboard->file);
2046 CLIPRDR_FORMAT* format = &clipboard->serverFormats[formatList->numFormats];
2047 format->formatId = CF_RAW;
2048 format->formatName = NULL;
2050 xf_cliprdr_provide_server_format_list(clipboard);
2051 clipboard->numTargets = 2;
2053 for (
size_t i = 0; i < formatList->numFormats; i++)
2057 for (
size_t j = 0; j < clipboard->numClientFormats; j++)
2059 const xfCliprdrFormat* clientFormat = &clipboard->clientFormats[j];
2060 if (xf_cliprdr_formats_equal(format, clientFormat))
2062 if ((clientFormat->formatName != NULL) &&
2063 (strcmp(type_FileGroupDescriptorW, clientFormat->formatName) == 0))
2065 if (!cliprdr_file_context_has_local_support(clipboard->file))
2068 xf_cliprdr_append_target(clipboard, clientFormat->atom);
2073 ret = xf_cliprdr_send_client_format_list_response(clipboard, TRUE);
2074 if (xfc->remote_app)
2075 xf_cliprdr_set_selection_owner(xfc, clipboard, CurrentTime);
2077 xf_cliprdr_prepare_to_set_selection_owner(xfc, clipboard);
2091 xf_cliprdr_server_format_list_response(CliprdrClientContext* context,
2094 WINPR_ASSERT(context);
2095 WINPR_ASSERT(formatListResponse);
2097 return CHANNEL_RC_OK;
2106 xf_cliprdr_server_format_data_request(CliprdrClientContext* context,
2109 BOOL rawTransfer = 0;
2110 const xfCliprdrFormat* format = NULL;
2111 UINT32 formatId = 0;
2112 xfContext* xfc = NULL;
2113 xfClipboard* clipboard = NULL;
2115 WINPR_ASSERT(context);
2116 WINPR_ASSERT(formatDataRequest);
2118 clipboard = cliprdr_file_context_get_context(context->custom);
2119 WINPR_ASSERT(clipboard);
2121 xfc = clipboard->xfc;
2124 formatId = formatDataRequest->requestedFormatId;
2126 rawTransfer = xf_cliprdr_is_raw_transfer_available(clipboard);
2130 format = xf_cliprdr_get_client_format_by_id(clipboard, CF_RAW);
2131 LogTagAndXChangeProperty(TAG, xfc->display, xfc->drawable, clipboard->property_atom,
2132 XA_INTEGER, 32, PropModeReplace, (BYTE*)&formatId, 1);
2135 format = xf_cliprdr_get_client_format_by_id(clipboard, formatId);
2137 clipboard->requestedFormatId = rawTransfer ? CF_RAW : WINPR_ASSERTING_INT_CAST(
int, formatId);
2139 return xf_cliprdr_send_data_response(clipboard, format, NULL, 0);
2141 DEBUG_CLIPRDR(
"requested format 0x%08" PRIx32
" [%s] {local 0x%08" PRIx32
"} [%s]",
2142 format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
2143 format->localFormat, format->formatName);
2144 LogTagAndXConvertSelection(TAG, xfc->display, clipboard->clipboard_atom, format->atom,
2145 clipboard->property_atom, xfc->drawable, CurrentTime);
2146 XFlush(xfc->display);
2148 return CHANNEL_RC_OK;
2157 xf_cliprdr_server_format_data_response(CliprdrClientContext* context,
2161 BYTE* pDstData = NULL;
2164 UINT32 srcFormatId = 0;
2165 UINT32 dstFormatId = 0;
2166 BOOL nullTerminated = FALSE;
2168 const BYTE* data = NULL;
2169 xfContext* xfc = NULL;
2170 xfClipboard* clipboard = NULL;
2171 xfCachedData* cached_data = NULL;
2173 WINPR_ASSERT(context);
2174 WINPR_ASSERT(formatDataResponse);
2176 clipboard = cliprdr_file_context_get_context(context->custom);
2177 WINPR_ASSERT(clipboard);
2179 xfc = clipboard->xfc;
2182 size = formatDataResponse->common.dataLen;
2183 data = formatDataResponse->requestedFormatData;
2185 if (formatDataResponse->common.msgFlags == CB_RESPONSE_FAIL)
2187 WLog_WARN(TAG,
"Format Data Response PDU msgFlags is CB_RESPONSE_FAIL");
2188 free(clipboard->respond);
2189 clipboard->respond = NULL;
2190 return CHANNEL_RC_OK;
2193 if (!clipboard->respond)
2194 return CHANNEL_RC_OK;
2201 const RequestedFormat* format = clipboard->requestedFormat;
2202 if (clipboard->data_raw_format)
2204 srcFormatId = CF_RAW;
2205 dstFormatId = CF_RAW;
2208 return ERROR_INTERNAL_ERROR;
2209 else if (format->formatName)
2211 ClipboardLock(clipboard->system);
2212 if (strcmp(format->formatName, type_HtmlFormat) == 0)
2214 srcFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
2215 dstFormatId = ClipboardGetFormatId(clipboard->system, mime_html);
2216 nullTerminated = TRUE;
2219 if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
2221 if (!cliprdr_file_context_update_server_data(clipboard->file, clipboard->system, data,
2223 WLog_WARN(TAG,
"failed to update file descriptors");
2225 srcFormatId = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
2226 const xfCliprdrFormat* dstTargetFormat =
2227 xf_cliprdr_get_client_format_by_atom(clipboard, clipboard->respond->target);
2228 if (!dstTargetFormat)
2230 dstFormatId = ClipboardGetFormatId(clipboard->system, mime_uri_list);
2234 dstFormatId = dstTargetFormat->localFormat;
2237 nullTerminated = TRUE;
2239 ClipboardUnlock(clipboard->system);
2243 srcFormatId = format->formatToRequest;
2244 dstFormatId = format->localFormat;
2245 switch (format->formatToRequest)
2248 nullTerminated = TRUE;
2252 nullTerminated = TRUE;
2255 case CF_UNICODETEXT:
2256 nullTerminated = TRUE;
2260 srcFormatId = CF_DIB;
2264 srcFormatId = CF_TIFF;
2272 DEBUG_CLIPRDR(
"requested format 0x%08" PRIx32
" [%s] {local 0x%08" PRIx32
"} [%s]",
2273 format->formatToRequest, ClipboardGetFormatIdString(format->formatToRequest),
2274 format->localFormat, format->formatName);
2277 DEBUG_CLIPRDR(
"srcFormatId: %u, dstFormatId: %u", srcFormatId, dstFormatId);
2279 ClipboardLock(clipboard->system);
2280 bSuccess = ClipboardSetData(clipboard->system, srcFormatId, data, SrcSize);
2282 BOOL willQuit = FALSE;
2287 WLog_DBG(TAG,
"skipping, empty data detected!");
2288 free(clipboard->respond);
2289 clipboard->respond = NULL;
2294 pDstData = (BYTE*)ClipboardGetData(clipboard->system, dstFormatId, &DstSize);
2298 WLog_WARN(TAG,
"failed to get clipboard data in format %s [source format %s]",
2299 ClipboardGetFormatName(clipboard->system, dstFormatId),
2300 ClipboardGetFormatName(clipboard->system, srcFormatId));
2303 if (nullTerminated && pDstData)
2305 BYTE* nullTerminator = memchr(pDstData,
'\0', DstSize);
2308 const intptr_t diff = nullTerminator - pDstData;
2309 WINPR_ASSERT(diff >= 0);
2310 WINPR_ASSERT(diff <= UINT32_MAX);
2311 DstSize = (UINT32)diff;
2316 ClipboardUnlock(clipboard->system);
2318 return CHANNEL_RC_OK;
2324 cached_data = xf_cached_data_new(pDstData, DstSize);
2327 WLog_WARN(TAG,
"Failed to allocate cache entry");
2329 return CHANNEL_RC_OK;
2331 if (!HashTable_Insert(clipboard->cachedData, format_to_cache_slot(dstFormatId),
2334 WLog_WARN(TAG,
"Failed to cache clipboard data");
2335 xf_cached_data_free(cached_data);
2336 return CHANNEL_RC_OK;
2346 xfCachedData* cached_raw_data = xf_cached_data_new_copy(data, size);
2347 if (!cached_raw_data)
2348 WLog_WARN(TAG,
"Failed to allocate cache entry");
2351 if (!HashTable_Insert(clipboard->cachedRawData, (
void*)(UINT_PTR)srcFormatId,
2354 WLog_WARN(TAG,
"Failed to cache clipboard data");
2355 xf_cached_data_free(cached_raw_data);
2362 xf_cliprdr_provide_data(clipboard, clipboard->respond, pDstData, DstSize);
2367 XSelectionEvent* sev;
2370 conv.sev = clipboard->respond;
2372 XSendEvent(xfc->display, clipboard->respond->requestor, 0, 0, conv.ev);
2373 XFlush(xfc->display);
2375 free(clipboard->respond);
2376 clipboard->respond = NULL;
2377 return CHANNEL_RC_OK;
2380 static BOOL xf_cliprdr_is_valid_unix_filename(LPCWSTR filename)
2385 if (filename[0] == L
'\0')
2389 for (
const WCHAR* c = filename; *c; ++c)
2398 xfClipboard* xf_clipboard_new(xfContext* xfc, BOOL relieveFilenameRestriction)
2401 rdpChannels* channels = NULL;
2402 xfClipboard* clipboard = NULL;
2403 const char* selectionAtom = NULL;
2404 xfCliprdrFormat* clientFormat = NULL;
2408 WINPR_ASSERT(xfc->common.context.settings);
2410 if (!(clipboard = (xfClipboard*)calloc(1,
sizeof(xfClipboard))))
2412 WLog_ERR(TAG,
"failed to allocate xfClipboard data");
2416 clipboard->file = cliprdr_file_context_new(clipboard);
2417 if (!clipboard->file)
2420 xfc->clipboard = clipboard;
2421 clipboard->xfc = xfc;
2422 channels = xfc->common.context.channels;
2423 clipboard->channels = channels;
2424 clipboard->system = ClipboardCreate();
2425 clipboard->requestedFormatId = -1;
2426 clipboard->root_window = DefaultRootWindow(xfc->display);
2431 selectionAtom =
"CLIPBOARD";
2433 clipboard->clipboard_atom = Logging_XInternAtom(xfc->log, xfc->display, selectionAtom, FALSE);
2435 if (clipboard->clipboard_atom == None)
2437 WLog_ERR(TAG,
"unable to get %s atom", selectionAtom);
2441 clipboard->timestamp_property_atom =
2442 Logging_XInternAtom(xfc->log, xfc->display,
"_FREERDP_TIMESTAMP_PROPERTY", FALSE);
2443 clipboard->property_atom =
2444 Logging_XInternAtom(xfc->log, xfc->display,
"_FREERDP_CLIPRDR", FALSE);
2445 clipboard->raw_transfer_atom =
2446 Logging_XInternAtom(xfc->log, xfc->display,
"_FREERDP_CLIPRDR_RAW", FALSE);
2447 clipboard->raw_format_list_atom =
2448 Logging_XInternAtom(xfc->log, xfc->display,
"_FREERDP_CLIPRDR_FORMATS", FALSE);
2449 xf_cliprdr_set_raw_transfer_enabled(clipboard, TRUE);
2450 XSelectInput(xfc->display, clipboard->root_window, PropertyChangeMask);
2453 if (XFixesQueryExtension(xfc->display, &clipboard->xfixes_event_base,
2454 &clipboard->xfixes_error_base))
2459 if (XFixesQueryVersion(xfc->display, &xfmajor, &xfminor))
2461 XFixesSelectSelectionInput(xfc->display, clipboard->root_window,
2462 clipboard->clipboard_atom,
2463 XFixesSetSelectionOwnerNotifyMask);
2464 clipboard->xfixes_supported = TRUE;
2468 WLog_ERR(TAG,
"Error querying X Fixes extension version");
2473 WLog_ERR(TAG,
"Error loading X Fixes extension");
2479 "Warning: Using clipboard redirection without XFIXES extension is strongly discouraged!");
2481 clientFormat = &clipboard->clientFormats[n++];
2482 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display,
"_FREERDP_RAW", False);
2483 clientFormat->localFormat = clientFormat->formatToRequest = CF_RAW;
2485 clientFormat = &clipboard->clientFormats[n++];
2486 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display,
"UTF8_STRING", False);
2487 clientFormat->formatToRequest = CF_UNICODETEXT;
2488 clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_text_plain);
2490 clientFormat = &clipboard->clientFormats[n++];
2491 clientFormat->atom = XA_STRING;
2492 clientFormat->formatToRequest = CF_TEXT;
2493 clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_text_plain);
2495 clientFormat = &clipboard->clientFormats[n++];
2496 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_tiff, False);
2497 clientFormat->formatToRequest = clientFormat->localFormat = CF_TIFF;
2499 for (
size_t x = 0; x < ARRAYSIZE(mime_bitmap); x++)
2501 const char* mime_bmp = mime_bitmap[x];
2502 const DWORD format = ClipboardGetFormatId(xfc->clipboard->system, mime_bmp);
2505 WLog_DBG(TAG,
"skipping local bitmap format %s [NOT SUPPORTED]", mime_bmp);
2509 WLog_DBG(TAG,
"register local bitmap format %s [0x%08" PRIx32
"]", mime_bmp, format);
2510 clientFormat = &clipboard->clientFormats[n++];
2511 clientFormat->localFormat = format;
2512 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_bmp, False);
2513 clientFormat->formatToRequest = CF_DIB;
2516 for (
size_t x = 0; x < ARRAYSIZE(mime_images); x++)
2518 const char* mime_bmp = mime_images[x];
2519 const DWORD format = ClipboardGetFormatId(xfc->clipboard->system, mime_bmp);
2522 WLog_DBG(TAG,
"skipping local bitmap format %s [NOT SUPPORTED]", mime_bmp);
2526 WLog_DBG(TAG,
"register local bitmap format %s [0x%08" PRIx32
"]", mime_bmp, format);
2527 clientFormat = &clipboard->clientFormats[n++];
2528 clientFormat->localFormat = format;
2529 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_bmp, False);
2530 clientFormat->formatToRequest = CF_DIB;
2533 clientFormat = &clipboard->clientFormats[n++];
2534 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_html, False);
2535 clientFormat->formatToRequest = ClipboardGetFormatId(xfc->clipboard->system, type_HtmlFormat);
2536 clientFormat->localFormat = ClipboardGetFormatId(xfc->clipboard->system, mime_html);
2537 clientFormat->formatName = _strdup(type_HtmlFormat);
2539 if (!clientFormat->formatName)
2542 clientFormat = &clipboard->clientFormats[n++];
2550 const UINT32 fgid = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
2551 const UINT32 uid = ClipboardGetFormatId(clipboard->system, mime_uri_list);
2554 cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
2555 clientFormat->atom = Logging_XInternAtom(xfc->log, xfc->display, mime_uri_list, False);
2556 clientFormat->localFormat = uid;
2557 clientFormat->formatToRequest = fgid;
2558 clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
2560 if (!clientFormat->formatName)
2563 clientFormat = &clipboard->clientFormats[n++];
2566 const UINT32 gid = ClipboardGetFormatId(clipboard->system, mime_gnome_copied_files);
2569 cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
2570 clientFormat->atom =
2571 Logging_XInternAtom(xfc->log, xfc->display, mime_gnome_copied_files, False);
2572 clientFormat->localFormat = gid;
2573 clientFormat->formatToRequest = fgid;
2574 clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
2576 if (!clientFormat->formatName)
2579 clientFormat = &clipboard->clientFormats[n++];
2582 const UINT32 mid = ClipboardGetFormatId(clipboard->system, mime_mate_copied_files);
2585 cliprdr_file_context_set_locally_available(clipboard->file, TRUE);
2586 clientFormat->atom =
2587 Logging_XInternAtom(xfc->log, xfc->display, mime_mate_copied_files, False);
2588 clientFormat->localFormat = mid;
2589 clientFormat->formatToRequest = fgid;
2590 clientFormat->formatName = _strdup(type_FileGroupDescriptorW);
2592 if (!clientFormat->formatName)
2596 clipboard->numClientFormats = WINPR_ASSERTING_INT_CAST(uint32_t, n);
2597 clipboard->targets[0] = Logging_XInternAtom(xfc->log, xfc->display,
"TIMESTAMP", FALSE);
2598 clipboard->targets[1] = Logging_XInternAtom(xfc->log, xfc->display,
"TARGETS", FALSE);
2599 clipboard->numTargets = 2;
2600 clipboard->incr_atom = Logging_XInternAtom(xfc->log, xfc->display,
"INCR", FALSE);
2602 if (relieveFilenameRestriction)
2604 WLog_DBG(TAG,
"Relieving CLIPRDR filename restriction");
2605 ClipboardGetDelegate(clipboard->system)->IsFileNameComponentValid =
2606 xf_cliprdr_is_valid_unix_filename;
2609 clipboard->cachedData = HashTable_New(TRUE);
2610 if (!clipboard->cachedData)
2613 obj = HashTable_ValueObject(clipboard->cachedData);
2614 obj->fnObjectFree = xf_cached_data_free;
2616 clipboard->cachedRawData = HashTable_New(TRUE);
2617 if (!clipboard->cachedRawData)
2620 obj = HashTable_ValueObject(clipboard->cachedRawData);
2621 obj->fnObjectFree = xf_cached_data_free;
2626 WINPR_PRAGMA_DIAG_PUSH
2627 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2628 xf_clipboard_free(clipboard);
2629 WINPR_PRAGMA_DIAG_POP
2633 void xf_clipboard_free(xfClipboard* clipboard)
2638 xf_clipboard_free_server_formats(clipboard);
2640 if (clipboard->numClientFormats)
2642 for (UINT32 i = 0; i < clipboard->numClientFormats; i++)
2644 xfCliprdrFormat* format = &clipboard->clientFormats[i];
2645 free(format->formatName);
2649 cliprdr_file_context_free(clipboard->file);
2651 ClipboardDestroy(clipboard->system);
2652 xf_clipboard_formats_free(clipboard);
2653 HashTable_Free(clipboard->cachedRawData);
2654 HashTable_Free(clipboard->cachedData);
2655 requested_format_free(&clipboard->requestedFormat);
2656 free(clipboard->respond);
2657 free(clipboard->incr_data);
2661 void xf_cliprdr_init(xfContext* xfc, CliprdrClientContext* cliprdr)
2664 WINPR_ASSERT(cliprdr);
2666 xfc->cliprdr = cliprdr;
2667 xfc->clipboard->context = cliprdr;
2669 cliprdr->MonitorReady = xf_cliprdr_monitor_ready;
2670 cliprdr->ServerCapabilities = xf_cliprdr_server_capabilities;
2671 cliprdr->ServerFormatList = xf_cliprdr_server_format_list;
2672 cliprdr->ServerFormatListResponse = xf_cliprdr_server_format_list_response;
2673 cliprdr->ServerFormatDataRequest = xf_cliprdr_server_format_data_request;
2674 cliprdr->ServerFormatDataResponse = xf_cliprdr_server_format_data_response;
2676 cliprdr_file_context_init(xfc->clipboard->file, cliprdr);
2679 void xf_cliprdr_uninit(xfContext* xfc, CliprdrClientContext* cliprdr)
2682 WINPR_ASSERT(cliprdr);
2684 xfc->cliprdr = NULL;
2688 cliprdr_file_context_uninit(xfc->clipboard->file, cliprdr);
2689 xfc->clipboard->context = NULL;
FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
This struct contains function pointer to initialize/free objects.