VirtualBox

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

Last change on this file since 47512 was 46383, checked in by vboxsync, 12 years ago

GuestHost/SharedClipboard: more X11 unit test clean-up.

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