VirtualBox

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

Last change on this file since 74878 was 69668, checked in by vboxsync, 7 years ago

GuestHost/SharedClipboard/x11: schedule synchronisation mechanism removal.
bugref:9041: clipboard sharing on macOS host is brittle, Guest->Host direction

The GuestHost part of the X11 shared clipboard code contains a synchronisation
mechanism in the form of a flag which is set before an X11 clipboard request
is triggered and cleared when it completes. This is because the Xt toolkit,
which we use for the shared clipboard, does not support multiple clipboard
operations at once. However, we run all X11 clipboard code on a single thread,
and any function which sets the flag also waits for the operation to complete,
so the flag should not be necessary at all. What is more, if there is any
valid use for it it is far from clear that it would actually work as intended.
This change schedules it for removal after 5.2 using conditional compilation.

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