FreeRDP
Loading...
Searching...
No Matches
wlf_cliprdr.c
1
21#include <freerdp/config.h>
22
23#include <stdlib.h>
24
25#include <winpr/crt.h>
26#include <winpr/image.h>
27#include <winpr/stream.h>
28#include <winpr/clipboard.h>
29
30#include <freerdp/log.h>
31#include <freerdp/client/cliprdr.h>
32#include <freerdp/channels/channels.h>
33#include <freerdp/channels/cliprdr.h>
34
35#include <freerdp/client/client_cliprdr_file.h>
36
37#include "wlf_cliprdr.h"
38
39#define TAG CLIENT_TAG("wayland.cliprdr")
40
41#define mime_text_plain "text/plain"
42// NOLINTNEXTLINE(bugprone-suspicious-missing-comma)
43static const char mime_text_utf8[] = mime_text_plain ";charset=utf-8";
44
45// NOLINTNEXTLINE(bugprone-suspicious-missing-comma)
46static const char* mime_text[] = { mime_text_plain, mime_text_utf8, "UTF8_STRING",
47 "COMPOUND_TEXT", "TEXT", "STRING" };
48
49static const char mime_png[] = "image/png";
50static const char mime_webp[] = "image/webp";
51static const char mime_jpg[] = "image/jpeg";
52static const char mime_tiff[] = "image/tiff";
53static const char mime_uri_list[] = "text/uri-list";
54static const char mime_html[] = "text/html";
55
56#define BMP_MIME_LIST "image/bmp", "image/x-bmp", "image/x-MS-bmp", "image/x-win-bitmap"
57static const char* mime_bitmap[] = { BMP_MIME_LIST };
58static const char* mime_image[] = { mime_png, mime_webp, mime_jpg, mime_tiff, BMP_MIME_LIST };
59
60static const char mime_gnome_copied_files[] = "x-special/gnome-copied-files";
61static const char mime_mate_copied_files[] = "x-special/mate-copied-files";
62
63static const char type_FileGroupDescriptorW[] = "FileGroupDescriptorW";
64static const char type_HtmlFormat[] = "HTML Format";
65
66typedef struct
67{
68 FILE* responseFile;
69 UINT32 responseFormat;
70 char* responseMime;
71} wlf_request;
72
73typedef struct
74{
75 const FILE* responseFile;
76 UINT32 responseFormat;
77 const char* responseMime;
78} wlf_const_request;
79
80struct wlf_clipboard
81{
82 wlfContext* wfc;
83 rdpChannels* channels;
84 CliprdrClientContext* context;
85 wLog* log;
86
87 UwacSeat* seat;
88 wClipboard* system;
89
90 size_t numClientFormats;
91 CLIPRDR_FORMAT* clientFormats;
92
93 size_t numServerFormats;
94 CLIPRDR_FORMAT* serverFormats;
95
96 BOOL sync;
97
99 CliprdrFileContext* file;
100
101 wQueue* request_queue;
102};
103
104static void wlf_request_free(void* rq)
105{
106 wlf_request* request = rq;
107 if (request)
108 {
109 free(request->responseMime);
110 if (request->responseFile)
111 (void)fclose(request->responseFile);
112 }
113 free(request);
114}
115
116static wlf_request* wlf_request_new(void)
117{
118 return calloc(1, sizeof(wlf_request));
119}
120
121static void* wlf_request_clone(const void* oth)
122{
123 const wlf_request* other = (const wlf_request*)oth;
124 wlf_request* copy = wlf_request_new();
125 if (!copy)
126 return nullptr;
127 *copy = *other;
128 if (other->responseMime)
129 {
130 copy->responseMime = _strdup(other->responseMime);
131 if (!copy->responseMime)
132 goto fail;
133 }
134 return copy;
135fail:
136 wlf_request_free(copy);
137 return nullptr;
138}
139
140static BOOL wlf_mime_is_file(const char* mime)
141{
142 if (strncmp(mime_uri_list, mime, sizeof(mime_uri_list)) == 0)
143 return TRUE;
144 if (strncmp(mime_gnome_copied_files, mime, sizeof(mime_gnome_copied_files)) == 0)
145 return TRUE;
146 if (strncmp(mime_mate_copied_files, mime, sizeof(mime_mate_copied_files)) == 0)
147 return TRUE;
148 return FALSE;
149}
150
151static BOOL wlf_mime_is_text(const char* mime)
152{
153 for (size_t x = 0; x < ARRAYSIZE(mime_text); x++)
154 {
155 if (strcmp(mime, mime_text[x]) == 0)
156 return TRUE;
157 }
158
159 return FALSE;
160}
161
162static BOOL wlf_mime_is_image(const char* mime)
163{
164 for (size_t x = 0; x < ARRAYSIZE(mime_image); x++)
165 {
166 if (strcmp(mime, mime_image[x]) == 0)
167 return TRUE;
168 }
169
170 return FALSE;
171}
172
173static BOOL wlf_mime_is_html(const char* mime)
174{
175 return strcmp(mime, mime_html) == 0;
176}
177
178static void wlf_cliprdr_free_server_formats(wfClipboard* clipboard)
179{
180 if (clipboard && clipboard->serverFormats)
181 {
182 for (size_t j = 0; j < clipboard->numServerFormats; j++)
183 {
184 CLIPRDR_FORMAT* format = &clipboard->serverFormats[j];
185 free(format->formatName);
186 }
187
188 free(clipboard->serverFormats);
189 clipboard->serverFormats = nullptr;
190 clipboard->numServerFormats = 0;
191 }
192
193 if (clipboard)
194 UwacClipboardOfferDestroy(clipboard->seat);
195}
196
197static void wlf_cliprdr_free_client_formats(wfClipboard* clipboard)
198{
199 if (clipboard && clipboard->numClientFormats)
200 {
201 for (size_t j = 0; j < clipboard->numClientFormats; j++)
202 {
203 CLIPRDR_FORMAT* format = &clipboard->clientFormats[j];
204 free(format->formatName);
205 }
206
207 free(clipboard->clientFormats);
208 clipboard->clientFormats = nullptr;
209 clipboard->numClientFormats = 0;
210 }
211
212 if (clipboard)
213 UwacClipboardOfferDestroy(clipboard->seat);
214}
215
221static UINT wlf_cliprdr_send_client_format_list(wfClipboard* clipboard)
222{
223 WINPR_ASSERT(clipboard);
224
225 const CLIPRDR_FORMAT_LIST formatList = { .common.msgFlags = 0,
226 .numFormats = (UINT32)clipboard->numClientFormats,
227 .formats = clipboard->clientFormats,
228 .common.msgType = CB_FORMAT_LIST };
229
230 if (!cliprdr_file_context_clear(clipboard->file))
231 return ERROR_INTERNAL_ERROR;
232
233 WLog_VRB(TAG, "-------------- client format list [%" PRIu32 "] ------------------",
234 formatList.numFormats);
235 for (UINT32 x = 0; x < formatList.numFormats; x++)
236 {
237 const CLIPRDR_FORMAT* format = &formatList.formats[x];
238 WLog_VRB(TAG, "client announces %" PRIu32 " [%s][%s]", format->formatId,
239 ClipboardGetFormatIdString(format->formatId), format->formatName);
240 }
241 WINPR_ASSERT(clipboard->context);
242 WINPR_ASSERT(clipboard->context->ClientFormatList);
243 return clipboard->context->ClientFormatList(clipboard->context, &formatList);
244}
245
246static void wfl_cliprdr_add_client_format_id(wfClipboard* clipboard, UINT32 formatId)
247{
248 CLIPRDR_FORMAT* format = nullptr;
249 const char* name = ClipboardGetFormatName(clipboard->system, formatId);
250
251 for (size_t x = 0; x < clipboard->numClientFormats; x++)
252 {
253 format = &clipboard->clientFormats[x];
254
255 if (format->formatId == formatId)
256 return;
257 }
258
259 format = realloc(clipboard->clientFormats,
260 (clipboard->numClientFormats + 1) * sizeof(CLIPRDR_FORMAT));
261
262 if (!format)
263 return;
264
265 clipboard->clientFormats = format;
266 format = &clipboard->clientFormats[clipboard->numClientFormats++];
267 format->formatId = formatId;
268 format->formatName = nullptr;
269
270 if (name && (formatId >= CF_MAX))
271 format->formatName = _strdup(name);
272}
273
274static BOOL wlf_cliprdr_add_client_format(wfClipboard* clipboard, const char* mime)
275{
276 WINPR_ASSERT(mime);
277 ClipboardLock(clipboard->system);
278 if (wlf_mime_is_html(mime))
279 {
280 UINT32 formatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
281 wfl_cliprdr_add_client_format_id(clipboard, formatId);
282 }
283 else if (wlf_mime_is_text(mime))
284 {
285 wfl_cliprdr_add_client_format_id(clipboard, CF_TEXT);
286 wfl_cliprdr_add_client_format_id(clipboard, CF_OEMTEXT);
287 wfl_cliprdr_add_client_format_id(clipboard, CF_UNICODETEXT);
288 }
289 else if (wlf_mime_is_image(mime))
290 {
291 for (size_t x = 0; x < ARRAYSIZE(mime_image); x++)
292 {
293 const char* mime_bmp = mime_image[x];
294 UINT32 formatId = ClipboardGetFormatId(clipboard->system, mime_bmp);
295 if (formatId != 0)
296 wfl_cliprdr_add_client_format_id(clipboard, formatId);
297 }
298 wfl_cliprdr_add_client_format_id(clipboard, CF_DIB);
299 wfl_cliprdr_add_client_format_id(clipboard, CF_TIFF);
300 }
301 else if (wlf_mime_is_file(mime))
302 {
303 const UINT32 fileFormatId =
304 ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
305 wfl_cliprdr_add_client_format_id(clipboard, fileFormatId);
306 }
307
308 ClipboardUnlock(clipboard->system);
309 return (wlf_cliprdr_send_client_format_list(clipboard) == CHANNEL_RC_OK);
310}
311
317static UINT wlf_cliprdr_send_data_request(wfClipboard* clipboard, const wlf_const_request* rq)
318{
319 WINPR_ASSERT(rq);
320
321 CLIPRDR_FORMAT_DATA_REQUEST request = { .requestedFormatId = rq->responseFormat };
322
323 if (!Queue_Enqueue(clipboard->request_queue, rq))
324 return ERROR_INTERNAL_ERROR;
325
326 WINPR_ASSERT(clipboard);
327 WINPR_ASSERT(clipboard->context);
328 WINPR_ASSERT(clipboard->context->ClientFormatDataRequest);
329 return clipboard->context->ClientFormatDataRequest(clipboard->context, &request);
330}
331
337static UINT wlf_cliprdr_send_data_response(wfClipboard* clipboard, const BYTE* data, size_t size)
338{
339 CLIPRDR_FORMAT_DATA_RESPONSE response = WINPR_C_ARRAY_INIT;
340
341 if (size > UINT32_MAX)
342 return ERROR_INVALID_PARAMETER;
343
344 response.common.msgFlags = (data) ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
345 response.common.dataLen = (UINT32)size;
346 response.requestedFormatData = data;
347
348 WINPR_ASSERT(clipboard);
349 WINPR_ASSERT(clipboard->context);
350 WINPR_ASSERT(clipboard->context->ClientFormatDataResponse);
351 return clipboard->context->ClientFormatDataResponse(clipboard->context, &response);
352}
353
354BOOL wlf_cliprdr_handle_event(wfClipboard* clipboard, const UwacClipboardEvent* event)
355{
356 if (!clipboard || !event)
357 return FALSE;
358
359 if (!clipboard->context)
360 return TRUE;
361
362 switch (event->type)
363 {
364 case UWAC_EVENT_CLIPBOARD_AVAILABLE:
365 clipboard->seat = event->seat;
366 return TRUE;
367
368 case UWAC_EVENT_CLIPBOARD_OFFER:
369 WLog_Print(clipboard->log, WLOG_DEBUG, "client announces mime %s", event->mime);
370 return wlf_cliprdr_add_client_format(clipboard, event->mime);
371
372 case UWAC_EVENT_CLIPBOARD_SELECT:
373 WLog_Print(clipboard->log, WLOG_DEBUG, "client announces new data");
374 wlf_cliprdr_free_client_formats(clipboard);
375 return TRUE;
376
377 default:
378 return FALSE;
379 }
380}
381
387static UINT wlf_cliprdr_send_client_capabilities(wfClipboard* clipboard)
388{
389 WINPR_ASSERT(clipboard);
390
391 CLIPRDR_GENERAL_CAPABILITY_SET generalCapabilitySet = {
392 .capabilitySetType = CB_CAPSTYPE_GENERAL,
393 .capabilitySetLength = 12,
394 .version = CB_CAPS_VERSION_2,
395 .generalFlags =
396 CB_USE_LONG_FORMAT_NAMES | cliprdr_file_context_current_flags(clipboard->file)
397 };
398 CLIPRDR_CAPABILITIES capabilities = { .cCapabilitiesSets = 1,
399 .capabilitySets =
400 (CLIPRDR_CAPABILITY_SET*)&(generalCapabilitySet) };
401
402 WINPR_ASSERT(clipboard);
403 WINPR_ASSERT(clipboard->context);
404 WINPR_ASSERT(clipboard->context->ClientCapabilities);
405 return clipboard->context->ClientCapabilities(clipboard->context, &capabilities);
406}
407
413static UINT wlf_cliprdr_send_client_format_list_response(wfClipboard* clipboard, BOOL status)
414{
415 const CLIPRDR_FORMAT_LIST_RESPONSE formatListResponse = {
416 .common.msgType = CB_FORMAT_LIST_RESPONSE,
417 .common.msgFlags = status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL,
418 .common.dataLen = 0
419 };
420 WINPR_ASSERT(clipboard);
421 WINPR_ASSERT(clipboard->context);
422 WINPR_ASSERT(clipboard->context->ClientFormatListResponse);
423 return clipboard->context->ClientFormatListResponse(clipboard->context, &formatListResponse);
424}
425
431static UINT wlf_cliprdr_monitor_ready(CliprdrClientContext* context,
432 const CLIPRDR_MONITOR_READY* monitorReady)
433{
434 UINT ret = 0;
435
436 WINPR_UNUSED(monitorReady);
437 WINPR_ASSERT(context);
438 WINPR_ASSERT(monitorReady);
439
440 wfClipboard* clipboard = cliprdr_file_context_get_context(context->custom);
441 WINPR_ASSERT(clipboard);
442
443 if ((ret = wlf_cliprdr_send_client_capabilities(clipboard)) != CHANNEL_RC_OK)
444 return ret;
445
446 if ((ret = wlf_cliprdr_send_client_format_list(clipboard)) != CHANNEL_RC_OK)
447 return ret;
448
449 clipboard->sync = TRUE;
450 return CHANNEL_RC_OK;
451}
452
458static UINT wlf_cliprdr_server_capabilities(CliprdrClientContext* context,
459 const CLIPRDR_CAPABILITIES* capabilities)
460{
461 WINPR_ASSERT(context);
462 WINPR_ASSERT(capabilities);
463
464 const BYTE* capsPtr = (const BYTE*)capabilities->capabilitySets;
465 WINPR_ASSERT(capsPtr);
466
467 wfClipboard* clipboard = cliprdr_file_context_get_context(context->custom);
468 WINPR_ASSERT(clipboard);
469
470 if (!cliprdr_file_context_remote_set_flags(clipboard->file, 0))
471 return ERROR_INTERNAL_ERROR;
472
473 for (UINT32 i = 0; i < capabilities->cCapabilitiesSets; i++)
474 {
475 const CLIPRDR_CAPABILITY_SET* caps = (const CLIPRDR_CAPABILITY_SET*)capsPtr;
476
477 if (caps->capabilitySetType == CB_CAPSTYPE_GENERAL)
478 {
479 const CLIPRDR_GENERAL_CAPABILITY_SET* generalCaps =
481
482 if (!cliprdr_file_context_remote_set_flags(clipboard->file, generalCaps->generalFlags))
483 return ERROR_INTERNAL_ERROR;
484 }
485
486 capsPtr += caps->capabilitySetLength;
487 }
488
489 return CHANNEL_RC_OK;
490}
491
492static UINT32 wlf_get_server_format_id(const wfClipboard* clipboard, const char* name)
493{
494 WINPR_ASSERT(clipboard);
495 WINPR_ASSERT(name);
496
497 for (UINT32 x = 0; x < clipboard->numServerFormats; x++)
498 {
499 const CLIPRDR_FORMAT* format = &clipboard->serverFormats[x];
500 if (!format->formatName)
501 continue;
502 if (strcmp(name, format->formatName) == 0)
503 return format->formatId;
504 }
505 return 0;
506}
507
508static const char* wlf_get_server_format_name(const wfClipboard* clipboard, UINT32 formatId)
509{
510 WINPR_ASSERT(clipboard);
511
512 for (UINT32 x = 0; x < clipboard->numServerFormats; x++)
513 {
514 const CLIPRDR_FORMAT* format = &clipboard->serverFormats[x];
515 if (format->formatId == formatId)
516 return format->formatName;
517 }
518 return nullptr;
519}
520
521static void wlf_cliprdr_transfer_data(UwacSeat* seat, void* context, const char* mime, int fd)
522{
523 wfClipboard* clipboard = (wfClipboard*)context;
524 WINPR_UNUSED(seat);
525
526 EnterCriticalSection(&clipboard->lock);
527
528 wlf_const_request request = WINPR_C_ARRAY_INIT;
529 if (wlf_mime_is_html(mime))
530 {
531 request.responseMime = mime_html;
532 request.responseFormat = wlf_get_server_format_id(clipboard, type_HtmlFormat);
533 }
534 else if (wlf_mime_is_file(mime))
535 {
536 request.responseMime = mime;
537 request.responseFormat = wlf_get_server_format_id(clipboard, type_FileGroupDescriptorW);
538 }
539 else if (wlf_mime_is_text(mime))
540 {
541 request.responseMime = mime_text_plain;
542 request.responseFormat = CF_UNICODETEXT;
543 }
544 else if (wlf_mime_is_image(mime))
545 {
546 request.responseMime = mime;
547 if (strcmp(mime, mime_tiff) == 0)
548 request.responseFormat = CF_TIFF;
549 else
550 request.responseFormat = CF_DIB;
551 }
552
553 if (request.responseMime != nullptr)
554 {
555 request.responseFile = fdopen(fd, "w");
556
557 if (request.responseFile)
558 wlf_cliprdr_send_data_request(clipboard, &request);
559 else
560 WLog_Print(clipboard->log, WLOG_ERROR,
561 "failed to open clipboard file descriptor for MIME %s",
562 request.responseMime);
563 }
564
565 LeaveCriticalSection(&clipboard->lock);
566}
567
568static void wlf_cliprdr_cancel_data(UwacSeat* seat, void* context)
569{
570 wfClipboard* clipboard = (wfClipboard*)context;
571
572 WINPR_UNUSED(seat);
573 WINPR_ASSERT(clipboard);
574 cliprdr_file_context_clear(clipboard->file);
575}
576
585static UINT wlf_cliprdr_server_format_list(CliprdrClientContext* context,
586 const CLIPRDR_FORMAT_LIST* formatList)
587{
588 BOOL html = FALSE;
589 BOOL text = FALSE;
590 BOOL image = FALSE;
591 BOOL file = FALSE;
592
593 if (!context || !context->custom)
594 return ERROR_INVALID_PARAMETER;
595
596 wfClipboard* clipboard = cliprdr_file_context_get_context(context->custom);
597 WINPR_ASSERT(clipboard);
598
599 wlf_cliprdr_free_server_formats(clipboard);
600 if (!cliprdr_file_context_clear(clipboard->file))
601 return ERROR_INTERNAL_ERROR;
602
603 if (!(clipboard->serverFormats =
604 (CLIPRDR_FORMAT*)calloc(formatList->numFormats, sizeof(CLIPRDR_FORMAT))))
605 {
606 WLog_Print(clipboard->log, WLOG_ERROR,
607 "failed to allocate %" PRIuz " CLIPRDR_FORMAT structs",
608 clipboard->numServerFormats);
609 return CHANNEL_RC_NO_MEMORY;
610 }
611
612 clipboard->numServerFormats = formatList->numFormats;
613
614 if (!clipboard->seat)
615 {
616 WLog_Print(clipboard->log, WLOG_ERROR,
617 "clipboard->seat=nullptr, check your client implementation");
618 return ERROR_INTERNAL_ERROR;
619 }
620
621 for (UINT32 i = 0; i < formatList->numFormats; i++)
622 {
623 const CLIPRDR_FORMAT* format = &formatList->formats[i];
624 CLIPRDR_FORMAT* srvFormat = &clipboard->serverFormats[i];
625 srvFormat->formatId = format->formatId;
626
627 if (format->formatName)
628 {
629 srvFormat->formatName = _strdup(format->formatName);
630
631 if (!srvFormat->formatName)
632 {
633 wlf_cliprdr_free_server_formats(clipboard);
634 return CHANNEL_RC_NO_MEMORY;
635 }
636 }
637
638 if (format->formatName)
639 {
640 if (strcmp(format->formatName, type_HtmlFormat) == 0)
641 {
642 text = TRUE;
643 html = TRUE;
644 }
645 else if (strcmp(format->formatName, type_FileGroupDescriptorW) == 0)
646 {
647 file = TRUE;
648 text = TRUE;
649 }
650 }
651 else
652 {
653 switch (format->formatId)
654 {
655 case CF_TEXT:
656 case CF_OEMTEXT:
657 case CF_UNICODETEXT:
658 text = TRUE;
659 break;
660
661 case CF_DIB:
662 image = TRUE;
663 break;
664
665 default:
666 break;
667 }
668 }
669 }
670
671 if (html)
672 {
673 UwacClipboardOfferCreate(clipboard->seat, mime_html);
674 }
675
676 if (file && cliprdr_file_context_has_local_support(clipboard->file))
677 {
678 UwacClipboardOfferCreate(clipboard->seat, mime_uri_list);
679 UwacClipboardOfferCreate(clipboard->seat, mime_gnome_copied_files);
680 UwacClipboardOfferCreate(clipboard->seat, mime_mate_copied_files);
681 }
682
683 if (text)
684 {
685 for (size_t x = 0; x < ARRAYSIZE(mime_text); x++)
686 UwacClipboardOfferCreate(clipboard->seat, mime_text[x]);
687 }
688
689 if (image)
690 {
691 for (size_t x = 0; x < ARRAYSIZE(mime_image); x++)
692 UwacClipboardOfferCreate(clipboard->seat, mime_image[x]);
693 }
694
695 UwacClipboardOfferAnnounce(clipboard->seat, clipboard, wlf_cliprdr_transfer_data,
696 wlf_cliprdr_cancel_data);
697 return wlf_cliprdr_send_client_format_list_response(clipboard, TRUE);
698}
699
705static UINT
706wlf_cliprdr_server_format_list_response(WINPR_ATTR_UNUSED CliprdrClientContext* context,
707 const CLIPRDR_FORMAT_LIST_RESPONSE* formatListResponse)
708{
709 WINPR_ASSERT(context);
710 WINPR_ASSERT(formatListResponse);
711
712 if (formatListResponse->common.msgFlags & CB_RESPONSE_FAIL)
713 WLog_WARN(TAG, "format list update failed");
714 return CHANNEL_RC_OK;
715}
716
722static UINT
723wlf_cliprdr_server_format_data_request(CliprdrClientContext* context,
724 const CLIPRDR_FORMAT_DATA_REQUEST* formatDataRequest)
725{
726 UINT rc = CHANNEL_RC_OK;
727 char* data = nullptr;
728 size_t size = 0;
729 const char* mime = nullptr;
730 UINT32 formatId = 0;
731 UINT32 localFormatId = 0;
732 wfClipboard* clipboard = nullptr;
733
734 UINT32 dsize = 0;
735 BYTE* ddata = nullptr;
736
737 WINPR_ASSERT(context);
738 WINPR_ASSERT(formatDataRequest);
739
740 localFormatId = formatId = formatDataRequest->requestedFormatId;
741 clipboard = cliprdr_file_context_get_context(context->custom);
742 WINPR_ASSERT(clipboard);
743
744 ClipboardLock(clipboard->system);
745 const UINT32 fileFormatId = ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
746 const UINT32 htmlFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
747
748 switch (formatId)
749 {
750 case CF_TEXT:
751 case CF_OEMTEXT:
752 case CF_UNICODETEXT:
753 localFormatId = ClipboardGetFormatId(clipboard->system, mime_text_plain);
754 mime = mime_text_utf8;
755 break;
756
757 case CF_DIB:
758 case CF_DIBV5:
759 mime = mime_bitmap[0];
760 break;
761
762 case CF_TIFF:
763 mime = mime_tiff;
764 break;
765
766 default:
767 if (formatId == fileFormatId)
768 {
769 localFormatId = ClipboardGetFormatId(clipboard->system, mime_uri_list);
770 mime = mime_uri_list;
771 }
772 else if (formatId == htmlFormatId)
773 {
774 localFormatId = ClipboardGetFormatId(clipboard->system, mime_html);
775 mime = mime_html;
776 }
777 else
778 goto fail;
779 break;
780 }
781
782 data = UwacClipboardDataGet(clipboard->seat, mime, &size);
783
784 if (!data || (size > UINT32_MAX))
785 goto fail;
786
787 if (fileFormatId == formatId)
788 {
789 if (!cliprdr_file_context_update_client_data(clipboard->file, data, size))
790 goto fail;
791 }
792
793 {
794 const BOOL res = ClipboardSetData(clipboard->system, localFormatId, data, (UINT32)size);
795 free(data);
796
797 UINT32 len = 0;
798 data = nullptr;
799 if (res)
800 data = ClipboardGetData(clipboard->system, formatId, &len);
801
802 if (!res || !data)
803 goto fail;
804
805 if (fileFormatId == formatId)
806 {
807 const UINT32 flags = cliprdr_file_context_remote_get_flags(clipboard->file);
808 const UINT32 error = cliprdr_serialize_file_list_ex(
809 flags, (const FILEDESCRIPTORW*)data, len / sizeof(FILEDESCRIPTORW), &ddata, &dsize);
810 if (error)
811 goto fail;
812 }
813 }
814fail:
815 ClipboardUnlock(clipboard->system);
816 rc = wlf_cliprdr_send_data_response(clipboard, ddata, dsize);
817 free(data);
818 return rc;
819}
820
826static UINT
827wlf_cliprdr_server_format_data_response(CliprdrClientContext* context,
828 const CLIPRDR_FORMAT_DATA_RESPONSE* formatDataResponse)
829{
830 UINT rc = ERROR_INTERNAL_ERROR;
831
832 WINPR_ASSERT(context);
833 WINPR_ASSERT(formatDataResponse);
834
835 const UINT32 size = formatDataResponse->common.dataLen;
836 const BYTE* data = formatDataResponse->requestedFormatData;
837
838 wfClipboard* clipboard = cliprdr_file_context_get_context(context->custom);
839 WINPR_ASSERT(clipboard);
840
841 wlf_request* request = Queue_Dequeue(clipboard->request_queue);
842 if (!request)
843 goto fail;
844
845 rc = CHANNEL_RC_OK;
846 if (formatDataResponse->common.msgFlags & CB_RESPONSE_FAIL)
847 {
848 WLog_WARN(TAG, "clipboard data request for format %" PRIu32 " [%s], mime %s failed",
849 request->responseFormat, ClipboardGetFormatIdString(request->responseFormat),
850 request->responseMime);
851 goto fail;
852 }
853 rc = ERROR_INTERNAL_ERROR;
854
855 ClipboardLock(clipboard->system);
856 EnterCriticalSection(&clipboard->lock);
857 {
858 BYTE* cdata = nullptr;
859 UINT32 srcFormatId = 0;
860 UINT32 dstFormatId = 0;
861 switch (request->responseFormat)
862 {
863 case CF_TEXT:
864 case CF_OEMTEXT:
865 case CF_UNICODETEXT:
866 srcFormatId = request->responseFormat;
867 dstFormatId = ClipboardGetFormatId(clipboard->system, request->responseMime);
868 break;
869
870 case CF_DIB:
871 case CF_DIBV5:
872 srcFormatId = request->responseFormat;
873 dstFormatId = ClipboardGetFormatId(clipboard->system, request->responseMime);
874 break;
875
876 default:
877 {
878 const char* name = wlf_get_server_format_name(clipboard, request->responseFormat);
879 if (name)
880 {
881 if (strcmp(type_FileGroupDescriptorW, name) == 0)
882 {
883 srcFormatId =
884 ClipboardGetFormatId(clipboard->system, type_FileGroupDescriptorW);
885 dstFormatId =
886 ClipboardGetFormatId(clipboard->system, request->responseMime);
887
888 if (!cliprdr_file_context_update_server_data(clipboard->file,
889 clipboard->system, data, size))
890 goto unlock;
891 }
892 else if (strcmp(type_HtmlFormat, name) == 0)
893 {
894 srcFormatId = ClipboardGetFormatId(clipboard->system, type_HtmlFormat);
895 dstFormatId =
896 ClipboardGetFormatId(clipboard->system, request->responseMime);
897 }
898 }
899 }
900 break;
901 }
902 {
903 UINT32 len = 0;
904
905 {
906 const BOOL sres = ClipboardSetData(clipboard->system, srcFormatId, data, size);
907 if (sres)
908 cdata = ClipboardGetData(clipboard->system, dstFormatId, &len);
909
910 if (!sres || !cdata)
911 goto unlock;
912 }
913
914 if (request->responseFile)
915 {
916 const size_t res = fwrite(cdata, 1, len, request->responseFile);
917 if (res == len)
918 rc = CHANNEL_RC_OK;
919 }
920 else
921 rc = CHANNEL_RC_OK;
922 }
923
924 unlock:
925 free(cdata);
926 }
927 ClipboardUnlock(clipboard->system);
928 LeaveCriticalSection(&clipboard->lock);
929fail:
930 wlf_request_free(request);
931 return rc;
932}
933
934wfClipboard* wlf_clipboard_new(wlfContext* wfc)
935{
936 rdpChannels* channels = nullptr;
937 wfClipboard* clipboard = nullptr;
938
939 WINPR_ASSERT(wfc);
940
941 clipboard = (wfClipboard*)calloc(1, sizeof(wfClipboard));
942
943 if (!clipboard)
944 goto fail;
945
946 InitializeCriticalSection(&clipboard->lock);
947 clipboard->wfc = wfc;
948 channels = wfc->common.context.channels;
949 clipboard->log = WLog_Get(TAG);
950 clipboard->channels = channels;
951 clipboard->system = ClipboardCreate();
952 if (!clipboard->system)
953 goto fail;
954
955 clipboard->file = cliprdr_file_context_new(clipboard);
956 if (!clipboard->file)
957 goto fail;
958
959 if (!cliprdr_file_context_set_locally_available(clipboard->file, TRUE))
960 goto fail;
961
962 clipboard->request_queue = Queue_New(TRUE, -1, -1);
963 if (!clipboard->request_queue)
964 goto fail;
965
966 {
967 wObject* obj = Queue_Object(clipboard->request_queue);
968 WINPR_ASSERT(obj);
969 obj->fnObjectFree = wlf_request_free;
970 obj->fnObjectNew = wlf_request_clone;
971 }
972
973 return clipboard;
974
975fail:
976 wlf_clipboard_free(clipboard);
977 return nullptr;
978}
979
980void wlf_clipboard_free(wfClipboard* clipboard)
981{
982 if (!clipboard)
983 return;
984
985 cliprdr_file_context_free(clipboard->file);
986
987 wlf_cliprdr_free_server_formats(clipboard);
988 wlf_cliprdr_free_client_formats(clipboard);
989 ClipboardDestroy(clipboard->system);
990
991 EnterCriticalSection(&clipboard->lock);
992
993 Queue_Free(clipboard->request_queue);
994 LeaveCriticalSection(&clipboard->lock);
995 DeleteCriticalSection(&clipboard->lock);
996 free(clipboard);
997}
998
999BOOL wlf_cliprdr_init(wfClipboard* clipboard, CliprdrClientContext* cliprdr)
1000{
1001 WINPR_ASSERT(clipboard);
1002 WINPR_ASSERT(cliprdr);
1003
1004 clipboard->context = cliprdr;
1005 cliprdr->MonitorReady = wlf_cliprdr_monitor_ready;
1006 cliprdr->ServerCapabilities = wlf_cliprdr_server_capabilities;
1007 cliprdr->ServerFormatList = wlf_cliprdr_server_format_list;
1008 cliprdr->ServerFormatListResponse = wlf_cliprdr_server_format_list_response;
1009 cliprdr->ServerFormatDataRequest = wlf_cliprdr_server_format_data_request;
1010 cliprdr->ServerFormatDataResponse = wlf_cliprdr_server_format_data_response;
1011
1012 return cliprdr_file_context_init(clipboard->file, cliprdr);
1013}
1014
1015BOOL wlf_cliprdr_uninit(wfClipboard* clipboard, CliprdrClientContext* cliprdr)
1016{
1017 WINPR_ASSERT(clipboard);
1018 if (!cliprdr_file_context_uninit(clipboard->file, cliprdr))
1019 return FALSE;
1020
1021 if (cliprdr)
1022 cliprdr->custom = nullptr;
1023
1024 return TRUE;
1025}
This struct contains function pointer to initialize/free objects.
Definition collections.h:52
OBJECT_FREE_FN fnObjectFree
Definition collections.h:58
OBJECT_NEW_FN fnObjectNew
Definition collections.h:53