VirtualBox

source: vbox/trunk/src/VBox/GuestHost/SharedClipboard/x11-clipboard.cpp@ 62538

Last change on this file since 62538 was 62492, checked in by vboxsync, 8 years ago

(C) 2016

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 97.8 KB
Line 
1/** @file
2 *
3 * Shared Clipboard:
4 * X11 backend code.
5 */
6
7/*
8 * Copyright (C) 2006-2016 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19/* Note: to automatically run regression tests on the shared clipboard,
20 * execute the tstClipboardX11 testcase. If you often make changes to the
21 * clipboard code, adding the line
22 * OTHERS += $(PATH_tstClipboardX11)/tstClipboardX11.run
23 * to LocalConfig.kmk will cause the tests to be run every time the code is
24 * changed. */
25
26#define LOG_GROUP LOG_GROUP_SHARED_CLIPBOARD
27
28#include <errno.h>
29
30#include <dlfcn.h>
31#include <fcntl.h>
32#include <unistd.h>
33
34#ifdef RT_OS_SOLARIS
35#include <tsol/label.h>
36#endif
37
38#include <X11/Xlib.h>
39#include <X11/Xatom.h>
40#include <X11/Intrinsic.h>
41#include <X11/Shell.h>
42#include <X11/Xproto.h>
43#include <X11/StringDefs.h>
44
45#include <iprt/types.h>
46#include <iprt/mem.h>
47#include <iprt/semaphore.h>
48#include <iprt/thread.h>
49
50#include <VBox/log.h>
51
52#include <VBox/GuestHost/SharedClipboard.h>
53#include <VBox/GuestHost/clipboard-helper.h>
54#include <VBox/HostServices/VBoxClipboardSvc.h>
55
56class formats;
57static Atom clipGetAtom(CLIPBACKEND *pCtx, const char *pszName);
58
59/** The different clipboard formats which we support. */
60enum CLIPFORMAT
61{
62 INVALID = 0,
63 TARGETS,
64 TEXT, /* Treat this as Utf8, but it may really be ascii */
65 UTF8,
66 BMP,
67 HTML
68};
69
70/** The table mapping X11 names to data formats and to the corresponding
71 * VBox clipboard formats (currently only Unicode) */
72static struct _CLIPFORMATTABLE
73{
74 /** The X11 atom name of the format (several names can match one format)
75 */
76 const char *pcszAtom;
77 /** The format corresponding to the name */
78 CLIPFORMAT enmFormat;
79 /** The corresponding VBox clipboard format */
80 uint32_t u32VBoxFormat;
81} g_aFormats[] =
82{
83 { "INVALID", INVALID, 0 },
84 { "UTF8_STRING", UTF8, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
85 { "text/plain;charset=UTF-8", UTF8,
86 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
87 { "text/plain;charset=utf-8", UTF8,
88 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
89 { "STRING", TEXT, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
90 { "TEXT", TEXT, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
91 { "text/plain", TEXT, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT },
92 { "text/html", HTML, VBOX_SHARED_CLIPBOARD_FMT_HTML },
93 { "text/html;charset=utf-8", HTML,
94 VBOX_SHARED_CLIPBOARD_FMT_HTML },
95 { "image/bmp", BMP, VBOX_SHARED_CLIPBOARD_FMT_BITMAP },
96 { "image/x-bmp", BMP, VBOX_SHARED_CLIPBOARD_FMT_BITMAP },
97 { "image/x-MS-bmp", BMP, VBOX_SHARED_CLIPBOARD_FMT_BITMAP }
98
99
100 /* TODO: Inkscape exports image/png but not bmp... */
101};
102
103typedef unsigned CLIPX11FORMAT;
104
105enum
106{
107 NIL_CLIPX11FORMAT = 0,
108 MAX_CLIP_X11_FORMATS = RT_ELEMENTS(g_aFormats)
109};
110
111/** Return the atom corresponding to a supported X11 format.
112 * @param widget a valid Xt widget
113 */
114static Atom clipAtomForX11Format(CLIPBACKEND *pCtx, CLIPX11FORMAT format)
115{
116 return clipGetAtom(pCtx, g_aFormats[format].pcszAtom);
117}
118
119/** Return the CLIPFORMAT corresponding to a supported X11 format. */
120static CLIPFORMAT clipRealFormatForX11Format(CLIPX11FORMAT format)
121{
122 return g_aFormats[format].enmFormat;
123}
124
125/** Return the atom corresponding to a supported X11 format. */
126static uint32_t clipVBoxFormatForX11Format(CLIPX11FORMAT format)
127{
128 return g_aFormats[format].u32VBoxFormat;
129}
130
131/** Lookup the X11 format matching a given X11 atom.
132 * @returns the format on success, NIL_CLIPX11FORMAT on failure
133 * @param widget a valid Xt widget
134 */
135static CLIPX11FORMAT clipFindX11FormatByAtom(CLIPBACKEND *pCtx, Atom atomFormat)
136{
137 for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i)
138 if (clipAtomForX11Format(pCtx, i) == atomFormat)
139 return i;
140 return NIL_CLIPX11FORMAT;
141}
142
143/** Lookup the X11 format matching a given X11 atom text.
144 * @returns the format on success, NIL_CLIPX11FORMAT on failure
145 * @param widget a valid Xt widget
146 */
147static CLIPX11FORMAT clipFindX11FormatByAtomText(const char *pcsz)
148{
149 for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i)
150 if (!strcmp(g_aFormats[i].pcszAtom, pcsz))
151 return i;
152 return NIL_CLIPX11FORMAT;
153}
154
155/**
156 * Enumerates supported X11 clipboard formats corresponding to a given VBox
157 * format.
158 * @returns the next matching X11 format in the list, or NIL_CLIPX11FORMAT if
159 * there are no more
160 * @param lastFormat The value returned from the last call of this function.
161 * Use NIL_CLIPX11FORMAT to start the enumeration.
162 */
163static CLIPX11FORMAT clipEnumX11Formats(uint32_t u32VBoxFormats,
164 CLIPX11FORMAT lastFormat)
165{
166 for (unsigned i = lastFormat + 1; i < RT_ELEMENTS(g_aFormats); ++i)
167 if (u32VBoxFormats & clipVBoxFormatForX11Format(i))
168 return i;
169 return NIL_CLIPX11FORMAT;
170}
171
172/** Global context information used by the X11 clipboard backend */
173struct _CLIPBACKEND
174{
175 /** Opaque data structure describing the front-end. */
176 VBOXCLIPBOARDCONTEXT *pFrontend;
177 /** Is an X server actually available? */
178 bool fHaveX11;
179 /** The X Toolkit application context structure */
180 XtAppContext appContext;
181
182 /** We have a separate thread to wait for Window and Clipboard events */
183 RTTHREAD thread;
184 /** The X Toolkit widget which we use as our clipboard client. It is never made visible. */
185 Widget widget;
186
187 /** Should we try to grab the clipboard on startup? */
188 bool fGrabClipboardOnStart;
189
190 /** The best text format X11 has to offer, as an index into the formats
191 * table */
192 CLIPX11FORMAT X11TextFormat;
193 /** The best bitmap format X11 has to offer, as an index into the formats
194 * table */
195 CLIPX11FORMAT X11BitmapFormat;
196 /** The best HTML format X11 has to offer, as an index into the formats
197 * table */
198 CLIPX11FORMAT X11HTMLFormat;
199 /** What formats does VBox have on offer? */
200 uint32_t vboxFormats;
201 /** Cache of the last unicode data that we received */
202 void *pvUnicodeCache;
203 /** Size of the unicode data in the cache */
204 uint32_t cbUnicodeCache;
205 /** When we wish the clipboard to exit, we have to wake up the event
206 * loop. We do this by writing into a pipe. This end of the pipe is
207 * the end that another thread can write to. */
208 int wakeupPipeWrite;
209 /** The reader end of the pipe */
210 int wakeupPipeRead;
211 /** A pointer to the XFixesSelectSelectionInput function */
212 void (*fixesSelectInput)(Display *, Window, Atom, unsigned long);
213 /** The first XFixes event number */
214 int fixesEventBase;
215 /** The Xt Intrinsics can only handle one outstanding clipboard operation
216 * at a time, so we keep track of whether one is in process. */
217 bool fBusy;
218 /** We can't handle a clipboard update event while we are busy, so remember
219 * it for later. */
220 bool fUpdateNeeded;
221};
222
223/** The number of simultaneous instances we support. For all normal purposes
224 * we should never need more than one. For the testcase it is convenient to
225 * have a second instance that the first can interact with in order to have
226 * a more controlled environment. */
227enum { CLIP_MAX_CONTEXTS = 20 };
228
229/** Array of structures for mapping Xt widgets to context pointers. We
230 * need this because the widget clipboard callbacks do not pass user data. */
231static struct {
232 /** The widget we want to associate the context with */
233 Widget widget;
234 /** The context associated with the widget */
235 CLIPBACKEND *pCtx;
236} g_contexts[CLIP_MAX_CONTEXTS];
237
238/** Register a new X11 clipboard context. */
239static int clipRegisterContext(CLIPBACKEND *pCtx)
240{
241 bool found = false;
242 AssertReturn(pCtx != NULL, VERR_INVALID_PARAMETER);
243 Widget widget = pCtx->widget;
244 AssertReturn(widget != NULL, VERR_INVALID_PARAMETER);
245 for (unsigned i = 0; i < RT_ELEMENTS(g_contexts); ++i)
246 {
247 AssertReturn( (g_contexts[i].widget != widget)
248 && (g_contexts[i].pCtx != pCtx), VERR_WRONG_ORDER);
249 if (g_contexts[i].widget == NULL && !found)
250 {
251 AssertReturn(g_contexts[i].pCtx == NULL, VERR_INTERNAL_ERROR);
252 g_contexts[i].widget = widget;
253 g_contexts[i].pCtx = pCtx;
254 found = true;
255 }
256 }
257 return found ? VINF_SUCCESS : VERR_OUT_OF_RESOURCES;
258}
259
260/** Unregister an X11 clipboard context. */
261static void clipUnregisterContext(CLIPBACKEND *pCtx)
262{
263 bool found = false;
264 AssertReturnVoid(pCtx != NULL);
265 Widget widget = pCtx->widget;
266 AssertReturnVoid(widget != NULL);
267 for (unsigned i = 0; i < RT_ELEMENTS(g_contexts); ++i)
268 {
269 Assert(!found || g_contexts[i].widget != widget);
270 if (g_contexts[i].widget == widget)
271 {
272 Assert(g_contexts[i].pCtx != NULL);
273 g_contexts[i].widget = NULL;
274 g_contexts[i].pCtx = NULL;
275 found = true;
276 }
277 }
278}
279
280/** Find an X11 clipboard context. */
281static CLIPBACKEND *clipLookupContext(Widget widget)
282{
283 AssertReturn(widget != NULL, NULL);
284 for (unsigned i = 0; i < RT_ELEMENTS(g_contexts); ++i)
285 {
286 if (g_contexts[i].widget == widget)
287 {
288 Assert(g_contexts[i].pCtx != NULL);
289 return g_contexts[i].pCtx;
290 }
291 }
292 return NULL;
293}
294
295/** Convert an atom name string to an X11 atom, looking it up in a cache
296 * before asking the server */
297static Atom clipGetAtom(CLIPBACKEND *pCtx, const char *pszName)
298{
299 AssertPtrReturn(pszName, None);
300 return XInternAtom(XtDisplay(pCtx->widget), pszName, False);
301}
302
303#ifdef TESTCASE
304static void testQueueToEventThread(void (*proc)(void *, void *),
305 void *client_data);
306#endif
307
308/** String written to the wakeup pipe. */
309#define WAKE_UP_STRING "WakeUp!"
310/** Length of the string written. */
311#define WAKE_UP_STRING_LEN ( sizeof(WAKE_UP_STRING) - 1 )
312
313/** Schedule a function call to run on the Xt event thread by passing it to
314 * the application context as a 0ms timeout and waking up the event loop by
315 * writing to the wakeup pipe which it monitors. */
316void clipQueueToEventThread(CLIPBACKEND *pCtx,
317 void (*proc)(void *, void *),
318 void *client_data)
319{
320 LogRel2(("clipQueueToEventThread: proc=%p, client_data=%p\n",
321 proc, client_data));
322#ifndef TESTCASE
323 XtAppAddTimeOut(pCtx->appContext, 0, (XtTimerCallbackProc)proc,
324 (XtPointer)client_data);
325 ssize_t cbWritten = write(pCtx->wakeupPipeWrite, WAKE_UP_STRING, WAKE_UP_STRING_LEN);
326 NOREF(cbWritten);
327#else
328 testQueueToEventThread(proc, client_data);
329#endif
330}
331
332/**
333 * Report the formats currently supported by the X11 clipboard to VBox.
334 */
335static void clipReportFormatsToVBox(CLIPBACKEND *pCtx)
336{
337 uint32_t u32VBoxFormats = clipVBoxFormatForX11Format(pCtx->X11TextFormat);
338 u32VBoxFormats |= clipVBoxFormatForX11Format(pCtx->X11BitmapFormat);
339 u32VBoxFormats |= clipVBoxFormatForX11Format(pCtx->X11HTMLFormat);
340 LogRelFlowFunc(("clipReportFormatsToVBox format: %d\n", u32VBoxFormats));
341 LogRelFlowFunc(("clipReportFormatsToVBox txt: %d, bitm: %d, html:%d, u32VBoxFormats: %d\n",
342 pCtx->X11TextFormat, pCtx->X11BitmapFormat, pCtx->X11HTMLFormat,
343 u32VBoxFormats ));
344 ClipReportX11Formats(pCtx->pFrontend, u32VBoxFormats);
345}
346
347/**
348 * Forget which formats were previously in the X11 clipboard. Called when we
349 * grab the clipboard. */
350static void clipResetX11Formats(CLIPBACKEND *pCtx)
351{
352 pCtx->X11TextFormat = INVALID;
353 pCtx->X11BitmapFormat = INVALID;
354 pCtx->X11HTMLFormat = INVALID;
355}
356
357/** Tell VBox that X11 currently has nothing in its clipboard. */
358static void clipReportEmptyX11CB(CLIPBACKEND *pCtx)
359{
360 clipResetX11Formats(pCtx);
361 clipReportFormatsToVBox(pCtx);
362}
363
364/**
365 * Go through an array of X11 clipboard targets to see if they contain a text
366 * format we can support, and if so choose the ones we prefer (e.g. we like
367 * Utf8 better than plain text).
368 * @param pCtx the clipboard backend context structure
369 * @param pTargets the list of targets
370 * @param cTargets the size of the list in @a pTargets
371 */
372static CLIPX11FORMAT clipGetTextFormatFromTargets(CLIPBACKEND *pCtx,
373 CLIPX11FORMAT *pTargets,
374 size_t cTargets)
375{
376 CLIPX11FORMAT bestTextFormat = NIL_CLIPX11FORMAT;
377 CLIPFORMAT enmBestTextTarget = INVALID;
378 AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT);
379 AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT);
380 for (unsigned i = 0; i < cTargets; ++i)
381 {
382 CLIPX11FORMAT format = pTargets[i];
383 if (format != NIL_CLIPX11FORMAT)
384 {
385 if ( (clipVBoxFormatForX11Format(format)
386 == VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
387 && enmBestTextTarget < clipRealFormatForX11Format(format))
388 {
389 enmBestTextTarget = clipRealFormatForX11Format(format);
390 bestTextFormat = format;
391 }
392 }
393 }
394 return bestTextFormat;
395}
396
397#ifdef TESTCASE
398static bool clipTestTextFormatConversion(CLIPBACKEND *pCtx)
399{
400 bool success = true;
401 CLIPX11FORMAT targets[2];
402 CLIPX11FORMAT x11Format;
403 targets[0] = clipFindX11FormatByAtomText("text/plain");
404 targets[1] = clipFindX11FormatByAtomText("image/bmp");
405 x11Format = clipGetTextFormatFromTargets(pCtx, targets, 2);
406 if (clipRealFormatForX11Format(x11Format) != TEXT)
407 success = false;
408 targets[0] = clipFindX11FormatByAtomText("UTF8_STRING");
409 targets[1] = clipFindX11FormatByAtomText("text/plain");
410 x11Format = clipGetTextFormatFromTargets(pCtx, targets, 2);
411 if (clipRealFormatForX11Format(x11Format) != UTF8)
412 success = false;
413 return success;
414}
415#endif
416
417/**
418 * Go through an array of X11 clipboard targets to see if they contain a bitmap
419 * format we can support, and if so choose the ones we prefer (e.g. we like
420 * BMP better than PNG because we don't have to convert).
421 * @param pCtx the clipboard backend context structure
422 * @param pTargets the list of targets
423 * @param cTargets the size of the list in @a pTargets
424 */
425static CLIPX11FORMAT clipGetBitmapFormatFromTargets(CLIPBACKEND *pCtx,
426 CLIPX11FORMAT *pTargets,
427 size_t cTargets)
428{
429 CLIPX11FORMAT bestBitmapFormat = NIL_CLIPX11FORMAT;
430 CLIPFORMAT enmBestBitmapTarget = INVALID;
431 AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT);
432 AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT);
433 for (unsigned i = 0; i < cTargets; ++i)
434 {
435 CLIPX11FORMAT format = pTargets[i];
436 if (format != NIL_CLIPX11FORMAT)
437 {
438 if ( (clipVBoxFormatForX11Format(format)
439 == VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
440 && enmBestBitmapTarget < clipRealFormatForX11Format(format))
441 {
442 enmBestBitmapTarget = clipRealFormatForX11Format(format);
443 bestBitmapFormat = format;
444 }
445 }
446 }
447 return bestBitmapFormat;
448}
449
450/**
451 * Go through an array of X11 clipboard targets to see if they contain a HTML
452 * format we can support, and if so choose the ones we prefer
453 * @param pCtx the clipboard backend context structure
454 * @param pTargets the list of targets
455 * @param cTargets the size of the list in @a pTargets
456 */
457static CLIPX11FORMAT clipGetHtmlFormatFromTargets(CLIPBACKEND *pCtx,
458 CLIPX11FORMAT *pTargets,
459 size_t cTargets)
460{
461 CLIPX11FORMAT bestHTMLFormat = NIL_CLIPX11FORMAT;
462 CLIPFORMAT enmBestHtmlTarget = INVALID;
463 AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT);
464 AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT);
465 for (unsigned i = 0; i < cTargets; ++i)
466 {
467 CLIPX11FORMAT format = pTargets[i];
468 if (format != NIL_CLIPX11FORMAT)
469 {
470 if ( (clipVBoxFormatForX11Format(format) == VBOX_SHARED_CLIPBOARD_FMT_HTML)
471 && enmBestHtmlTarget < clipRealFormatForX11Format(format))
472 {
473 enmBestHtmlTarget = clipRealFormatForX11Format(format);
474 bestHTMLFormat = format;
475 }
476 }
477 }
478 return bestHTMLFormat;
479}
480
481
482/**
483 * Go through an array of X11 clipboard targets to see if we can support any
484 * of them and if relevant to choose the ones we prefer (e.g. we like Utf8
485 * better than plain text).
486 * @param pCtx the clipboard backend context structure
487 * @param pTargets the list of targets
488 * @param cTargets the size of the list in @a pTargets
489 */
490static void clipGetFormatsFromTargets(CLIPBACKEND *pCtx,
491 CLIPX11FORMAT *pTargets, size_t cTargets)
492{
493 AssertPtrReturnVoid(pCtx);
494 AssertPtrReturnVoid(pTargets);
495 CLIPX11FORMAT bestTextFormat;
496 CLIPX11FORMAT bestBitmapFormat;
497 CLIPX11FORMAT bestHtmlFormat;
498 bestTextFormat = clipGetTextFormatFromTargets(pCtx, pTargets, cTargets);
499 if (pCtx->X11TextFormat != bestTextFormat)
500 {
501 pCtx->X11TextFormat = bestTextFormat;
502 }
503 pCtx->X11BitmapFormat = INVALID; /* not yet supported */
504 bestBitmapFormat = clipGetBitmapFormatFromTargets(pCtx, pTargets, cTargets);
505 if (pCtx->X11BitmapFormat != bestBitmapFormat)
506 {
507 pCtx->X11BitmapFormat = bestBitmapFormat;
508 }
509 bestHtmlFormat = clipGetHtmlFormatFromTargets(pCtx, pTargets, cTargets);
510 if(pCtx->X11HTMLFormat != bestHtmlFormat)
511 {
512 pCtx->X11HTMLFormat = bestHtmlFormat;
513 }
514}
515
516static void clipQueryX11CBFormats(CLIPBACKEND *pCtx);
517
518/**
519 * Update the context's information about targets currently supported by X11,
520 * based on an array of X11 atoms.
521 * @param pCtx the context to be updated
522 * @param pTargets the array of atoms describing the targets supported
523 * @param cTargets the size of the array @a pTargets
524 */
525static void clipUpdateX11Targets(CLIPBACKEND *pCtx, CLIPX11FORMAT *pTargets,
526 size_t cTargets)
527{
528 LogRel2 (("%s: called\n", __FUNCTION__));
529 pCtx->fBusy = false;
530 if (pCtx->fUpdateNeeded)
531 {
532 /* We may already be out of date. */
533 pCtx->fUpdateNeeded = false;
534 clipQueryX11CBFormats(pCtx);
535 return;
536 }
537 if (pTargets == NULL) {
538 /* No data available */
539 clipReportEmptyX11CB(pCtx);
540 return;
541 }
542 clipGetFormatsFromTargets(pCtx, pTargets, cTargets);
543 clipReportFormatsToVBox(pCtx);
544}
545
546/**
547 * Notify the VBox clipboard about available data formats, based on the
548 * "targets" information obtained from the X11 clipboard.
549 * @note Callback for XtGetSelectionValue
550 * @note This function is treated as API glue, and as such is not part of any
551 * unit test. So keep it simple, be paranoid and log everything.
552 */
553static void clipConvertX11Targets(Widget widget, XtPointer pClientData,
554 Atom * /* selection */, Atom *atomType,
555 XtPointer pValue, long unsigned int *pcLen,
556 int *piFormat)
557{
558 CLIPBACKEND *pCtx =
559 reinterpret_cast<CLIPBACKEND *>(pClientData);
560 Atom *pAtoms = (Atom *)pValue;
561 unsigned i, j;
562 LogRel2(("%s: pValue=%p, *pcLen=%u, *atomType=%d%s\n", __FUNCTION__,
563 pValue, *pcLen, *atomType,
564 *atomType == XT_CONVERT_FAIL ? " (XT_CONVERT_FAIL)" : ""));
565 CLIPX11FORMAT *pFormats = NULL;
566 if (*pcLen && pValue && (*atomType != XT_CONVERT_FAIL /* time out */))
567 pFormats = (CLIPX11FORMAT *)RTMemAllocZ(*pcLen * sizeof(CLIPX11FORMAT));
568#if defined(DEBUG) && !defined(TESTCASE)
569 if (pValue)
570 for (i = 0; i < *pcLen; ++i)
571 if (pAtoms[i])
572 {
573 char *pszName = XGetAtomName(XtDisplay(widget), pAtoms[i]);
574 LogRel2(("%s: found target %s\n", __FUNCTION__,
575 pszName));
576 XFree(pszName);
577 }
578 else
579 LogRel2(("%s: found empty target.\n", __FUNCTION__));
580#endif
581 if (pFormats)
582 {
583 for (i = 0; i < *pcLen; ++i)
584 {
585 for (j = 0; j < RT_ELEMENTS(g_aFormats); ++j)
586 {
587 Atom target = XInternAtom(XtDisplay(widget),
588 g_aFormats[j].pcszAtom, False);
589 if (*(pAtoms + i) == target)
590 pFormats[i] = j;
591 }
592#if defined(DEBUG) && !defined(TESTCASE)
593 LogRel2(("%s: reporting format %d (%s)\n", __FUNCTION__,
594 pFormats[i], g_aFormats[pFormats[i]].pcszAtom));
595#endif
596 }
597 }
598 else
599 LogRel2(("%s: reporting empty targets (none reported or allocation failure).\n",
600 __FUNCTION__));
601 clipUpdateX11Targets(pCtx, pFormats, *pcLen);
602 RTMemFree(pFormats);
603 XtFree(reinterpret_cast<char *>(pValue));
604}
605
606#ifdef TESTCASE
607 void testRequestTargets(CLIPBACKEND *pCtx);
608#endif
609
610/**
611 * Callback to notify us when the contents of the X11 clipboard change.
612 */
613static void clipQueryX11CBFormats(CLIPBACKEND *pCtx)
614{
615 LogRel2 (("%s: requesting the targets that the X11 clipboard offers\n",
616 __PRETTY_FUNCTION__));
617 if (pCtx->fBusy)
618 {
619 pCtx->fUpdateNeeded = true;
620 return;
621 }
622 pCtx->fBusy = true;
623#ifndef TESTCASE
624 XtGetSelectionValue(pCtx->widget,
625 clipGetAtom(pCtx, "CLIPBOARD"),
626 clipGetAtom(pCtx, "TARGETS"),
627 clipConvertX11Targets, pCtx,
628 CurrentTime);
629#else
630 testRequestTargets(pCtx);
631#endif
632}
633
634#ifndef TESTCASE
635
636typedef struct {
637 int type; /* event base */
638 unsigned long serial;
639 Bool send_event;
640 Display *display;
641 Window window;
642 int subtype;
643 Window owner;
644 Atom selection;
645 Time timestamp;
646 Time selection_timestamp;
647} XFixesSelectionNotifyEvent;
648
649/**
650 * Wait until an event arrives and handle it if it is an XFIXES selection
651 * event, which Xt doesn't know about.
652 */
653void clipPeekEventAndDoXFixesHandling(CLIPBACKEND *pCtx)
654{
655 union
656 {
657 XEvent event;
658 XFixesSelectionNotifyEvent fixes;
659 } event = { { 0 } };
660
661 if (XtAppPeekEvent(pCtx->appContext, &event.event))
662 if ( (event.event.type == pCtx->fixesEventBase)
663 && (event.fixes.owner != XtWindow(pCtx->widget)))
664 {
665 if ( (event.fixes.subtype == 0 /* XFixesSetSelectionOwnerNotify */)
666 && (event.fixes.owner != 0))
667 clipQueryX11CBFormats(pCtx);
668 else
669 clipReportEmptyX11CB(pCtx);
670 }
671}
672
673/**
674 * The main loop of our clipboard reader.
675 * @note X11 backend code.
676 */
677static DECLCALLBACK(int) clipEventThread(RTTHREAD self, void *pvUser)
678{
679 LogRel(("Shared clipboard: Starting shared clipboard thread\n"));
680
681 CLIPBACKEND *pCtx = (CLIPBACKEND *)pvUser;
682
683 if (pCtx->fGrabClipboardOnStart)
684 clipQueryX11CBFormats(pCtx);
685 while (XtAppGetExitFlag(pCtx->appContext) == FALSE)
686 {
687 clipPeekEventAndDoXFixesHandling(pCtx);
688 XtAppProcessEvent(pCtx->appContext, XtIMAll);
689 }
690 LogRel(("Shared clipboard: Shared clipboard thread terminated successfully\n"));
691 return VINF_SUCCESS;
692}
693#endif
694
695/** X11 specific uninitialisation for the shared clipboard.
696 * @note X11 backend code.
697 */
698static void clipUninit(CLIPBACKEND *pCtx)
699{
700 AssertPtrReturnVoid(pCtx);
701 if (pCtx->widget)
702 {
703 /* Valid widget + invalid appcontext = bug. But don't return yet. */
704 AssertPtr(pCtx->appContext);
705 clipUnregisterContext(pCtx);
706 XtDestroyWidget(pCtx->widget);
707 }
708 pCtx->widget = NULL;
709 if (pCtx->appContext)
710 XtDestroyApplicationContext(pCtx->appContext);
711 pCtx->appContext = NULL;
712 if (pCtx->wakeupPipeRead != 0)
713 close(pCtx->wakeupPipeRead);
714 if (pCtx->wakeupPipeWrite != 0)
715 close(pCtx->wakeupPipeWrite);
716 pCtx->wakeupPipeRead = 0;
717 pCtx->wakeupPipeWrite = 0;
718}
719
720/** Worker function for stopping the clipboard which runs on the event
721 * thread. */
722static void clipStopEventThreadWorker(void *pUserData, void *)
723{
724
725 CLIPBACKEND *pCtx = (CLIPBACKEND *)pUserData;
726
727 /* This might mean that we are getting stopped twice. */
728 Assert(pCtx->widget != NULL);
729
730 /* Set the termination flag to tell the Xt event loop to exit. We
731 * reiterate that any outstanding requests from the X11 event loop to
732 * the VBox part *must* have returned before we do this. */
733 XtAppSetExitFlag(pCtx->appContext);
734}
735
736#ifndef TESTCASE
737/** Setup the XFixes library and load the XFixesSelectSelectionInput symbol */
738static int clipLoadXFixes(Display *pDisplay, CLIPBACKEND *pCtx)
739{
740 int dummy1 = 0, dummy2 = 0, rc = VINF_SUCCESS;
741 void *hFixesLib;
742
743 hFixesLib = dlopen("libXfixes.so.1", RTLD_LAZY);
744 if (!hFixesLib)
745 hFixesLib = dlopen("libXfixes.so.2", RTLD_LAZY);
746 if (!hFixesLib)
747 hFixesLib = dlopen("libXfixes.so.3", RTLD_LAZY);
748 if (hFixesLib)
749 pCtx->fixesSelectInput =
750 (void (*)(Display *, Window, Atom, long unsigned int))
751 (uintptr_t)dlsym(hFixesLib, "XFixesSelectSelectionInput");
752 /* For us, a NULL function pointer is a failure */
753 if (!hFixesLib || !pCtx->fixesSelectInput)
754 rc = VERR_NOT_SUPPORTED;
755 if ( RT_SUCCESS(rc)
756 && !XQueryExtension(pDisplay, "XFIXES", &dummy1,
757 &pCtx->fixesEventBase, &dummy2))
758 rc = VERR_NOT_SUPPORTED;
759 if (RT_SUCCESS(rc) && pCtx->fixesEventBase < 0)
760 rc = VERR_NOT_SUPPORTED;
761 return rc;
762}
763#endif
764
765/** This is the callback which is scheduled when data is available on the
766 * wakeup pipe. It simply reads all data from the pipe. */
767static void clipDrainWakeupPipe(XtPointer pUserData, int *, XtInputId *)
768{
769 CLIPBACKEND *pCtx = (CLIPBACKEND *)pUserData;
770 char acBuf[WAKE_UP_STRING_LEN];
771
772 LogRel2(("clipDrainWakeupPipe: called\n"));
773 while (read(pCtx->wakeupPipeRead, acBuf, sizeof(acBuf)) > 0) {}
774}
775
776/** X11 specific initialisation for the shared clipboard.
777 * @note X11 backend code.
778 */
779static int clipInit(CLIPBACKEND *pCtx)
780{
781 /* Create a window and make it a clipboard viewer. */
782 int cArgc = 0;
783 char *pcArgv = 0;
784 int rc = VINF_SUCCESS;
785 Display *pDisplay;
786
787 /* Make sure we are thread safe */
788 XtToolkitThreadInitialize();
789 /* Set up the Clipboard application context and main window. We call all
790 * these functions directly instead of calling XtOpenApplication() so
791 * that we can fail gracefully if we can't get an X11 display. */
792 XtToolkitInitialize();
793 pCtx->appContext = XtCreateApplicationContext();
794 pDisplay = XtOpenDisplay(pCtx->appContext, 0, 0, "VBoxClipboard", 0, 0, &cArgc, &pcArgv);
795 if (NULL == pDisplay)
796 {
797 LogRel(("Shared clipboard: Failed to connect to the X11 clipboard - the window system may not be running.\n"));
798 rc = VERR_NOT_SUPPORTED;
799 }
800#ifndef TESTCASE
801 if (RT_SUCCESS(rc))
802 {
803 rc = clipLoadXFixes(pDisplay, pCtx);
804 if (RT_FAILURE(rc))
805 LogRel(("Shared clipboard: Failed to load the XFIXES extension.\n"));
806 }
807#endif
808 if (RT_SUCCESS(rc))
809 {
810 pCtx->widget = XtVaAppCreateShell(0, "VBoxClipboard",
811 applicationShellWidgetClass,
812 pDisplay, XtNwidth, 1, XtNheight,
813 1, NULL);
814 if (NULL == pCtx->widget)
815 {
816 LogRel(("Shared clipboard: Failed to construct the X11 window for the shared clipboard manager.\n"));
817 rc = VERR_NO_MEMORY;
818 }
819 else
820 rc = clipRegisterContext(pCtx);
821 }
822 if (RT_SUCCESS(rc))
823 {
824 EventMask mask = 0;
825
826 XtSetMappedWhenManaged(pCtx->widget, false);
827 XtRealizeWidget(pCtx->widget);
828#ifndef TESTCASE
829 /* Enable clipboard update notification */
830 pCtx->fixesSelectInput(pDisplay, XtWindow(pCtx->widget),
831 clipGetAtom(pCtx, "CLIPBOARD"),
832 7 /* All XFixes*Selection*NotifyMask flags */);
833#endif
834 }
835 /* Create the pipes */
836 int pipes[2];
837 if (!pipe(pipes))
838 {
839 pCtx->wakeupPipeRead = pipes[0];
840 pCtx->wakeupPipeWrite = pipes[1];
841 if (!XtAppAddInput(pCtx->appContext, pCtx->wakeupPipeRead,
842 (XtPointer) XtInputReadMask,
843 clipDrainWakeupPipe, (XtPointer) pCtx))
844 rc = VERR_NO_MEMORY; /* What failure means is not doc'ed. */
845 if ( RT_SUCCESS(rc)
846 && (fcntl(pCtx->wakeupPipeRead, F_SETFL, O_NONBLOCK) != 0))
847 rc = RTErrConvertFromErrno(errno);
848 if (RT_FAILURE(rc))
849 LogRel(("Shared clipboard: Failed to setup the termination mechanism.\n"));
850 }
851 else
852 rc = RTErrConvertFromErrno(errno);
853 if (RT_FAILURE(rc))
854 clipUninit(pCtx);
855 if (RT_FAILURE(rc))
856 LogRel(("Shared clipboard: Initialisation failed: %Rrc\n", rc));
857 return rc;
858}
859
860/**
861 * Construct the X11 backend of the shared clipboard.
862 * @note X11 backend code
863 */
864CLIPBACKEND *ClipConstructX11(VBOXCLIPBOARDCONTEXT *pFrontend, bool fHeadless)
865{
866 int rc;
867
868 CLIPBACKEND *pCtx = (CLIPBACKEND *)
869 RTMemAllocZ(sizeof(CLIPBACKEND));
870 if (pCtx && fHeadless)
871 {
872 /*
873 * If we don't find the DISPLAY environment variable we assume that
874 * we are not connected to an X11 server. Don't actually try to do
875 * this then, just fail silently and report success on every call.
876 * This is important for VBoxHeadless.
877 */
878 LogRelFunc(("X11 DISPLAY variable not set -- disabling shared clipboard\n"));
879 pCtx->fHaveX11 = false;
880 return pCtx;
881 }
882
883 pCtx->fHaveX11 = true;
884
885 LogRel(("Shared clipboard: Initializing X11 clipboard backend\n"));
886 if (pCtx)
887 pCtx->pFrontend = pFrontend;
888 return pCtx;
889}
890
891/**
892 * Destruct the shared clipboard X11 backend.
893 * @note X11 backend code
894 */
895void ClipDestructX11(CLIPBACKEND *pCtx)
896{
897 /*
898 * Immediately return if we are not connected to the X server.
899 */
900 if (!pCtx->fHaveX11)
901 return;
902
903 /* We set this to NULL when the event thread exits. It really should
904 * have exited at this point, when we are about to unload the code from
905 * memory. */
906 Assert(pCtx->widget == NULL);
907}
908
909/**
910 * Announce to the X11 backend that we are ready to start.
911 * @param grab whether we should try to grab the shared clipboard at once
912 */
913int ClipStartX11(CLIPBACKEND *pCtx, bool grab)
914{
915 int rc = VINF_SUCCESS;
916 LogRelFlowFunc(("\n"));
917 /*
918 * Immediately return if we are not connected to the X server.
919 */
920 if (!pCtx->fHaveX11)
921 return VINF_SUCCESS;
922
923 rc = clipInit(pCtx);
924 if (RT_SUCCESS(rc))
925 {
926 clipResetX11Formats(pCtx);
927 pCtx->fGrabClipboardOnStart = grab;
928 }
929#ifndef TESTCASE
930 if (RT_SUCCESS(rc))
931 {
932 rc = RTThreadCreate(&pCtx->thread, clipEventThread, pCtx, 0,
933 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "SHCLIP");
934 if (RT_FAILURE(rc))
935 {
936 LogRel(("Shared clipboard: Failed to start the shared clipboard thread.\n"));
937 clipUninit(pCtx);
938 }
939 }
940#endif
941 return rc;
942}
943
944/**
945 * Shut down the shared clipboard X11 backend.
946 * @note X11 backend code
947 * @note Any requests from this object to get clipboard data from VBox
948 * *must* have completed or aborted before we are called, as
949 * otherwise the X11 event loop will still be waiting for the request
950 * to return and will not be able to terminate.
951 */
952int ClipStopX11(CLIPBACKEND *pCtx)
953{
954 int rc, rcThread;
955 unsigned count = 0;
956 /*
957 * Immediately return if we are not connected to the X server.
958 */
959 if (!pCtx->fHaveX11)
960 return VINF_SUCCESS;
961
962 LogRelFunc(("stopping the shared clipboard X11 backend\n"));
963 /* Write to the "stop" pipe */
964 clipQueueToEventThread(pCtx, clipStopEventThreadWorker, (XtPointer) pCtx);
965#ifndef TESTCASE
966 do
967 {
968 rc = RTThreadWait(pCtx->thread, 1000, &rcThread);
969 ++count;
970 Assert(RT_SUCCESS(rc) || ((VERR_TIMEOUT == rc) && (count != 5)));
971 } while ((VERR_TIMEOUT == rc) && (count < 300));
972#else
973 rc = VINF_SUCCESS;
974 rcThread = VINF_SUCCESS;
975#endif
976 if (RT_SUCCESS(rc))
977 AssertRC(rcThread);
978 else
979 LogRelFunc(("rc=%Rrc\n", rc));
980 clipUninit(pCtx);
981 LogRelFlowFunc(("returning %Rrc.\n", rc));
982 return rc;
983}
984
985/**
986 * Satisfy a request from X11 for clipboard targets supported by VBox.
987 *
988 * @returns iprt status code
989 * @param atomTypeReturn The type of the data we are returning
990 * @param pValReturn A pointer to the data we are returning. This
991 * should be set to memory allocated by XtMalloc,
992 * which will be freed later by the Xt toolkit.
993 * @param pcLenReturn The length of the data we are returning
994 * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are
995 * returning
996 * @note X11 backend code, called by the XtOwnSelection callback.
997 */
998static int clipCreateX11Targets(CLIPBACKEND *pCtx, Atom *atomTypeReturn,
999 XtPointer *pValReturn,
1000 unsigned long *pcLenReturn,
1001 int *piFormatReturn)
1002{
1003 Atom *atomTargets = (Atom *)XtMalloc( (MAX_CLIP_X11_FORMATS + 3)
1004 * sizeof(Atom));
1005 unsigned cTargets = 0;
1006 LogRelFlowFunc (("called\n"));
1007 CLIPX11FORMAT format = NIL_CLIPX11FORMAT;
1008 do
1009 {
1010 format = clipEnumX11Formats(pCtx->vboxFormats, format);
1011 if (format != NIL_CLIPX11FORMAT)
1012 {
1013 atomTargets[cTargets] = clipAtomForX11Format(pCtx, format);
1014 ++cTargets;
1015 }
1016 } while (format != NIL_CLIPX11FORMAT);
1017 /* We always offer these */
1018 atomTargets[cTargets] = clipGetAtom(pCtx, "TARGETS");
1019 atomTargets[cTargets + 1] = clipGetAtom(pCtx, "MULTIPLE");
1020 atomTargets[cTargets + 2] = clipGetAtom(pCtx, "TIMESTAMP");
1021 *atomTypeReturn = XA_ATOM;
1022 *pValReturn = (XtPointer)atomTargets;
1023 *pcLenReturn = cTargets + 3;
1024 *piFormatReturn = 32;
1025 return VINF_SUCCESS;
1026}
1027
1028/** This is a wrapper around ClipRequestDataForX11 that will cache the
1029 * data returned.
1030 */
1031static int clipReadVBoxClipboard(CLIPBACKEND *pCtx, uint32_t u32Format,
1032 void **ppv, uint32_t *pcb)
1033{
1034 int rc = VINF_SUCCESS;
1035 LogRelFlowFunc(("pCtx=%p, u32Format=%02X, ppv=%p, pcb=%p\n", pCtx,
1036 u32Format, ppv, pcb));
1037 if (u32Format == VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
1038 {
1039 if (pCtx->pvUnicodeCache == NULL)
1040 rc = ClipRequestDataForX11(pCtx->pFrontend, u32Format,
1041 &pCtx->pvUnicodeCache,
1042 &pCtx->cbUnicodeCache);
1043 if (RT_SUCCESS(rc))
1044 {
1045 *ppv = RTMemDup(pCtx->pvUnicodeCache, pCtx->cbUnicodeCache);
1046 *pcb = pCtx->cbUnicodeCache;
1047 if (*ppv == NULL)
1048 rc = VERR_NO_MEMORY;
1049 }
1050 }
1051 else
1052 rc = ClipRequestDataForX11(pCtx->pFrontend, u32Format,
1053 ppv, pcb);
1054 LogRelFlowFunc(("returning %Rrc\n", rc));
1055 if (RT_SUCCESS(rc))
1056 LogRelFlowFunc(("*ppv=%.*ls, *pcb=%u\n", *pcb, *ppv, *pcb));
1057 return rc;
1058}
1059
1060/**
1061 * Calculate a buffer size large enough to hold the source Windows format
1062 * text converted into Unix Utf8, including the null terminator
1063 * @returns iprt status code
1064 * @param pwsz the source text in UCS-2 with Windows EOLs
1065 * @param cwc the size in USC-2 elements of the source text, with or
1066 * without the terminator
1067 * @param pcbActual where to store the buffer size needed
1068 */
1069static int clipWinTxtBufSizeForUtf8(PRTUTF16 pwsz, size_t cwc,
1070 size_t *pcbActual)
1071{
1072 size_t cbRet = 0;
1073 int rc = RTUtf16CalcUtf8LenEx(pwsz, cwc, &cbRet);
1074 if (RT_SUCCESS(rc))
1075 *pcbActual = cbRet + 1; /* null terminator */
1076 return rc;
1077}
1078
1079/**
1080 * Convert text from Windows format (UCS-2 with CRLF line endings) to standard
1081 * Utf-8.
1082 *
1083 * @returns iprt status code
1084 *
1085 * @param pwszSrc the text to be converted
1086 * @param cbSrc the length of @a pwszSrc in bytes
1087 * @param pszBuf where to write the converted string
1088 * @param cbBuf the size of the buffer pointed to by @a pszBuf
1089 * @param pcbActual where to store the size of the converted string.
1090 * optional.
1091 */
1092static int clipWinTxtToUtf8(PRTUTF16 pwszSrc, size_t cbSrc, char *pszBuf,
1093 size_t cbBuf, size_t *pcbActual)
1094{
1095 PRTUTF16 pwszTmp = NULL;
1096 size_t cwSrc = cbSrc / 2, cwTmp = 0, cbDest = 0;
1097 int rc = VINF_SUCCESS;
1098
1099 LogRelFlowFunc (("pwszSrc=%.*ls, cbSrc=%u\n", cbSrc, pwszSrc, cbSrc));
1100 /* How long will the converted text be? */
1101 AssertPtr(pwszSrc);
1102 AssertPtr(pszBuf);
1103 rc = vboxClipboardUtf16GetLinSize(pwszSrc, cwSrc, &cwTmp);
1104 if (RT_SUCCESS(rc) && cwTmp == 0)
1105 rc = VERR_NO_DATA;
1106 if (RT_SUCCESS(rc))
1107 pwszTmp = (PRTUTF16)RTMemAlloc(cwTmp * 2);
1108 if (!pwszTmp)
1109 rc = VERR_NO_MEMORY;
1110 /* Convert the text. */
1111 if (RT_SUCCESS(rc))
1112 rc = vboxClipboardUtf16WinToLin(pwszSrc, cwSrc, pwszTmp, cwTmp);
1113 if (RT_SUCCESS(rc))
1114 /* Convert the Utf16 string to Utf8. */
1115 rc = RTUtf16ToUtf8Ex(pwszTmp + 1, cwTmp - 1, &pszBuf, cbBuf,
1116 &cbDest);
1117 RTMemFree(reinterpret_cast<void *>(pwszTmp));
1118 if (pcbActual)
1119 *pcbActual = cbDest + 1;
1120 LogRelFlowFunc(("returning %Rrc\n", rc));
1121 if (RT_SUCCESS(rc))
1122 LogRelFlowFunc (("converted string is %.*s. Returning.\n", cbDest,
1123 pszBuf));
1124 return rc;
1125}
1126
1127/**
1128 * Satisfy a request from X11 to convert the clipboard text to Utf-8. We
1129 * return null-terminated text, but can cope with non-null-terminated input.
1130 *
1131 * @returns iprt status code
1132 * @param pDisplay an X11 display structure, needed for conversions
1133 * performed by Xlib
1134 * @param pv the text to be converted (UCS-2 with Windows EOLs)
1135 * @param cb the length of the text in @cb in bytes
1136 * @param atomTypeReturn where to store the atom for the type of the data
1137 * we are returning
1138 * @param pValReturn where to store the pointer to the data we are
1139 * returning. This should be to memory allocated by
1140 * XtMalloc, which will be freed by the Xt toolkit
1141 * later.
1142 * @param pcLenReturn where to store the length of the data we are
1143 * returning
1144 * @param piFormatReturn where to store the bit width (8, 16, 32) of the
1145 * data we are returning
1146 */
1147static int clipWinTxtToUtf8ForX11CB(Display *pDisplay, PRTUTF16 pwszSrc,
1148 size_t cbSrc, Atom *atomTarget,
1149 Atom *atomTypeReturn,
1150 XtPointer *pValReturn,
1151 unsigned long *pcLenReturn,
1152 int *piFormatReturn)
1153{
1154 /* This may slightly overestimate the space needed. */
1155 size_t cbDest = 0;
1156 int rc = clipWinTxtBufSizeForUtf8(pwszSrc, cbSrc / 2, &cbDest);
1157 if (RT_SUCCESS(rc))
1158 {
1159 char *pszDest = (char *)XtMalloc(cbDest);
1160 size_t cbActual = 0;
1161 if (pszDest)
1162 rc = clipWinTxtToUtf8(pwszSrc, cbSrc, pszDest, cbDest,
1163 &cbActual);
1164 if (RT_SUCCESS(rc))
1165 {
1166 *atomTypeReturn = *atomTarget;
1167 *pValReturn = (XtPointer)pszDest;
1168 *pcLenReturn = cbActual;
1169 *piFormatReturn = 8;
1170 }
1171 }
1172 return rc;
1173}
1174
1175/**
1176 * Satisfy a request from X11 to convert the clipboard HTML fragment to Utf-8. We
1177 * return null-terminated text, but can cope with non-null-terminated input.
1178 *
1179 * @returns iprt status code
1180 * @param pDisplay an X11 display structure, needed for conversions
1181 * performed by Xlib
1182 * @param pv the text to be converted (UTF8 with Windows EOLs)
1183 * @param cb the length of the text in @cb in bytes
1184 * @param atomTypeReturn where to store the atom for the type of the data
1185 * we are returning
1186 * @param pValReturn where to store the pointer to the data we are
1187 * returning. This should be to memory allocated by
1188 * XtMalloc, which will be freed by the Xt toolkit
1189 * later.
1190 * @param pcLenReturn where to store the length of the data we are
1191 * returning
1192 * @param piFormatReturn where to store the bit width (8, 16, 32) of the
1193 * data we are returning
1194 */
1195static int clipWinHTMLToUtf8ForX11CB(Display *pDisplay, const char *pszSrc,
1196 size_t cbSrc, Atom *atomTarget,
1197 Atom *atomTypeReturn,
1198 XtPointer *pValReturn,
1199 unsigned long *pcLenReturn,
1200 int *piFormatReturn)
1201{
1202 /* This may slightly overestimate the space needed. */
1203 LogRelFlowFunc(("source: %s", pszSrc));
1204
1205 char *pszDest = (char *)XtMalloc(cbSrc);
1206 if(pszDest == NULL)
1207 return VERR_NO_MEMORY;
1208
1209 memcpy(pszDest, pszSrc, cbSrc);
1210
1211 *atomTypeReturn = *atomTarget;
1212 *pValReturn = (XtPointer)pszDest;
1213 *pcLenReturn = cbSrc;
1214 *piFormatReturn = 8;
1215
1216 return VINF_SUCCESS;
1217}
1218
1219
1220/**
1221 * Does this atom correspond to one of the two selection types we support?
1222 * @param widget a valid Xt widget
1223 * @param selType the atom in question
1224 */
1225static bool clipIsSupportedSelectionType(CLIPBACKEND *pCtx, Atom selType)
1226{
1227 return( (selType == clipGetAtom(pCtx, "CLIPBOARD"))
1228 || (selType == clipGetAtom(pCtx, "PRIMARY")));
1229}
1230
1231/**
1232 * Remove a trailing nul character from a string by adjusting the string
1233 * length. Some X11 applications don't like zero-terminated text...
1234 * @param pText the text in question
1235 * @param pcText the length of the text, adjusted on return
1236 * @param format the format of the text
1237 */
1238static void clipTrimTrailingNul(XtPointer pText, unsigned long *pcText,
1239 CLIPFORMAT format)
1240{
1241 AssertPtrReturnVoid(pText);
1242 AssertPtrReturnVoid(pcText);
1243 AssertReturnVoid((format == UTF8) || (format == TEXT) || (format == HTML));
1244 if (((char *)pText)[*pcText - 1] == '\0')
1245 --(*pcText);
1246}
1247
1248static int clipConvertVBoxCBForX11(CLIPBACKEND *pCtx, Atom *atomTarget,
1249 Atom *atomTypeReturn,
1250 XtPointer *pValReturn,
1251 unsigned long *pcLenReturn,
1252 int *piFormatReturn)
1253{
1254 int rc = VINF_SUCCESS;
1255 CLIPX11FORMAT x11Format = clipFindX11FormatByAtom(pCtx, *atomTarget);
1256 CLIPFORMAT format = clipRealFormatForX11Format(x11Format);
1257 if ( ((format == UTF8) || (format == TEXT))
1258 && (pCtx->vboxFormats & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT))
1259 {
1260 void *pv = NULL;
1261 uint32_t cb = 0;
1262 rc = clipReadVBoxClipboard(pCtx,
1263 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
1264 &pv, &cb);
1265 if (RT_SUCCESS(rc) && (cb == 0))
1266 rc = VERR_NO_DATA;
1267 if (RT_SUCCESS(rc) && ((format == UTF8) || (format == TEXT)))
1268 rc = clipWinTxtToUtf8ForX11CB(XtDisplay(pCtx->widget),
1269 (PRTUTF16)pv, cb, atomTarget,
1270 atomTypeReturn, pValReturn,
1271 pcLenReturn, piFormatReturn);
1272 if (RT_SUCCESS(rc))
1273 clipTrimTrailingNul(*(XtPointer *)pValReturn, pcLenReturn, format);
1274 RTMemFree(pv);
1275 }
1276 else if ( (format == BMP)
1277 && (pCtx->vboxFormats & VBOX_SHARED_CLIPBOARD_FMT_BITMAP))
1278 {
1279 void *pv = NULL;
1280 uint32_t cb = 0;
1281 rc = clipReadVBoxClipboard(pCtx,
1282 VBOX_SHARED_CLIPBOARD_FMT_BITMAP,
1283 &pv, &cb);
1284 if (RT_SUCCESS(rc) && (cb == 0))
1285 rc = VERR_NO_DATA;
1286 if (RT_SUCCESS(rc) && (format == BMP))
1287 {
1288 /* Create a full BMP from it */
1289 rc = vboxClipboardDibToBmp(pv, cb, (void **)pValReturn,
1290 (size_t *)pcLenReturn);
1291 }
1292 else
1293 rc = VERR_NOT_SUPPORTED;
1294
1295 if (RT_SUCCESS(rc))
1296 {
1297 *atomTypeReturn = *atomTarget;
1298 *piFormatReturn = 8;
1299 }
1300 RTMemFree(pv);
1301 }
1302 else if ( (format == HTML)
1303 && (pCtx->vboxFormats & VBOX_SHARED_CLIPBOARD_FMT_HTML))
1304 {
1305 void *pv = NULL;
1306 uint32_t cb = 0;
1307 rc = clipReadVBoxClipboard(pCtx,
1308 VBOX_SHARED_CLIPBOARD_FMT_HTML,
1309 &pv, &cb);
1310 if (RT_SUCCESS(rc) && (cb == 0))
1311 rc = VERR_NO_DATA;
1312 if (RT_SUCCESS(rc))
1313 {
1314 /*
1315 * The common VBox HTML encoding will be - Utf8
1316 * becuase it more general for HTML formats then UTF16
1317 * X11 clipboard returns UTF16, so before sending it we should
1318 * convert it to UTF8
1319 * It's very strange but here we get utf16 from x11 clipboard
1320 * in same time we send utf8 to x11 clipboard and it's work
1321 */
1322 rc = clipWinHTMLToUtf8ForX11CB(XtDisplay(pCtx->widget),
1323 (const char*)pv, cb, atomTarget,
1324 atomTypeReturn, pValReturn,
1325 pcLenReturn, piFormatReturn);
1326
1327
1328 if (RT_SUCCESS(rc))
1329 clipTrimTrailingNul(*(XtPointer *)pValReturn, pcLenReturn, format);
1330 RTMemFree(pv);
1331 }
1332 }
1333 else
1334 rc = VERR_NOT_SUPPORTED;
1335 return rc;
1336}
1337
1338/**
1339 * Return VBox's clipboard data for an X11 client.
1340 * @note X11 backend code, callback for XtOwnSelection
1341 */
1342static Boolean clipXtConvertSelectionProc(Widget widget, Atom *atomSelection,
1343 Atom *atomTarget,
1344 Atom *atomTypeReturn,
1345 XtPointer *pValReturn,
1346 unsigned long *pcLenReturn,
1347 int *piFormatReturn)
1348{
1349 CLIPBACKEND *pCtx = clipLookupContext(widget);
1350 int rc = VINF_SUCCESS;
1351
1352 LogRelFlowFunc(("\n"));
1353 if (!pCtx)
1354 return false;
1355 if (!clipIsSupportedSelectionType(pCtx, *atomSelection))
1356 return false;
1357 if (*atomTarget == clipGetAtom(pCtx, "TARGETS"))
1358 rc = clipCreateX11Targets(pCtx, atomTypeReturn, pValReturn,
1359 pcLenReturn, piFormatReturn);
1360 else
1361 rc = clipConvertVBoxCBForX11(pCtx, atomTarget, atomTypeReturn,
1362 pValReturn, pcLenReturn, piFormatReturn);
1363 LogRelFlowFunc(("returning, internal status code %Rrc\n", rc));
1364 return RT_SUCCESS(rc);
1365}
1366
1367/** Structure used to pass information about formats that VBox supports */
1368typedef struct _CLIPNEWVBOXFORMATS
1369{
1370 /** Context information for the X11 clipboard */
1371 CLIPBACKEND *pCtx;
1372 /** Formats supported by VBox */
1373 uint32_t formats;
1374} CLIPNEWVBOXFORMATS;
1375
1376/** Invalidates the local cache of the data in the VBox clipboard. */
1377static void clipInvalidateVBoxCBCache(CLIPBACKEND *pCtx)
1378{
1379 if (pCtx->pvUnicodeCache != NULL)
1380 {
1381 RTMemFree(pCtx->pvUnicodeCache);
1382 pCtx->pvUnicodeCache = NULL;
1383 }
1384}
1385
1386/**
1387 * Take possession of the X11 clipboard (and middle-button selection).
1388 */
1389static void clipGrabX11CB(CLIPBACKEND *pCtx, uint32_t u32Formats)
1390{
1391 if (XtOwnSelection(pCtx->widget, clipGetAtom(pCtx, "CLIPBOARD"),
1392 CurrentTime, clipXtConvertSelectionProc, NULL, 0))
1393 {
1394 pCtx->vboxFormats = u32Formats;
1395 /* Grab the middle-button paste selection too. */
1396 XtOwnSelection(pCtx->widget, clipGetAtom(pCtx, "PRIMARY"),
1397 CurrentTime, clipXtConvertSelectionProc, NULL, 0);
1398#ifndef TESTCASE
1399 /* Xt suppresses these if we already own the clipboard, so send them
1400 * ourselves. */
1401 XSetSelectionOwner(XtDisplay(pCtx->widget),
1402 clipGetAtom(pCtx, "CLIPBOARD"),
1403 XtWindow(pCtx->widget), CurrentTime);
1404 XSetSelectionOwner(XtDisplay(pCtx->widget),
1405 clipGetAtom(pCtx, "PRIMARY"),
1406 XtWindow(pCtx->widget), CurrentTime);
1407#endif
1408 }
1409}
1410
1411/**
1412 * Worker function for ClipAnnounceFormatToX11 which runs on the
1413 * event thread.
1414 * @param pUserData Pointer to a CLIPNEWVBOXFORMATS structure containing
1415 * information about the VBox formats available and the
1416 * clipboard context data. Must be freed by the worker.
1417 */
1418static void clipNewVBoxFormatsWorker(void *pUserData,
1419 void * /* interval */)
1420{
1421 CLIPNEWVBOXFORMATS *pFormats = (CLIPNEWVBOXFORMATS *)pUserData;
1422 CLIPBACKEND *pCtx = pFormats->pCtx;
1423 uint32_t u32Formats = pFormats->formats;
1424 RTMemFree(pFormats);
1425 LogRelFlowFunc (("u32Formats=%d\n", u32Formats));
1426 clipInvalidateVBoxCBCache(pCtx);
1427 clipGrabX11CB(pCtx, u32Formats);
1428 clipResetX11Formats(pCtx);
1429 LogRelFlowFunc(("returning\n"));
1430}
1431
1432/**
1433 * VBox is taking possession of the shared clipboard.
1434 *
1435 * @param u32Formats Clipboard formats that VBox is offering
1436 * @note X11 backend code
1437 */
1438void ClipAnnounceFormatToX11(CLIPBACKEND *pCtx,
1439 uint32_t u32Formats)
1440{
1441 /*
1442 * Immediately return if we are not connected to the X server.
1443 */
1444 if (!pCtx->fHaveX11)
1445 return;
1446 /* This must be freed by the worker callback */
1447 CLIPNEWVBOXFORMATS *pFormats =
1448 (CLIPNEWVBOXFORMATS *) RTMemAlloc(sizeof(CLIPNEWVBOXFORMATS));
1449 if (pFormats != NULL) /* if it is we will soon have other problems */
1450 {
1451 pFormats->pCtx = pCtx;
1452 pFormats->formats = u32Formats;
1453 clipQueueToEventThread(pCtx, clipNewVBoxFormatsWorker,
1454 (XtPointer) pFormats);
1455 }
1456}
1457
1458/**
1459 * Massage generic Utf16 with CR end-of-lines into the format Windows expects
1460 * and return the result in a RTMemAlloc allocated buffer.
1461 * @returns IPRT status code
1462 * @param pwcSrc The source Utf16
1463 * @param cwcSrc The number of 16bit elements in @a pwcSrc, not counting
1464 * the terminating zero
1465 * @param ppwszDest Where to store the buffer address
1466 * @param pcbDest On success, where to store the number of bytes written.
1467 * Undefined otherwise. Optional
1468 */
1469static int clipUtf16ToWinTxt(RTUTF16 *pwcSrc, size_t cwcSrc,
1470 PRTUTF16 *ppwszDest, uint32_t *pcbDest)
1471{
1472 LogRelFlowFunc(("pwcSrc=%p, cwcSrc=%u, ppwszDest=%p\n", pwcSrc, cwcSrc,
1473 ppwszDest));
1474 AssertPtrReturn(pwcSrc, VERR_INVALID_POINTER);
1475 AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER);
1476 if (pcbDest)
1477 *pcbDest = 0;
1478 PRTUTF16 pwszDest = NULL;
1479 size_t cwcDest;
1480 int rc = vboxClipboardUtf16GetWinSize(pwcSrc, cwcSrc + 1, &cwcDest);
1481 if (RT_SUCCESS(rc))
1482 {
1483 pwszDest = (PRTUTF16) RTMemAlloc(cwcDest * 2);
1484 if (!pwszDest)
1485 rc = VERR_NO_MEMORY;
1486 }
1487 if (RT_SUCCESS(rc))
1488 rc = vboxClipboardUtf16LinToWin(pwcSrc, cwcSrc + 1, pwszDest,
1489 cwcDest);
1490 if (RT_SUCCESS(rc))
1491 {
1492 LogRelFlowFunc (("converted string is %.*ls\n", cwcDest, pwszDest));
1493 *ppwszDest = pwszDest;
1494 if (pcbDest)
1495 *pcbDest = cwcDest * 2;
1496 }
1497 else
1498 RTMemFree(pwszDest);
1499 LogRelFlowFunc(("returning %Rrc\n", rc));
1500 if (pcbDest)
1501 LogRelFlowFunc(("*pcbDest=%u\n", *pcbDest));
1502 return rc;
1503}
1504
1505/**
1506 * Convert Utf-8 text with CR end-of-lines into Utf-16 as Windows expects it
1507 * and return the result in a RTMemAlloc allocated buffer.
1508 * @returns IPRT status code
1509 * @param pcSrc The source Utf-8
1510 * @param cbSrc The size of the source in bytes, not counting the
1511 * terminating zero
1512 * @param ppwszDest Where to store the buffer address
1513 * @param pcbDest On success, where to store the number of bytes written.
1514 * Undefined otherwise. Optional
1515 */
1516static int clipUtf8ToWinTxt(const char *pcSrc, unsigned cbSrc,
1517 PRTUTF16 *ppwszDest, uint32_t *pcbDest)
1518{
1519 LogRelFlowFunc(("pcSrc=%p, cbSrc=%u, ppwszDest=%p\n", pcSrc, cbSrc,
1520 ppwszDest));
1521 AssertPtrReturn(pcSrc, VERR_INVALID_POINTER);
1522 AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER);
1523 if (pcbDest)
1524 *pcbDest = 0;
1525 /* Intermediate conversion to UTF16 */
1526 size_t cwcTmp;
1527 PRTUTF16 pwcTmp = NULL;
1528 int rc = RTStrToUtf16Ex(pcSrc, cbSrc, &pwcTmp, 0, &cwcTmp);
1529 if (RT_SUCCESS(rc))
1530 rc = clipUtf16ToWinTxt(pwcTmp, cwcTmp, ppwszDest, pcbDest);
1531 RTUtf16Free(pwcTmp);
1532 LogRelFlowFunc(("Returning %Rrc\n", rc));
1533 if (pcbDest)
1534 LogRelFlowFunc(("*pcbDest=%u\n", *pcbDest));
1535 return rc;
1536}
1537
1538/**
1539 * Convert Latin-1 text with CR end-of-lines into Utf-16 as Windows expects
1540 * it and return the result in a RTMemAlloc allocated buffer.
1541 * @returns IPRT status code
1542 * @param pcSrc The source text
1543 * @param cbSrc The size of the source in bytes, not counting the
1544 * terminating zero
1545 * @param ppwszDest Where to store the buffer address
1546 * @param pcbDest On success, where to store the number of bytes written.
1547 * Undefined otherwise. Optional
1548 */
1549static int clipLatin1ToWinTxt(char *pcSrc, unsigned cbSrc,
1550 PRTUTF16 *ppwszDest, uint32_t *pcbDest)
1551{
1552 LogRelFlowFunc (("pcSrc=%.*s, cbSrc=%u, ppwszDest=%p\n", cbSrc,
1553 (char *) pcSrc, cbSrc, ppwszDest));
1554 AssertPtrReturn(pcSrc, VERR_INVALID_POINTER);
1555 AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER);
1556 PRTUTF16 pwszDest = NULL;
1557 int rc = VINF_SUCCESS;
1558
1559 /* Calculate the space needed */
1560 unsigned cwcDest = 0;
1561 for (unsigned i = 0; i < cbSrc && pcSrc[i] != '\0'; ++i)
1562 if (pcSrc[i] == LINEFEED)
1563 cwcDest += 2;
1564 else
1565 ++cwcDest;
1566 ++cwcDest; /* Leave space for the terminator */
1567 if (pcbDest)
1568 *pcbDest = cwcDest * 2;
1569 pwszDest = (PRTUTF16) RTMemAlloc(cwcDest * 2);
1570 if (!pwszDest)
1571 rc = VERR_NO_MEMORY;
1572
1573 /* And do the conversion, bearing in mind that Latin-1 expands "naturally"
1574 * to Utf-16. */
1575 if (RT_SUCCESS(rc))
1576 {
1577 for (unsigned i = 0, j = 0; i < cbSrc; ++i, ++j)
1578 if (pcSrc[i] != LINEFEED)
1579 pwszDest[j] = pcSrc[i];
1580 else
1581 {
1582 pwszDest[j] = CARRIAGERETURN;
1583 pwszDest[j + 1] = LINEFEED;
1584 ++j;
1585 }
1586 pwszDest[cwcDest - 1] = '\0'; /* Make sure we are zero-terminated. */
1587 LogRelFlowFunc (("converted text is %.*ls\n", cwcDest, pwszDest));
1588 }
1589 if (RT_SUCCESS(rc))
1590 *ppwszDest = pwszDest;
1591 else
1592 RTMemFree(pwszDest);
1593 LogRelFlowFunc(("Returning %Rrc\n", rc));
1594 if (pcbDest)
1595 LogRelFlowFunc(("*pcbDest=%u\n", *pcbDest));
1596 return rc;
1597}
1598
1599
1600/**
1601* Convert Utf16 text into UTF8 as Windows expects
1602* it and return the result in a RTMemAlloc allocated buffer.
1603* @returns IPRT status code
1604* @param pcSrc The source text
1605* @param cbSrc The size of the source in bytes, not counting the
1606* terminating zero
1607* @param ppwszDest Where to store the buffer address
1608* @param pcbDest On success, where to store the number of bytes written.
1609* Undefined otherwise. Optional
1610*/
1611int clipUTF16ToWinHTML(RTUTF16 *pwcBuf, size_t cb, char **ppszOut, uint32_t *pcOut)
1612{
1613 Assert(pwcBuf);
1614 Assert(cb);
1615 Assert(ppszOut);
1616 Assert(pcOut);
1617
1618 if (cb % 2)
1619 return VERR_INVALID_PARAMETER;
1620 size_t cwc = cb / 2;
1621 size_t i = 0;
1622 RTUTF16 *pwc = pwcBuf;
1623 char *pchRes = NULL;
1624 size_t cRes = 0;
1625 LogRelFlowFunc(("clipUTF16ToWinHTML src= %ls cb=%d i=%i, %x %x\n", pwcBuf, cb, i, ppszOut, pcOut));
1626 while (i < cwc)
1627 {
1628 /* find zero symbol (end of string) */
1629 for (; i < cwc && pwcBuf[i] != 0; i++)
1630 ;
1631 LogRelFlowFunc(("skipped nulls i=%d cwc=%d\n", i, cwc));
1632
1633 /* convert found string */
1634 char *psz = NULL;
1635 size_t cch = 0;
1636 int rc = RTUtf16ToUtf8Ex(pwc, cwc, &psz, pwc - pwcBuf, &cch);
1637 LogRelFlowFunc(("utf16toutf8 src= %ls res=%s i=%i\n", pwc, psz, i));
1638 if (RT_FAILURE(rc))
1639 {
1640 RTMemFree(pchRes);
1641 return rc;
1642 }
1643
1644 /* append new substring */
1645 char *pchNew = (char*)RTMemRealloc(pchRes, cRes + cch + 1);
1646 if (!pchNew)
1647 {
1648 RTMemFree(pchRes);
1649 RTStrFree(psz);
1650 return VERR_NO_MEMORY;
1651 }
1652 pchRes = pchNew;
1653 memcpy(pchRes + cRes, psz, cch + 1);
1654 LogRelFlowFunc(("Temp result res=%s\n", pchRes + cRes));
1655
1656 /* remove temporary buffer */
1657 RTStrFree(psz);
1658 cRes += cch + 1;
1659 /* skip zero symbols */
1660 for (; i < cwc && pwcBuf[i] == 0; i++)
1661 ;
1662 /* remember start of string */
1663 pwc += i;
1664 }
1665 *ppszOut = pchRes;
1666 *pcOut = cRes;
1667
1668 return VINF_SUCCESS;
1669}
1670
1671
1672
1673/** A structure containing information about where to store a request
1674 * for the X11 clipboard contents. */
1675struct _CLIPREADX11CBREQ
1676{
1677 /** The format VBox would like the data in */
1678 uint32_t mFormat;
1679 /** The text format we requested from X11 if we requested text */
1680 CLIPX11FORMAT mTextFormat;
1681 /** The bitmap format we requested from X11 if we requested bitmap */
1682 CLIPX11FORMAT mBitmapFormat;
1683 /** The HTML format we requested from X11 if we requested HTML */
1684 CLIPX11FORMAT mHtmlFormat;
1685 /** The clipboard context this request is associated with */
1686 CLIPBACKEND *mCtx;
1687 /** The request structure passed in from the backend. */
1688 CLIPREADCBREQ *mReq;
1689};
1690
1691typedef struct _CLIPREADX11CBREQ CLIPREADX11CBREQ;
1692
1693/**
1694 * Convert the data obtained from the X11 clipboard to the required format,
1695 * place it in the buffer supplied and signal that data has arrived.
1696 * Convert the text obtained UTF-16LE with Windows EOLs.
1697 * Convert full BMP data to DIB format.
1698 * @note X11 backend code, callback for XtGetSelectionValue, for use when
1699 * the X11 clipboard contains a format we understand.
1700 */
1701static void clipConvertX11CB(void *pClientData, void *pvSrc, unsigned cbSrc)
1702{
1703 CLIPREADX11CBREQ *pReq = (CLIPREADX11CBREQ *) pClientData;
1704 LogRelFlowFunc(("pReq->mFormat=%02X, pReq->mTextFormat=%u, "
1705 "pReq->mBitmapFormat=%u, pReq->mHtmlFormat=%u, pReq->mCtx=%p\n",
1706 pReq->mFormat, pReq->mTextFormat, pReq->mBitmapFormat,
1707 pReq->mHtmlFormat, pReq->mCtx));
1708 AssertPtr(pReq->mCtx);
1709 Assert(pReq->mFormat != 0); /* sanity */
1710 int rc = VINF_SUCCESS;
1711 CLIPBACKEND *pCtx = pReq->mCtx;
1712 void *pvDest = NULL;
1713 uint32_t cbDest = 0;
1714
1715 pCtx->fBusy = false;
1716 if (pCtx->fUpdateNeeded)
1717 clipQueryX11CBFormats(pCtx);
1718 if (pvSrc == NULL)
1719 /* The clipboard selection may have changed before we could get it. */
1720 rc = VERR_NO_DATA;
1721 else if (pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
1722 {
1723 /* In which format is the clipboard data? */
1724 switch (clipRealFormatForX11Format(pReq->mTextFormat))
1725 {
1726 case UTF8:
1727 case TEXT:
1728 {
1729 /* If we are given broken Utf-8, we treat it as Latin1. Is
1730 * this acceptable? */
1731 if (RT_SUCCESS(RTStrValidateEncodingEx((char *)pvSrc, cbSrc,
1732 0)))
1733 rc = clipUtf8ToWinTxt((const char *)pvSrc, cbSrc,
1734 (PRTUTF16 *) &pvDest, &cbDest);
1735 else
1736 rc = clipLatin1ToWinTxt((char *) pvSrc, cbSrc,
1737 (PRTUTF16 *) &pvDest, &cbDest);
1738 break;
1739 }
1740 default:
1741 rc = VERR_INVALID_PARAMETER;
1742 }
1743 }
1744 else if (pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
1745 {
1746 /* In which format is the clipboard data? */
1747 switch (clipRealFormatForX11Format(pReq->mBitmapFormat))
1748 {
1749 case BMP:
1750 {
1751 const void *pDib;
1752 size_t cbDibSize;
1753 rc = vboxClipboardBmpGetDib((const void *)pvSrc, cbSrc,
1754 &pDib, &cbDibSize);
1755 if (RT_SUCCESS(rc))
1756 {
1757 pvDest = RTMemAlloc(cbDibSize);
1758 if (!pvDest)
1759 rc = VERR_NO_MEMORY;
1760 else
1761 {
1762 memcpy(pvDest, pDib, cbDibSize);
1763 cbDest = cbDibSize;
1764 }
1765 }
1766 break;
1767 }
1768 default:
1769 rc = VERR_INVALID_PARAMETER;
1770 }
1771 }
1772 else if(pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_HTML)
1773 {
1774 /* In which format is the clipboard data? */
1775 switch (clipRealFormatForX11Format(pReq->mHtmlFormat))
1776 {
1777 case HTML:
1778 {
1779 /* The common VBox HTML encoding will be - Utf8
1780 * becuase it more general for HTML formats then UTF16
1781 * X11 clipboard returns UTF16, so before sending it we should
1782 * convert it to UTF8
1783 */
1784 pvDest = NULL;
1785 cbDest = 0;
1786 /* Some applications sends data in utf16, some in itf8,
1787 * without indication it in MIME.
1788 * But in case of utf16, at least an OpenOffice adds Byte Order Mark - 0xfeff
1789 * at start of clipboard data
1790 */
1791 if( cbSrc >= sizeof(RTUTF16) && *(PRTUTF16)pvSrc == 0xfeff )
1792 {
1793 LogRelFlowFunc((" \n"));
1794 rc = clipUTF16ToWinHTML((RTUTF16*)pvSrc, cbSrc,
1795 (char**)&pvDest, &cbDest);
1796 }
1797 else
1798 {
1799 pvDest = RTMemAlloc(cbSrc);
1800 if(pvDest)
1801 {
1802 memcpy(pvDest, pvSrc, cbSrc);
1803 cbDest = cbSrc;
1804 }
1805 else
1806 {
1807 rc = VERR_NO_MEMORY;
1808 break;
1809 }
1810 }
1811
1812 LogRelFlowFunc(("Source unicode %ls, cbSrc = %d\n, Byte Order Mark = %hx",
1813 pvSrc, cbSrc, ((PRTUTF16)pvSrc)[0]));
1814 LogRelFlowFunc(("converted to win unicode %s, cbDest = %d, rc = %Rrc\n", pvDest, cbDest, rc));
1815 rc = VINF_SUCCESS;
1816 break;
1817 }
1818 default:
1819 {
1820 rc = VERR_INVALID_PARAMETER;
1821 }
1822 }
1823 }
1824 else
1825 rc = VERR_NOT_IMPLEMENTED;
1826 ClipCompleteDataRequestFromX11(pReq->mCtx->pFrontend, rc, pReq->mReq,
1827 pvDest, cbDest);
1828 RTMemFree(pvDest);
1829 RTMemFree(pReq);
1830 LogRelFlowFunc(("rc=%Rrc\n", rc));
1831}
1832
1833/**
1834 * Convert the data obtained from the X11 clipboard to the required format,
1835 * place it in the buffer supplied and signal that data has arrived.
1836 * Convert the text obtained UTF-16LE with Windows EOLs.
1837 * Convert full BMP data to DIB format.
1838 * @note X11 backend code, callback for XtGetSelectionValue, for use when
1839 * the X11 clipboard contains a format we understand.
1840 */
1841static void cbConvertX11CB(Widget widget, XtPointer pClientData,
1842 Atom * /* selection */, Atom *atomType,
1843 XtPointer pvSrc, long unsigned int *pcLen,
1844 int *piFormat)
1845{
1846 if (*atomType == XT_CONVERT_FAIL) /* Xt timeout */
1847 clipConvertX11CB(pClientData, NULL, 0);
1848 else
1849 clipConvertX11CB(pClientData, pvSrc, (*pcLen) * (*piFormat) / 8);
1850
1851 XtFree((char *)pvSrc);
1852}
1853
1854#ifdef TESTCASE
1855static void testRequestData(CLIPBACKEND* pCtx, CLIPX11FORMAT target,
1856 void *closure);
1857#endif
1858
1859static void getSelectionValue(CLIPBACKEND *pCtx, CLIPX11FORMAT format,
1860 CLIPREADX11CBREQ *pReq)
1861{
1862#ifndef TESTCASE
1863 XtGetSelectionValue(pCtx->widget, clipGetAtom(pCtx, "CLIPBOARD"),
1864 clipAtomForX11Format(pCtx, format),
1865 cbConvertX11CB,
1866 reinterpret_cast<XtPointer>(pReq),
1867 CurrentTime);
1868#else
1869 testRequestData(pCtx, format, (void *)pReq);
1870#endif
1871}
1872
1873/** Worker function for ClipRequestDataFromX11 which runs on the event
1874 * thread. */
1875static void vboxClipboardReadX11Worker(void *pUserData,
1876 void * /* interval */)
1877{
1878 CLIPREADX11CBREQ *pReq = (CLIPREADX11CBREQ *)pUserData;
1879 CLIPBACKEND *pCtx = pReq->mCtx;
1880 LogRelFlowFunc (("pReq->mFormat = %02X\n", pReq->mFormat));
1881
1882 int rc = VINF_SUCCESS;
1883 bool fBusy = pCtx->fBusy;
1884 pCtx->fBusy = true;
1885 if (fBusy)
1886 /* If the clipboard is busy just fend off the request. */
1887 rc = VERR_TRY_AGAIN;
1888 else if (pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
1889 {
1890 /*
1891 * VBox wants to read data in the given format.
1892 */
1893 pReq->mTextFormat = pCtx->X11TextFormat;
1894 if (pReq->mTextFormat == INVALID)
1895 /* VBox thinks we have data and we don't */
1896 rc = VERR_NO_DATA;
1897 else
1898 /* Send out a request for the data to the current clipboard
1899 * owner */
1900 getSelectionValue(pCtx, pCtx->X11TextFormat, pReq);
1901 }
1902 else if (pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
1903 {
1904 pReq->mBitmapFormat = pCtx->X11BitmapFormat;
1905 if (pReq->mBitmapFormat == INVALID)
1906 /* VBox thinks we have data and we don't */
1907 rc = VERR_NO_DATA;
1908 else
1909 /* Send out a request for the data to the current clipboard
1910 * owner */
1911 getSelectionValue(pCtx, pCtx->X11BitmapFormat, pReq);
1912 }
1913 else if(pReq->mFormat == VBOX_SHARED_CLIPBOARD_FMT_HTML)
1914 {
1915 /* Send out a request for the data to the current clipboard
1916 * owner */
1917 pReq->mHtmlFormat = pCtx->X11HTMLFormat;
1918 if(pReq->mHtmlFormat == INVALID)
1919 /* VBox thinks we have data and we don't */
1920 rc = VERR_NO_DATA;
1921 else
1922 /* Send out a request for the data to the current clipboard
1923 * owner */
1924 getSelectionValue(pCtx, pCtx->X11HTMLFormat, pReq);
1925 }
1926 else
1927 rc = VERR_NOT_IMPLEMENTED;
1928 if (RT_FAILURE(rc))
1929 {
1930 /* The clipboard callback was never scheduled, so we must signal
1931 * that the request processing is finished and clean up ourselves. */
1932 ClipCompleteDataRequestFromX11(pReq->mCtx->pFrontend, rc, pReq->mReq,
1933 NULL, 0);
1934 RTMemFree(pReq);
1935 }
1936 LogRelFlowFunc(("status %Rrc\n", rc));
1937}
1938
1939/**
1940 * Called when VBox wants to read the X11 clipboard.
1941 *
1942 * @returns iprt status code
1943 * @param pCtx Context data for the clipboard backend
1944 * @param u32Format The format that the VBox would like to receive the data
1945 * in
1946 * @param pv Where to write the data to
1947 * @param cb The size of the buffer to write the data to
1948 * @param pcbActual Where to write the actual size of the written data
1949 * @note We allocate a request structure which must be freed by the worker
1950 */
1951int ClipRequestDataFromX11(CLIPBACKEND *pCtx, uint32_t u32Format,
1952 CLIPREADCBREQ *pReq)
1953{
1954 /*
1955 * Immediately return if we are not connected to the X server.
1956 */
1957 if (!pCtx->fHaveX11)
1958 return VERR_NO_DATA;
1959 int rc = VINF_SUCCESS;
1960 CLIPREADX11CBREQ *pX11Req;
1961 pX11Req = (CLIPREADX11CBREQ *)RTMemAllocZ(sizeof(*pX11Req));
1962 if (!pX11Req)
1963 rc = VERR_NO_MEMORY;
1964 else
1965 {
1966 pX11Req->mFormat = u32Format;
1967 pX11Req->mCtx = pCtx;
1968 pX11Req->mReq = pReq;
1969 /* We use this to schedule a worker function on the event thread. */
1970 clipQueueToEventThread(pCtx, vboxClipboardReadX11Worker,
1971 (XtPointer) pX11Req);
1972 }
1973 return rc;
1974}
1975
1976#ifdef TESTCASE
1977
1978/** @todo This unit test currently works by emulating the X11 and X toolkit
1979 * APIs to exercise the code, since I didn't want to rewrite the code too much
1980 * when I wrote the tests. However, this makes it rather ugly and hard to
1981 * understand. Anyone doing any work on the code should feel free to
1982 * rewrite the tests and the code to make them cleaner and more readable. */
1983
1984#include <iprt/test.h>
1985#include <poll.h>
1986
1987#define TEST_WIDGET (Widget)0xffff
1988
1989/* For the purpose of the test case, we just execute the procedure to be
1990 * scheduled, as we are running single threaded. */
1991void testQueueToEventThread(void (*proc)(void *, void *),
1992 void *client_data)
1993{
1994 proc(client_data, NULL);
1995}
1996
1997void XtFree(char *ptr)
1998{ RTMemFree((void *) ptr); }
1999
2000/* The data in the simulated VBox clipboard */
2001static int g_vboxDataRC = VINF_SUCCESS;
2002static void *g_vboxDatapv = NULL;
2003static uint32_t g_vboxDatacb = 0;
2004
2005/* Set empty data in the simulated VBox clipboard. */
2006static void clipEmptyVBox(CLIPBACKEND *pCtx, int retval)
2007{
2008 g_vboxDataRC = retval;
2009 RTMemFree(g_vboxDatapv);
2010 g_vboxDatapv = NULL;
2011 g_vboxDatacb = 0;
2012 ClipAnnounceFormatToX11(pCtx, 0);
2013}
2014
2015/* Set the data in the simulated VBox clipboard. */
2016static int clipSetVBoxUtf16(CLIPBACKEND *pCtx, int retval,
2017 const char *pcszData, size_t cb)
2018{
2019 PRTUTF16 pwszData = NULL;
2020 size_t cwData = 0;
2021 int rc = RTStrToUtf16Ex(pcszData, RTSTR_MAX, &pwszData, 0, &cwData);
2022 if (RT_FAILURE(rc))
2023 return rc;
2024 AssertReturn(cb <= cwData * 2 + 2, VERR_BUFFER_OVERFLOW);
2025 void *pv = RTMemDup(pwszData, cb);
2026 RTUtf16Free(pwszData);
2027 if (pv == NULL)
2028 return VERR_NO_MEMORY;
2029 if (g_vboxDatapv)
2030 RTMemFree(g_vboxDatapv);
2031 g_vboxDataRC = retval;
2032 g_vboxDatapv = pv;
2033 g_vboxDatacb = cb;
2034 ClipAnnounceFormatToX11(pCtx,
2035 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
2036 return VINF_SUCCESS;
2037}
2038
2039/* Return the data in the simulated VBox clipboard. */
2040int ClipRequestDataForX11(VBOXCLIPBOARDCONTEXT *pCtx,
2041 uint32_t u32Format, void **ppv,
2042 uint32_t *pcb)
2043{
2044 *pcb = g_vboxDatacb;
2045 if (g_vboxDatapv != NULL)
2046 {
2047 void *pv = RTMemDup(g_vboxDatapv, g_vboxDatacb);
2048 *ppv = pv;
2049 return pv != NULL ? g_vboxDataRC : VERR_NO_MEMORY;
2050 }
2051 *ppv = NULL;
2052 return g_vboxDataRC;
2053}
2054
2055Display *XtDisplay(Widget w)
2056{ return (Display *) 0xffff; }
2057
2058void XtAppSetExitFlag(XtAppContext app_context) {}
2059
2060void XtDestroyWidget(Widget w) {}
2061
2062XtAppContext XtCreateApplicationContext(void) { return (XtAppContext)0xffff; }
2063
2064void XtDestroyApplicationContext(XtAppContext app_context) {}
2065
2066void XtToolkitInitialize(void) {}
2067
2068Boolean XtToolkitThreadInitialize(void) { return True; }
2069
2070Display *XtOpenDisplay(XtAppContext app_context,
2071 _Xconst _XtString display_string,
2072 _Xconst _XtString application_name,
2073 _Xconst _XtString application_class,
2074 XrmOptionDescRec *options, Cardinal num_options,
2075 int *argc, char **argv)
2076{ return (Display *)0xffff; }
2077
2078Widget XtVaAppCreateShell(_Xconst _XtString application_name,
2079 _Xconst _XtString application_class,
2080 WidgetClass widget_class, Display *display, ...)
2081{ return TEST_WIDGET; }
2082
2083void XtSetMappedWhenManaged(Widget widget, _XtBoolean mapped_when_managed) {}
2084
2085void XtRealizeWidget(Widget widget) {}
2086
2087XtInputId XtAppAddInput(XtAppContext app_context, int source,
2088 XtPointer condition, XtInputCallbackProc proc,
2089 XtPointer closure)
2090{ return 0xffff; }
2091
2092/* Atoms we need other than the formats we support. */
2093static const char *g_apszSupAtoms[] =
2094{
2095 "PRIMARY", "CLIPBOARD", "TARGETS", "MULTIPLE", "TIMESTAMP"
2096};
2097
2098/* This just looks for the atom names in a couple of tables and returns an
2099 * index with an offset added. */
2100Atom XInternAtom(Display *, const char *pcsz, int)
2101{
2102 Atom atom = 0;
2103 for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i)
2104 if (!strcmp(pcsz, g_aFormats[i].pcszAtom))
2105 atom = (Atom) (i + 0x1000);
2106 for (unsigned i = 0; i < RT_ELEMENTS(g_apszSupAtoms); ++i)
2107 if (!strcmp(pcsz, g_apszSupAtoms[i]))
2108 atom = (Atom) (i + 0x2000);
2109 Assert(atom); /* Have we missed any atoms? */
2110 return atom;
2111}
2112
2113/* Take a request for the targets we are currently offering. */
2114static CLIPX11FORMAT g_selTargets[10] = { 0 };
2115static size_t g_cTargets = 0;
2116
2117void testRequestTargets(CLIPBACKEND* pCtx)
2118{
2119 clipUpdateX11Targets(pCtx, g_selTargets, g_cTargets);
2120}
2121
2122/* The current values of the X selection, which will be returned to the
2123 * XtGetSelectionValue callback. */
2124static Atom g_selType = 0;
2125static const void *g_pSelData = NULL;
2126static unsigned long g_cSelData = 0;
2127static int g_selFormat = 0;
2128
2129void testRequestData(CLIPBACKEND* pCtx, CLIPX11FORMAT target, void *closure)
2130{
2131 unsigned long count = 0;
2132 int format = 0;
2133 if (target != g_selTargets[0])
2134 {
2135 clipConvertX11CB(closure, NULL, 0); /* Could not convert to target. */
2136 return;
2137 }
2138 void *pValue = NULL;
2139 pValue = g_pSelData ? RTMemDup(g_pSelData, g_cSelData) : NULL;
2140 count = g_pSelData ? g_cSelData : 0;
2141 format = g_selFormat;
2142 if (!pValue)
2143 {
2144 count = 0;
2145 format = 0;
2146 }
2147 clipConvertX11CB(closure, pValue, count * format / 8);
2148}
2149
2150/* The formats currently on offer from X11 via the shared clipboard */
2151static uint32_t g_fX11Formats = 0;
2152
2153void ClipReportX11Formats(VBOXCLIPBOARDCONTEXT* pCtx,
2154 uint32_t u32Formats)
2155{
2156 g_fX11Formats = u32Formats;
2157}
2158
2159static uint32_t clipQueryFormats()
2160{
2161 return g_fX11Formats;
2162}
2163
2164static void clipInvalidateFormats()
2165{
2166 g_fX11Formats = ~0;
2167}
2168
2169/* Does our clipboard code currently own the selection? */
2170static bool g_ownsSel = false;
2171/* The procedure that is called when we should convert the selection to a
2172 * given format. */
2173static XtConvertSelectionProc g_pfnSelConvert = NULL;
2174/* The procedure which is called when we lose the selection. */
2175static XtLoseSelectionProc g_pfnSelLose = NULL;
2176/* The procedure which is called when the selection transfer has completed. */
2177static XtSelectionDoneProc g_pfnSelDone = NULL;
2178
2179Boolean XtOwnSelection(Widget widget, Atom selection, Time time,
2180 XtConvertSelectionProc convert,
2181 XtLoseSelectionProc lose,
2182 XtSelectionDoneProc done)
2183{
2184 if (selection != XInternAtom(NULL, "CLIPBOARD", 0))
2185 return True; /* We don't really care about this. */
2186 g_ownsSel = true; /* Always succeed. */
2187 g_pfnSelConvert = convert;
2188 g_pfnSelLose = lose;
2189 g_pfnSelDone = done;
2190 return True;
2191}
2192
2193void XtDisownSelection(Widget widget, Atom selection, Time time)
2194{
2195 g_ownsSel = false;
2196 g_pfnSelConvert = NULL;
2197 g_pfnSelLose = NULL;
2198 g_pfnSelDone = NULL;
2199}
2200
2201/* Request the shared clipboard to convert its data to a given format. */
2202static bool clipConvertSelection(const char *pcszTarget, Atom *type,
2203 XtPointer *value, unsigned long *length,
2204 int *format)
2205{
2206 Atom target = XInternAtom(NULL, pcszTarget, 0);
2207 if (target == 0)
2208 return false;
2209 /* Initialise all return values in case we make a quick exit. */
2210 *type = XA_STRING;
2211 *value = NULL;
2212 *length = 0;
2213 *format = 0;
2214 if (!g_ownsSel)
2215 return false;
2216 if (!g_pfnSelConvert)
2217 return false;
2218 Atom clipAtom = XInternAtom(NULL, "CLIPBOARD", 0);
2219 if (!g_pfnSelConvert(TEST_WIDGET, &clipAtom, &target, type,
2220 value, length, format))
2221 return false;
2222 if (g_pfnSelDone)
2223 g_pfnSelDone(TEST_WIDGET, &clipAtom, &target);
2224 return true;
2225}
2226
2227/* Set the current X selection data */
2228static void clipSetSelectionValues(const char *pcszTarget, Atom type,
2229 const void *data,
2230 unsigned long count, int format)
2231{
2232 Atom clipAtom = XInternAtom(NULL, "CLIPBOARD", 0);
2233 g_selTargets[0] = clipFindX11FormatByAtomText(pcszTarget);
2234 g_cTargets = 1;
2235 g_selType = type;
2236 g_pSelData = data;
2237 g_cSelData = count;
2238 g_selFormat = format;
2239 if (g_pfnSelLose)
2240 g_pfnSelLose(TEST_WIDGET, &clipAtom);
2241 g_ownsSel = false;
2242}
2243
2244static void clipSendTargetUpdate(CLIPBACKEND *pCtx)
2245{
2246 clipQueryX11CBFormats(pCtx);
2247}
2248
2249/* Configure if and how the X11 TARGETS clipboard target will fail */
2250static void clipSetTargetsFailure(void)
2251{
2252 g_cTargets = 0;
2253}
2254
2255char *XtMalloc(Cardinal size) { return (char *) RTMemAlloc(size); }
2256
2257char *XGetAtomName(Display *display, Atom atom)
2258{
2259 AssertReturn((unsigned)atom < RT_ELEMENTS(g_aFormats) + 1, NULL);
2260 const char *pcszName = NULL;
2261 if (atom < 0x1000)
2262 return NULL;
2263 else if (0x1000 <= atom && atom < 0x2000)
2264 {
2265 unsigned index = atom - 0x1000;
2266 AssertReturn(index < RT_ELEMENTS(g_aFormats), NULL);
2267 pcszName = g_aFormats[index].pcszAtom;
2268 }
2269 else
2270 {
2271 unsigned index = atom - 0x2000;
2272 AssertReturn(index < RT_ELEMENTS(g_apszSupAtoms), NULL);
2273 pcszName = g_apszSupAtoms[index];
2274 }
2275 return (char *)RTMemDup(pcszName, sizeof(pcszName) + 1);
2276}
2277
2278int XFree(void *data)
2279{
2280 RTMemFree(data);
2281 return 0;
2282}
2283
2284void XFreeStringList(char **list)
2285{
2286 if (list)
2287 RTMemFree(*list);
2288 RTMemFree(list);
2289}
2290
2291#define MAX_BUF_SIZE 256
2292
2293static int g_completedRC = VINF_SUCCESS;
2294static int g_completedCB = 0;
2295static CLIPREADCBREQ *g_completedReq = NULL;
2296static char g_completedBuf[MAX_BUF_SIZE];
2297
2298void ClipCompleteDataRequestFromX11(VBOXCLIPBOARDCONTEXT *pCtx, int rc,
2299 CLIPREADCBREQ *pReq, void *pv,
2300 uint32_t cb)
2301{
2302 if (cb <= MAX_BUF_SIZE)
2303 {
2304 g_completedRC = rc;
2305 memcpy(g_completedBuf, pv, cb);
2306 }
2307 else
2308 g_completedRC = VERR_BUFFER_OVERFLOW;
2309 g_completedCB = cb;
2310 g_completedReq = pReq;
2311}
2312
2313static void clipGetCompletedRequest(int *prc, char ** ppc, uint32_t *pcb,
2314 CLIPREADCBREQ **ppReq)
2315{
2316 *prc = g_completedRC;
2317 *ppc = g_completedBuf;
2318 *pcb = g_completedCB;
2319 *ppReq = g_completedReq;
2320}
2321#ifdef RT_OS_SOLARIS_10
2322char XtStrings [] = "";
2323_WidgetClassRec* applicationShellWidgetClass;
2324char XtShellStrings [] = "";
2325int XmbTextPropertyToTextList(
2326 Display* /* display */,
2327 XTextProperty* /* text_prop */,
2328 char*** /* list_return */,
2329 int* /* count_return */
2330)
2331{
2332 return 0;
2333}
2334#else
2335const char XtStrings [] = "";
2336_WidgetClassRec* applicationShellWidgetClass;
2337const char XtShellStrings [] = "";
2338#endif
2339
2340static void testStringFromX11(RTTEST hTest, CLIPBACKEND *pCtx,
2341 const char *pcszExp, int rcExp)
2342{
2343 bool retval = true;
2344 clipSendTargetUpdate(pCtx);
2345 if (clipQueryFormats() != VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
2346 RTTestFailed(hTest, "Wrong targets reported: %02X\n",
2347 clipQueryFormats());
2348 else
2349 {
2350 char *pc;
2351 CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL;
2352 ClipRequestDataFromX11(pCtx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
2353 pReq);
2354 int rc = VINF_SUCCESS;
2355 uint32_t cbActual = 0;
2356 clipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet);
2357 if (rc != rcExp)
2358 RTTestFailed(hTest, "Wrong return code, expected %Rrc, got %Rrc\n",
2359 rcExp, rc);
2360 else if (pReqRet != pReq)
2361 RTTestFailed(hTest, "Wrong returned request data, expected %p, got %p\n",
2362 pReq, pReqRet);
2363 else if (RT_FAILURE(rcExp))
2364 retval = true;
2365 else
2366 {
2367 RTUTF16 wcExp[MAX_BUF_SIZE / 2];
2368 RTUTF16 *pwcExp = wcExp;
2369 size_t cwc = 0;
2370 rc = RTStrToUtf16Ex(pcszExp, RTSTR_MAX, &pwcExp,
2371 RT_ELEMENTS(wcExp), &cwc);
2372 size_t cbExp = cwc * 2 + 2;
2373 AssertRC(rc);
2374 if (RT_SUCCESS(rc))
2375 {
2376 if (cbActual != cbExp)
2377 {
2378 RTTestFailed(hTest, "Returned string is the wrong size, string \"%.*ls\", size %u, expected \"%s\", size %u\n",
2379 RT_MIN(MAX_BUF_SIZE, cbActual), pc, cbActual,
2380 pcszExp, cbExp);
2381 }
2382 else
2383 {
2384 if (memcmp(pc, wcExp, cbExp) == 0)
2385 retval = true;
2386 else
2387 RTTestFailed(hTest, "Returned string \"%.*ls\" does not match expected string \"%s\"\n",
2388 MAX_BUF_SIZE, pc, pcszExp);
2389 }
2390 }
2391 }
2392 }
2393 if (!retval)
2394 RTTestFailureDetails(hTest, "Expected: string \"%s\", rc %Rrc\n",
2395 pcszExp, rcExp);
2396}
2397
2398static void testLatin1FromX11(RTTEST hTest, CLIPBACKEND *pCtx,
2399 const char *pcszExp, int rcExp)
2400{
2401 bool retval = false;
2402 clipSendTargetUpdate(pCtx);
2403 if (clipQueryFormats() != VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
2404 RTTestFailed(hTest, "Wrong targets reported: %02X\n",
2405 clipQueryFormats());
2406 else
2407 {
2408 char *pc;
2409 CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL;
2410 ClipRequestDataFromX11(pCtx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
2411 pReq);
2412 int rc = VINF_SUCCESS;
2413 uint32_t cbActual = 0;
2414 clipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet);
2415 if (rc != rcExp)
2416 RTTestFailed(hTest, "Wrong return code, expected %Rrc, got %Rrc\n",
2417 rcExp, rc);
2418 else if (pReqRet != pReq)
2419 RTTestFailed(hTest, "Wrong returned request data, expected %p, got %p\n",
2420 pReq, pReqRet);
2421 else if (RT_FAILURE(rcExp))
2422 retval = true;
2423 else
2424 {
2425 RTUTF16 wcExp[MAX_BUF_SIZE / 2];
2426 RTUTF16 *pwcExp = wcExp;
2427 size_t cwc;
2428 for (cwc = 0; cwc == 0 || pcszExp[cwc - 1] != '\0'; ++cwc)
2429 wcExp[cwc] = pcszExp[cwc];
2430 size_t cbExp = cwc * 2;
2431 if (cbActual != cbExp)
2432 {
2433 RTTestFailed(hTest, "Returned string is the wrong size, string \"%.*ls\", size %u, expected \"%s\", size %u\n",
2434 RT_MIN(MAX_BUF_SIZE, cbActual), pc, cbActual,
2435 pcszExp, cbExp);
2436 }
2437 else
2438 {
2439 if (memcmp(pc, wcExp, cbExp) == 0)
2440 retval = true;
2441 else
2442 RTTestFailed(hTest, "Returned string \"%.*ls\" does not match expected string \"%s\"\n",
2443 MAX_BUF_SIZE, pc, pcszExp);
2444 }
2445 }
2446 }
2447 if (!retval)
2448 RTTestFailureDetails(hTest, "Expected: string \"%s\", rc %Rrc\n",
2449 pcszExp, rcExp);
2450}
2451
2452static void testStringFromVBox(RTTEST hTest, CLIPBACKEND *pCtx,
2453 const char *pcszTarget, Atom typeExp,
2454 const char *valueExp)
2455{
2456 bool retval = false;
2457 Atom type;
2458 XtPointer value = NULL;
2459 unsigned long length;
2460 int format;
2461 size_t lenExp = strlen(valueExp);
2462 if (clipConvertSelection(pcszTarget, &type, &value, &length, &format))
2463 {
2464 if ( type != typeExp
2465 || length != lenExp
2466 || format != 8
2467 || memcmp((const void *) value, (const void *)valueExp,
2468 lenExp))
2469 {
2470 RTTestFailed(hTest, "Bad data: type %d, (expected %d), length %u, (%u), format %d (%d), value \"%.*s\" (\"%.*s\")\n",
2471 type, typeExp, length, lenExp, format, 8,
2472 RT_MIN(length, 20), value, RT_MIN(lenExp, 20), valueExp);
2473 }
2474 else
2475 retval = true;
2476 }
2477 else
2478 RTTestFailed(hTest, "Conversion failed\n");
2479 XtFree((char *)value);
2480 if (!retval)
2481 RTTestFailureDetails(hTest, "Conversion to %s, expected \"%s\"\n",
2482 pcszTarget, valueExp);
2483}
2484
2485static void testNoX11(CLIPBACKEND *pCtx, const char *pcszTestCtx)
2486{
2487 CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL;
2488 int rc = ClipRequestDataFromX11(pCtx,
2489 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
2490 pReq);
2491 RTTESTI_CHECK_MSG(rc == VERR_NO_DATA, ("context: %s\n", pcszTestCtx));
2492}
2493
2494static void testStringFromVBoxFailed(RTTEST hTest, CLIPBACKEND *pCtx,
2495 const char *pcszTarget)
2496{
2497 bool retval = false;
2498 Atom type;
2499 XtPointer value = NULL;
2500 unsigned long length;
2501 int format;
2502 RTTEST_CHECK_MSG(hTest, !clipConvertSelection(pcszTarget, &type, &value,
2503 &length, &format),
2504 (hTest, "Conversion to target %s, should have failed but didn't, returned type %d, length %u, format %d, value \"%.*s\"\n",
2505 pcszTarget, type, length, format, RT_MIN(length, 20),
2506 value));
2507 XtFree((char *)value);
2508}
2509
2510static void testNoSelectionOwnership(CLIPBACKEND *pCtx,
2511 const char *pcszTestCtx)
2512{
2513 RTTESTI_CHECK_MSG(!g_ownsSel, ("context: %s\n", pcszTestCtx));
2514}
2515
2516int main()
2517{
2518 /*
2519 * Init the runtime, test and say hello.
2520 */
2521 RTTEST hTest;
2522 int rc = RTTestInitAndCreate("tstClipboardX11", &hTest);
2523 if (rc)
2524 return rc;
2525 RTTestBanner(hTest);
2526
2527 /*
2528 * Run the test.
2529 */
2530 CLIPBACKEND *pCtx = ClipConstructX11(NULL, false);
2531 char *pc;
2532 uint32_t cbActual;
2533 CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL;
2534 rc = ClipStartX11(pCtx);
2535 AssertRCReturn(rc, 1);
2536
2537 /*** Utf-8 from X11 ***/
2538 RTTestSub(hTest, "reading Utf-8 from X11");
2539 /* Simple test */
2540 clipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world",
2541 sizeof("hello world"), 8);
2542 testStringFromX11(hTest, pCtx, "hello world", VINF_SUCCESS);
2543 /* With an embedded carriage return */
2544 clipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING,
2545 "hello\nworld", sizeof("hello\nworld"), 8);
2546 testStringFromX11(hTest, pCtx, "hello\r\nworld", VINF_SUCCESS);
2547 /* With an embedded CRLF */
2548 clipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING,
2549 "hello\r\nworld", sizeof("hello\r\nworld"), 8);
2550 testStringFromX11(hTest, pCtx, "hello\r\r\nworld", VINF_SUCCESS);
2551 /* With an embedded LFCR */
2552 clipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING,
2553 "hello\n\rworld", sizeof("hello\n\rworld"), 8);
2554 testStringFromX11(hTest, pCtx, "hello\r\n\rworld", VINF_SUCCESS);
2555 /* An empty string */
2556 clipSetSelectionValues("text/plain;charset=utf-8", XA_STRING, "",
2557 sizeof(""), 8);
2558 testStringFromX11(hTest, pCtx, "", VINF_SUCCESS);
2559 /* With an embedded Utf-8 character. */
2560 clipSetSelectionValues("STRING", XA_STRING,
2561 "100\xE2\x82\xAC" /* 100 Euro */,
2562 sizeof("100\xE2\x82\xAC"), 8);
2563 testStringFromX11(hTest, pCtx, "100\xE2\x82\xAC", VINF_SUCCESS);
2564 /* A non-zero-terminated string */
2565 clipSetSelectionValues("TEXT", XA_STRING,
2566 "hello world", sizeof("hello world") - 1, 8);
2567 testStringFromX11(hTest, pCtx, "hello world", VINF_SUCCESS);
2568
2569 /*** Latin1 from X11 ***/
2570 RTTestSub(hTest, "reading Latin1 from X11");
2571 /* Simple test */
2572 clipSetSelectionValues("STRING", XA_STRING, "Georges Dupr\xEA",
2573 sizeof("Georges Dupr\xEA"), 8);
2574 testLatin1FromX11(hTest, pCtx, "Georges Dupr\xEA", VINF_SUCCESS);
2575 /* With an embedded carriage return */
2576 clipSetSelectionValues("TEXT", XA_STRING, "Georges\nDupr\xEA",
2577 sizeof("Georges\nDupr\xEA"), 8);
2578 testLatin1FromX11(hTest, pCtx, "Georges\r\nDupr\xEA", VINF_SUCCESS);
2579 /* With an embedded CRLF */
2580 clipSetSelectionValues("TEXT", XA_STRING, "Georges\r\nDupr\xEA",
2581 sizeof("Georges\r\nDupr\xEA"), 8);
2582 testLatin1FromX11(hTest, pCtx, "Georges\r\r\nDupr\xEA", VINF_SUCCESS);
2583 /* With an embedded LFCR */
2584 clipSetSelectionValues("TEXT", XA_STRING, "Georges\n\rDupr\xEA",
2585 sizeof("Georges\n\rDupr\xEA"), 8);
2586 testLatin1FromX11(hTest, pCtx, "Georges\r\n\rDupr\xEA", VINF_SUCCESS);
2587 /* A non-zero-terminated string */
2588 clipSetSelectionValues("text/plain", XA_STRING,
2589 "Georges Dupr\xEA!",
2590 sizeof("Georges Dupr\xEA!") - 1, 8);
2591 testLatin1FromX11(hTest, pCtx, "Georges Dupr\xEA!", VINF_SUCCESS);
2592
2593 /*** Unknown X11 format ***/
2594 RTTestSub(hTest, "handling of an unknown X11 format");
2595 clipInvalidateFormats();
2596 clipSetSelectionValues("CLIPBOARD", XA_STRING, "Test",
2597 sizeof("Test"), 8);
2598 clipSendTargetUpdate(pCtx);
2599 RTTEST_CHECK_MSG(hTest, clipQueryFormats() == 0,
2600 (hTest, "Failed to send a format update notification\n"));
2601
2602 /*** Timeout from X11 ***/
2603 RTTestSub(hTest, "X11 timeout");
2604 clipSetSelectionValues("UTF8_STRING", XT_CONVERT_FAIL, NULL,0, 8);
2605 testStringFromX11(hTest, pCtx, "", VERR_NO_DATA);
2606
2607 /*** No data in X11 clipboard ***/
2608 RTTestSub(hTest, "a data request from an empty X11 clipboard");
2609 clipSetSelectionValues("UTF8_STRING", XA_STRING, NULL,
2610 0, 8);
2611 ClipRequestDataFromX11(pCtx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT,
2612 pReq);
2613 clipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet);
2614 RTTEST_CHECK_MSG(hTest, rc == VERR_NO_DATA,
2615 (hTest, "Returned %Rrc instead of VERR_NO_DATA\n",
2616 rc));
2617 RTTEST_CHECK_MSG(hTest, pReqRet == pReq,
2618 (hTest, "Wrong returned request data, expected %p, got %p\n",
2619 pReq, pReqRet));
2620
2621 /*** Ensure that VBox is notified when we return the CB to X11 ***/
2622 RTTestSub(hTest, "notification of switch to X11 clipboard");
2623 clipInvalidateFormats();
2624 clipReportEmptyX11CB(pCtx);
2625 RTTEST_CHECK_MSG(hTest, clipQueryFormats() == 0,
2626 (hTest, "Failed to send a format update (release) notification\n"));
2627
2628 /*** request for an invalid VBox format from X11 ***/
2629 RTTestSub(hTest, "a request for an invalid VBox format from X11");
2630 ClipRequestDataFromX11(pCtx, 0xffff, pReq);
2631 clipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet);
2632 RTTEST_CHECK_MSG(hTest, rc == VERR_NOT_IMPLEMENTED,
2633 (hTest, "Returned %Rrc instead of VERR_NOT_IMPLEMENTED\n",
2634 rc));
2635 RTTEST_CHECK_MSG(hTest, pReqRet == pReq,
2636 (hTest, "Wrong returned request data, expected %p, got %p\n",
2637 pReq, pReqRet));
2638
2639 /*** Targets failure from X11 ***/
2640 RTTestSub(hTest, "X11 targets conversion failure");
2641 clipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world",
2642 sizeof("hello world"), 8);
2643 clipSetTargetsFailure();
2644 Atom atom = XA_STRING;
2645 long unsigned int cLen = 0;
2646 int format = 8;
2647 clipConvertX11Targets(NULL, (XtPointer) pCtx, NULL, &atom, NULL, &cLen,
2648 &format);
2649 RTTEST_CHECK_MSG(hTest, clipQueryFormats() == 0,
2650 (hTest, "Wrong targets reported: %02X\n",
2651 clipQueryFormats()));
2652
2653 /*** X11 text format conversion ***/
2654 RTTestSub(hTest, "handling of X11 selection targets");
2655 RTTEST_CHECK_MSG(hTest, clipTestTextFormatConversion(pCtx),
2656 (hTest, "failed to select the right X11 text formats\n"));
2657
2658 /*** Utf-8 from VBox ***/
2659 RTTestSub(hTest, "reading Utf-8 from VBox");
2660 /* Simple test */
2661 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello world",
2662 sizeof("hello world") * 2);
2663 testStringFromVBox(hTest, pCtx, "UTF8_STRING",
2664 clipGetAtom(pCtx, "UTF8_STRING"), "hello world");
2665 /* With an embedded carriage return */
2666 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello\r\nworld",
2667 sizeof("hello\r\nworld") * 2);
2668 testStringFromVBox(hTest, pCtx, "text/plain;charset=UTF-8",
2669 clipGetAtom(pCtx, "text/plain;charset=UTF-8"),
2670 "hello\nworld");
2671 /* With an embedded CRCRLF */
2672 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello\r\r\nworld",
2673 sizeof("hello\r\r\nworld") * 2);
2674 testStringFromVBox(hTest, pCtx, "text/plain;charset=UTF-8",
2675 clipGetAtom(pCtx, "text/plain;charset=UTF-8"),
2676 "hello\r\nworld");
2677 /* With an embedded CRLFCR */
2678 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello\r\n\rworld",
2679 sizeof("hello\r\n\rworld") * 2);
2680 testStringFromVBox(hTest, pCtx, "text/plain;charset=UTF-8",
2681 clipGetAtom(pCtx, "text/plain;charset=UTF-8"),
2682 "hello\n\rworld");
2683 /* An empty string */
2684 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "", 2);
2685 testStringFromVBox(hTest, pCtx, "text/plain;charset=utf-8",
2686 clipGetAtom(pCtx, "text/plain;charset=utf-8"), "");
2687 /* With an embedded Utf-8 character. */
2688 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "100\xE2\x82\xAC" /* 100 Euro */,
2689 10);
2690 testStringFromVBox(hTest, pCtx, "STRING",
2691 clipGetAtom(pCtx, "STRING"), "100\xE2\x82\xAC");
2692 /* A non-zero-terminated string */
2693 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello world",
2694 sizeof("hello world") * 2 - 2);
2695 testStringFromVBox(hTest, pCtx, "TEXT", clipGetAtom(pCtx, "TEXT"),
2696 "hello world");
2697
2698 /*** Timeout from VBox ***/
2699 RTTestSub(hTest, "reading from VBox with timeout");
2700 clipEmptyVBox(pCtx, VERR_TIMEOUT);
2701 testStringFromVBoxFailed(hTest, pCtx, "UTF8_STRING");
2702
2703 /*** No data in VBox clipboard ***/
2704 RTTestSub(hTest, "an empty VBox clipboard");
2705 clipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8);
2706 clipEmptyVBox(pCtx, VINF_SUCCESS);
2707 RTTEST_CHECK_MSG(hTest, g_ownsSel,
2708 (hTest, "VBox grabbed the clipboard with no data and we ignored it\n"));
2709 testStringFromVBoxFailed(hTest, pCtx, "UTF8_STRING");
2710
2711 /*** An unknown VBox format ***/
2712 RTTestSub(hTest, "reading an unknown VBox format");
2713 clipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8);
2714 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "", 2);
2715 ClipAnnounceFormatToX11(pCtx, 0xa0000);
2716 RTTEST_CHECK_MSG(hTest, g_ownsSel,
2717 (hTest, "VBox grabbed the clipboard with unknown data and we ignored it\n"));
2718 testStringFromVBoxFailed(hTest, pCtx, "UTF8_STRING");
2719 rc = ClipStopX11(pCtx);
2720 AssertRCReturn(rc, 1);
2721 ClipDestructX11(pCtx);
2722
2723 /*** Headless clipboard tests ***/
2724
2725 pCtx = ClipConstructX11(NULL, true);
2726 rc = ClipStartX11(pCtx);
2727 AssertRCReturn(rc, 1);
2728
2729 /*** Read from X11 ***/
2730 RTTestSub(hTest, "reading from X11, headless clipboard");
2731 /* Simple test */
2732 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "",
2733 sizeof("") * 2);
2734 clipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world",
2735 sizeof("hello world"), 8);
2736 testNoX11(pCtx, "reading from X11, headless clipboard");
2737
2738 /*** Read from VBox ***/
2739 RTTestSub(hTest, "reading from VBox, headless clipboard");
2740 /* Simple test */
2741 clipEmptyVBox(pCtx, VERR_WRONG_ORDER);
2742 clipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8);
2743 clipSetVBoxUtf16(pCtx, VINF_SUCCESS, "hello world",
2744 sizeof("hello world") * 2);
2745 testNoSelectionOwnership(pCtx, "reading from VBox, headless clipboard");
2746
2747 rc = ClipStopX11(pCtx);
2748 AssertRCReturn(rc, 1);
2749 ClipDestructX11(pCtx);
2750
2751 return RTTestSummaryAndDestroy(hTest);
2752}
2753
2754#endif
2755
2756#ifdef SMOKETEST
2757
2758/* This is a simple test case that just starts a copy of the X11 clipboard
2759 * backend, checks the X11 clipboard and exits. If ever needed I will add an
2760 * interactive mode in which the user can read and copy to the clipboard from
2761 * the command line. */
2762
2763#include <iprt/env.h>
2764#include <iprt/test.h>
2765
2766int ClipRequestDataForX11(VBOXCLIPBOARDCONTEXT *pCtx,
2767 uint32_t u32Format, void **ppv,
2768 uint32_t *pcb)
2769{
2770 return VERR_NO_DATA;
2771}
2772
2773void ClipReportX11Formats(VBOXCLIPBOARDCONTEXT *pCtx,
2774 uint32_t u32Formats)
2775{}
2776
2777void ClipCompleteDataRequestFromX11(VBOXCLIPBOARDCONTEXT *pCtx, int rc,
2778 CLIPREADCBREQ *pReq, void *pv,
2779 uint32_t cb)
2780{}
2781
2782int main()
2783{
2784 /*
2785 * Init the runtime, test and say hello.
2786 */
2787 RTTEST hTest;
2788 int rc = RTTestInitAndCreate("tstClipboardX11Smoke", &hTest);
2789 if (rc)
2790 return rc;
2791 RTTestBanner(hTest);
2792
2793 /*
2794 * Run the test.
2795 */
2796 rc = VINF_SUCCESS;
2797 /* We can't test anything without an X session, so just return success
2798 * in that case. */
2799 if (!RTEnvExist("DISPLAY"))
2800 {
2801 RTTestPrintf(hTest, RTTESTLVL_INFO,
2802 "X11 not available, not running test\n");
2803 return RTTestSummaryAndDestroy(hTest);
2804 }
2805 CLIPBACKEND *pCtx = ClipConstructX11(NULL, false);
2806 AssertReturn(pCtx, 1);
2807 rc = ClipStartX11(pCtx);
2808 AssertRCReturn(rc, 1);
2809 /* Give the clipboard time to synchronise. */
2810 RTThreadSleep(500);
2811 rc = ClipStopX11(pCtx);
2812 AssertRCReturn(rc, 1);
2813 ClipDestructX11(pCtx);
2814 return RTTestSummaryAndDestroy(hTest);
2815}
2816
2817#endif /* SMOKETEST defined */
2818
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette