VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedClipboard/x11.cpp@ 9018

Last change on this file since 9018 was 8951, checked in by vboxsync, 17 years ago

HostServices/SharedClipboard and Additions/x11: support middle-button pasting (but not copying) in X11 guests and hosts

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 56.8 KB
Line 
1/** @file
2 *
3 * Shared Clipboard:
4 * Linux host.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#define USE_UTF16
24#define USE_UTF8
25#define USE_CTEXT
26
27#include <vector>
28
29#include <VBox/HostServices/VBoxClipboardSvc.h>
30
31#include <iprt/alloc.h>
32#include <iprt/asm.h> /* For atomic operations */
33#include <iprt/assert.h>
34#include <iprt/mem.h>
35#include <iprt/string.h>
36#include <iprt/thread.h>
37#include <iprt/process.h>
38#include <iprt/semaphore.h>
39#include <string.h>
40#include <stdio.h>
41#include <stdint.h>
42
43#include "VBoxClipboard.h"
44#include "clipboard-helper.h"
45
46#include <X11/Xlib.h>
47#include <X11/Xatom.h>
48#include <X11/Intrinsic.h>
49#include <X11/Shell.h>
50#include <X11/Xproto.h>
51#include <X11/StringDefs.h>
52
53/** The different clipboard formats which we support. */
54enum g_eClipboardFormats
55{
56 INVALID = 0,
57 TARGETS,
58 CTEXT,
59 UTF8,
60 UTF16
61};
62
63/** The X11 clipboard uses several names for the same format. This structure maps an X11
64 name to a format. */
65typedef struct {
66 Atom atom;
67 g_eClipboardFormats format;
68 unsigned guestFormat;
69} VBOXCLIPBOARDFORMAT;
70
71/** Does the host or the guest currently own the clipboard? */
72enum g_eClipboardOwner { NONE = 0, HOST, GUEST };
73
74typedef struct {
75 /** BMP file type marker - must always contain 'BM' */
76 uint16_t bfType;
77 /** The size of the BMP file in bytes (the MS docs say that this is not reliable) */
78 uint32_t bfSize;
79 /** Reserved, must always be zero */
80 uint16_t bfReserved1;
81 /** Reserved, must always be zero */
82 uint16_t bfReserved2;
83 /** Offset from the beginning of this header to the actual image bits */
84} VBOXBITMAPFILEHEADER;
85
86/** Global clipboard context information */
87struct _VBOXCLIPBOARDCONTEXT
88{
89 /** The X Toolkit application context structure */
90 XtAppContext appContext;
91
92 /** We have a separate thread to wait for Window and Clipboard events */
93 RTTHREAD thread;
94 /** The X Toolkit widget which we use as our clipboard client. It is never made visible. */
95 Widget widget;
96
97 /** X11 atom refering to the clipboard: CLIPBOARD */
98 Atom atomClipboard;
99 /** X11 atom refering to the selection: PRIMARY */
100 Atom atomPrimary;
101 /** X11 atom refering to the clipboard targets: TARGETS */
102 Atom atomTargets;
103 /** X11 atom refering to the clipboard multiple target: MULTIPLE */
104 Atom atomMultiple;
105 /** X11 atom refering to the clipboard timestamp target: TIMESTAMP */
106 Atom atomTimestamp;
107 /** X11 atom refering to the clipboard utf16 text format: text/plain;charset=ISO-10646-UCS-2 */
108 Atom atomUtf16;
109 /** X11 atom refering to the clipboard utf8 text format: UTF8_STRING */
110 Atom atomUtf8;
111 /** X11 atom refering to the clipboard compound text format: COMPOUND_TEXT */
112 Atom atomCText;
113
114 /** A list of the X11 formats which we support, mapped to our identifier for them, in the
115 order we prefer to have them in. */
116 std::vector<VBOXCLIPBOARDFORMAT> formatList;
117
118 /** Does the host or the guest currently own the clipboard? */
119 volatile enum g_eClipboardOwner eOwner;
120
121 /** What is the best text format the host has to offer? INVALID for none. */
122 g_eClipboardFormats hostTextFormat;
123 /** Atom corresponding to the host text format */
124 Atom atomHostTextFormat;
125 /** What is the best bitmap format the host has to offer? INVALID for none. */
126 g_eClipboardFormats hostBitmapFormat;
127 /** Atom corresponding to the host Bitmap format */
128 Atom atomHostBitmapFormat;
129 /** What formats does the guest have on offer? */
130 int guestFormats;
131 /** Windows caches the clipboard data it receives. Since we have no way of knowing whether
132 that data is still valid, we always send a "data changed" message after a successful
133 transfer to invalidate the cache. */
134 bool notifyGuest;
135
136 /** Since the clipboard data moves asynchronously, we use an event semaphore to wait for
137 it. When a function issues a request for clipboard data it must wait for this
138 semaphore, which is triggered when the data arrives. */
139 RTSEMEVENT waitForData;
140 /** And because it would not do for the guest to be waiting for the host while the host
141 is waiting for the guest, we set a flag and assert horribly if we spot a deadlock. */
142 uint32_t waiter;
143 /** This mutex is held while an asynchronous operation completes (i.e. the host clipboard is
144 being queried) to make sure that the clipboard is not disconnected during that time. It
145 is also grabbed when the clipboard client disconnects. When an asynchronous operation
146 starts completing, it checks that the same client is still connected immediately after
147 grabbing the mutex. */
148 RTSEMMUTEX asyncMutex;
149
150 /** Format which we are reading from the host clipboard (valid during a request for the
151 host clipboard) */
152 g_eClipboardFormats requestHostFormat;
153 /** The guest buffer to write host clipboard data to (valid during a request for the host
154 clipboard) */
155 void *requestBuffer;
156 /** The size of the guest buffer to write host clipboard data to (valid during a request for
157 the host clipboard) */
158 unsigned requestBufferSize;
159 /** The size of the host clipboard data written to the guest buffer (valid during a request
160 for the host clipboard) */
161 uint32_t *requestActualSize;
162
163 /** Pointer to the client data structure */
164 VBOXCLIPBOARDCLIENTDATA *pClient;
165};
166
167/* Only one client is supported. There seems to be no need for more clients. */
168static VBOXCLIPBOARDCONTEXT g_ctx;
169
170/**
171 * Send a request to the guest to transfer the contents of its clipboard to the host.
172 *
173 * @param pCtx Pointer to the host clipboard structure
174 * @param u32Format The format in which the data should be transfered
175 */
176static int vboxClipboardReadDataFromClient (VBOXCLIPBOARDCONTEXT *pCtx, uint32_t u32Format)
177{
178 VBOXCLIPBOARDCLIENTDATA *pClient = pCtx->pClient;
179
180 LogFlowFunc(("u32Format=%02X\n", u32Format));
181 if (pClient == 0)
182 {
183 Log(("vboxClipboardReadDataFromClient: host requested guest clipboard data after guest had disconnected.\n"));
184 pCtx->guestFormats = 0;
185 pCtx->waiter = 0;
186 return VERR_TIMEOUT;
187 }
188 if (!( pCtx->pClient->data.pv == NULL
189 && pCtx->pClient->data.cb == 0
190 && pCtx->pClient->data.u32Format == 0))
191 {
192 LogRel(("vboxClipboardReadDataFromClient: a guest to host clipboard transfer has been requested, but another is in progress, or has not cleaned up properly.\n"));
193 AssertMsgFailed(("A guest to host clipboard transfer has been requested, but another is in progress, or has not cleaned up properly.\n"));
194 }
195
196 /* Only one of the guest and the host should be waiting at any one time */
197 if (RT_FAILURE(ASMAtomicCmpXchgU32(&pCtx->waiter, 1, 0)))
198 {
199 LogRel(("vboxClipboardReadDataFromClient: deadlock situation - the host and the guest are both waiting for data from the other."));
200 return VERR_DEADLOCK;
201 }
202 /* Request data from the guest */
203 vboxSvcClipboardReportMsg (pCtx->pClient, VBOX_SHARED_CLIPBOARD_HOST_MSG_READ_DATA, u32Format);
204 /* Which will signal us when it is ready. */
205 if (RTSemEventWait(pCtx->waitForData, CLIPBOARDTIMEOUT) != VINF_SUCCESS)
206 {
207 LogRel (("vboxClipboardReadDataFromClient: vboxSvcClipboardReportMsg failed to complete within %d milliseconds\n", CLIPBOARDTIMEOUT));
208 pCtx->guestFormats = 0;
209 pCtx->waiter = 0;
210 return VERR_TIMEOUT;
211 }
212 pCtx->waiter = 0;
213 LogFlowFunc(("wait completed. Returning.\n"));
214 return VINF_SUCCESS;
215}
216
217/**
218 * Convert the UTF-16 text returned from the X11 clipboard to UTF-16LE with Windows EOLs
219 * and place it in the global g_pcClipboardText variable. We are reading the host clipboard to
220 * make it available to the guest.
221 *
222 * @param pValue Source UTF-16 text
223 * @param cwSourceLen Length in 16-bit words of the source text
224 * @param pv Where to store the converted data
225 * @param cb Length in bytes of the buffer pointed to by cb
226 * @param pcbActual Where to store the size of the converted data
227 * @param pClient Pointer to the client context structure
228 */
229static void vboxClipboardGetUtf16(XtPointer pValue, unsigned cwSrcLen, void *pv, unsigned cb,
230 uint32_t *pcbActual)
231{
232 size_t cwDestLen;
233 PRTUTF16 pu16SrcText = reinterpret_cast<PRTUTF16>(pValue);
234 PRTUTF16 pu16DestText = reinterpret_cast<PRTUTF16>(pv);
235 int rc;
236
237 LogFlowFunc (("converting Utf-16 to Utf-16LE. cwSrcLen=%d, cb=%d, pu16SrcText+1=%.*ls\n",
238 cwSrcLen, cb, cwSrcLen - 1, pu16SrcText + 1));
239 /* How long will the converted text be? */
240 rc = vboxClipboardUtf16GetWinSize(pu16SrcText, cwSrcLen, &cwDestLen);
241 if (RT_FAILURE(rc))
242 {
243 XtFree(reinterpret_cast<char *>(pValue));
244 LogRel(("vboxClipboardGetUtf16: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
245 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
246 cb, cwDestLen * 2));
247 *pcbActual = cwDestLen * 2;
248 RTSemEventSignal(g_ctx.waitForData);
249 AssertReturnVoid(RT_SUCCESS(rc));
250 }
251 if (cb < cwDestLen * 2)
252 {
253 XtFree(reinterpret_cast<char *>(pValue));
254 /* Report the amount of buffer space needed for the transfer */
255 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
256 cb, cwDestLen * 2));
257 *pcbActual = cwDestLen * 2;
258 RTSemEventSignal(g_ctx.waitForData);
259 return;
260 }
261 /* Convert the text. */
262 rc = vboxClipboardUtf16LinToWin(pu16SrcText, cwSrcLen, pu16DestText, cb / 2);
263 if (RT_FAILURE(rc))
264 {
265 LogRel(("vboxClipboardGetUtf16: clipboard conversion failed. vboxClipboardUtf16LinToWin returned %Vrc. Abandoning.\n", rc));
266 XtFree(reinterpret_cast<char *>(pValue));
267 *pcbActual = 0;
268 RTSemEventSignal(g_ctx.waitForData);
269 return;
270 }
271 LogFlowFunc (("converted string is %.*ls. Returning.\n", cwDestLen, pu16DestText));
272 *pcbActual = cwDestLen * 2;
273 XtFree(reinterpret_cast<char *>(pValue));
274 RTSemEventSignal(g_ctx.waitForData);
275}
276
277/**
278 * Convert the UTF-8 text returned from the X11 clipboard to UTF-16LE with Windows EOLS.
279 * We are reading the host clipboard to make it available to the guest.
280 *
281 * @param pValue Source UTF-8 text
282 * @param cbSourceLen Length in 8-bit bytes of the source text
283 * @param pv Where to store the converted data
284 * @param cb Length in bytes of the buffer pointed to by pv
285 * @param pcbActual Where to store the size of the converted data
286 * @param pClient Pointer to the client context structure
287 */
288static void vboxClipboardGetUtf8(XtPointer pValue, unsigned cbSrcLen, void *pv, unsigned cb,
289 uint32_t *pcbActual)
290{
291 size_t cwSrcLen, cwDestLen;
292 char *pu8SrcText = reinterpret_cast<char *>(pValue);
293 PRTUTF16 pu16SrcText;
294 PRTUTF16 pu16DestText = reinterpret_cast<PRTUTF16>(pv);
295 int rc;
296
297 LogFlowFunc (("converting Utf-8 to Utf-16LE. cbSrcLen=%d, cb=%d, pu8SrcText=%.*s\n",
298 cbSrcLen, cb, cbSrcLen, pu8SrcText));
299 /* First convert the UTF8 to UTF16 */
300 rc = RTStrToUtf16Ex(pu8SrcText, cbSrcLen, &pu16SrcText, 0, &cwSrcLen);
301 XtFree(reinterpret_cast<char *>(pValue));
302 if (RT_FAILURE(rc))
303 {
304 LogRel(("vboxClipboardGetUtf8: clipboard conversion failed. RTStrToUtf16Ex returned %Vrc. Abandoning.\n", rc));
305 *pcbActual = 0;
306 RTSemEventSignal(g_ctx.waitForData);
307 return;
308 }
309 /* Check how much longer will the converted text will be. */
310 rc = vboxClipboardUtf16GetWinSize(pu16SrcText, cwSrcLen, &cwDestLen);
311 if (RT_FAILURE(rc))
312 {
313 LogRel(("vboxClipboardGetUtf8: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
314 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
315 cb, cwDestLen * 2));
316 RTUtf16Free(pu16SrcText);
317 *pcbActual = cwDestLen * 2;
318 RTSemEventSignal(g_ctx.waitForData);
319 AssertReturnVoid(RT_SUCCESS(rc));
320 }
321 if (cb < cwDestLen * 2)
322 {
323 RTUtf16Free(pu16SrcText);
324 /* Report the amount of buffer space needed for the transfer */
325 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
326 cb, cwDestLen * 2));
327 *pcbActual = cwDestLen * 2;
328 RTSemEventSignal(g_ctx.waitForData);
329 return;
330 }
331 /* Convert the text. */
332 rc = vboxClipboardUtf16LinToWin(pu16SrcText, cwSrcLen, pu16DestText, cb / 2);
333 RTUtf16Free(pu16SrcText);
334 if (RT_FAILURE(rc))
335 {
336 LogRel(("vboxClipboardGetUtf8: clipboard conversion failed. vboxClipboardUtf16LinToWin returned %Vrc. Abandoning.\n", rc));
337 *pcbActual = 0;
338 RTSemEventSignal(g_ctx.waitForData);
339 return;
340 }
341 LogFlowFunc (("converted string is %.*ls. Returning.\n", cwDestLen, pu16DestText));
342 *pcbActual = cwDestLen * 2;
343 RTSemEventSignal(g_ctx.waitForData);
344}
345
346/**
347 * Convert the COMPOUND_TEXT text returned from the X11 clipboard to UTF-16LE with Windows
348 * EOLS. We are reading the host clipboard to make it available to the guest.
349 *
350 * @param pValue Source COMPOUND_TEXT text
351 * @param cbSourceLen Length in 8-bit bytes of the source text
352 * @param pv Where to store the converted data
353 * @param cb Length in bytes of the buffer pointed to by pv
354 * @param pcbActual Where to store the size of the converted data
355 * @param pClient Pointer to the client context structure
356 */
357static void vboxClipboardGetCText(XtPointer pValue, unsigned cbSrcLen, void *pv, unsigned cb,
358 uint32_t *pcbActual)
359{
360 size_t cwSrcLen, cwDestLen;
361 char **ppu8SrcText;
362 PRTUTF16 pu16SrcText;
363 PRTUTF16 pu16DestText = reinterpret_cast<PRTUTF16>(pv);
364 XTextProperty property;
365 int rc, cProps;
366
367 LogFlowFunc (("converting COMPOUND TEXT to Utf-16LE. cbSrcLen=%d, cb=%d, pu8SrcText=%.*s\n",
368 cbSrcLen, cb, cbSrcLen, reinterpret_cast<char *>(pValue)));
369 /* First convert the compound text to Utf8 */
370 property.value = reinterpret_cast<unsigned char *>(pValue);
371 property.encoding = g_ctx.atomCText;
372 property.format = 8;
373 property.nitems = cbSrcLen;
374#ifdef RT_OS_SOLARIS
375 rc = XmbTextPropertyToTextList(XtDisplay(g_ctx.widget), &property, &ppu8SrcText, &cProps);
376#else
377 rc = Xutf8TextPropertyToTextList(XtDisplay(g_ctx.widget), &property, &ppu8SrcText, &cProps);
378#endif
379 XtFree(reinterpret_cast<char *>(pValue));
380 if (rc < 0)
381 {
382 const char *pcReason;
383 switch(rc)
384 {
385 case XNoMemory:
386 pcReason = "out of memory";
387 break;
388 case XLocaleNotSupported:
389 pcReason = "locale (Utf8) not supported";
390 break;
391 case XConverterNotFound:
392 pcReason = "converter not found";
393 break;
394 default:
395 pcReason = "unknown error";
396 }
397 XFreeStringList(ppu8SrcText);
398 LogRel(("vboxClipboardGetCText: Xutf8TextPropertyToTextList failed. Reason: %s\n",
399 pcReason));
400 *pcbActual = 0;
401 RTSemEventSignal(g_ctx.waitForData);
402 return;
403 }
404 /* Now convert the UTF8 to UTF16 */
405 rc = RTStrToUtf16Ex(*ppu8SrcText, cbSrcLen, &pu16SrcText, 0, &cwSrcLen);
406 XFreeStringList(ppu8SrcText);
407 if (RT_FAILURE(rc))
408 {
409 LogRel(("vboxClipboardGetCText: clipboard conversion failed. RTStrToUtf16Ex returned %Vrc. Abandoning.\n", rc));
410 *pcbActual = 0;
411 RTSemEventSignal(g_ctx.waitForData);
412 return;
413 }
414 /* Check how much longer will the converted text will be. */
415 rc = vboxClipboardUtf16GetWinSize(pu16SrcText, cwSrcLen, &cwDestLen);
416 if (RT_FAILURE(rc))
417 {
418 LogRel(("vboxClipboardGetCText: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
419 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
420 cb, cwDestLen * 2));
421 RTUtf16Free(pu16SrcText);
422 *pcbActual = cwDestLen * 2;
423 RTSemEventSignal(g_ctx.waitForData);
424 AssertReturnVoid(RT_SUCCESS(rc));
425 }
426 if (cb < cwDestLen * 2)
427 {
428 RTUtf16Free(pu16SrcText);
429 /* Report the amount of buffer space needed for the transfer */
430 LogFlowFunc (("guest buffer too small: size %d bytes, needed %d. Returning.\n",
431 cb, cwDestLen * 2));
432 *pcbActual = cwDestLen * 2;
433 RTSemEventSignal(g_ctx.waitForData);
434 return;
435 }
436 /* Convert the text. */
437 rc = vboxClipboardUtf16LinToWin(pu16SrcText, cwSrcLen, pu16DestText, cb / 2);
438 RTUtf16Free(pu16SrcText);
439 if (RT_FAILURE(rc))
440 {
441 LogRel(("vboxClipboardGetCText: clipboard conversion failed. vboxClipboardUtf16LinToWin returned %Vrc. Abandoning.\n", rc));
442 *pcbActual = 0;
443 RTSemEventSignal(g_ctx.waitForData);
444 return;
445 }
446 LogFlowFunc (("converted string is %.*ls. Returning.\n", cwDestLen, pu16DestText));
447 *pcbActual = cwDestLen * 2;
448 RTSemEventSignal(g_ctx.waitForData);
449}
450
451/**
452 * Convert the Latin1 text returned from the X11 clipboard to UTF-16LE with Windows EOLS
453 * and place it in the global g_pcClipboardText variable. We are reading the host clipboard to
454 * make it available to the guest.
455 *
456 * @param pValue Source Latin1 text
457 * @param cbSourceLen Length in 8-bit bytes of the source text
458 * @param pv Where to store the converted data
459 * @param cb Length in bytes of the buffer pointed to by cb
460 * @param pcbActual Where to store the size of the converted data
461 * @param pClient Pointer to the client context structure
462 */
463static void vboxClipboardGetLatin1(XtPointer pValue, unsigned cbSourceLen, void *pv, unsigned cb,
464 uint32_t *pcbActual)
465{
466 unsigned cwDestLen = cbSourceLen + 1;
467 char *pu8SourceText = reinterpret_cast<char *>(pValue);
468 PRTUTF16 pu16DestText = reinterpret_cast<PRTUTF16>(pv);
469
470 LogFlow (("vboxClipboardGetLatin1: converting Latin1 to Utf-16LE. Original is %.*s\n",
471 cbSourceLen, pu8SourceText));
472 for (unsigned i = 0; i < cbSourceLen; i++)
473 if (pu8SourceText[i] == LINEFEED)
474 ++cwDestLen;
475 if (cb < cwDestLen * 2)
476 {
477 XtFree(reinterpret_cast<char *>(pValue));
478 /* Report the amount of buffer space needed for the transfer */
479 Log2 (("vboxClipboardGetLatin1: guest buffer too small: size %d bytes\n", cb));
480 *pcbActual = cwDestLen * 2;
481 RTSemEventSignal(g_ctx.waitForData);
482 return;
483 }
484 for (unsigned i = 0, j = 0; i < cbSourceLen; ++i, ++j)
485 if (pu8SourceText[i] != LINEFEED)
486 pu16DestText[j] = pu8SourceText[i]; /* latin1 < utf-16LE */
487 else
488 {
489 pu16DestText[j] = CARRIAGERETURN;
490 ++j;
491 pu16DestText[j] = LINEFEED;
492 }
493 pu16DestText[cwDestLen - 1] = 0;
494 *pcbActual = cwDestLen * 2;
495 Log2 (("vboxClipboardGetLatin1: converted text is %.*ls\n", cwDestLen, pu16DestText));
496 XtFree(reinterpret_cast<char *>(pValue));
497 RTSemEventSignal(g_ctx.waitForData);
498}
499
500/** Convert the clipboard text from the current format to Utf-16 with Windows line breaks.
501 We are reading the host clipboard to make it available to the guest. */
502static void vboxClipboardGetProc(Widget, XtPointer pClientData, Atom * /* selection */,
503 Atom *atomType, XtPointer pValue, long unsigned int *pcLen,
504 int *piFormat)
505{
506 LogFlowFunc(("pClientData=%p, *pcLen=%lu, *piFormat=%d\n", pClientData, *pcLen, *piFormat));
507 LogFlowFunc(("g_ctx.requestHostFormat=%d, g_ctx.requestBufferSize=%d\n",
508 g_ctx.requestHostFormat, g_ctx.requestBufferSize));
509 unsigned cTextLen = (*pcLen) * (*piFormat) / 8;
510 /* The X Toolkit may have failed to get the clipboard selection for us. */
511 if (*atomType == XT_CONVERT_FAIL)
512 return;
513 /* The clipboard selection may have changed before we could get it. */
514 if (NULL == pValue)
515 return;
516 /* We grab this mutex whenever an asynchronous clipboard operation completes and while
517 disconnecting a client from the clipboard to stop these operations colliding. */
518 RTSemMutexRequest(g_ctx.asyncMutex, RT_INDEFINITE_WAIT);
519 if (reinterpret_cast<VBOXCLIPBOARDCLIENTDATA *>(pClientData) != g_ctx.pClient)
520 {
521 /* If the client is no longer connected, just return. */
522 XtFree(reinterpret_cast<char *>(pValue));
523 LogFlowFunc(("client is no longer connected, returning\n"));
524 RTSemMutexRelease(g_ctx.asyncMutex);
525 return;
526 }
527
528 /* In which format did we request the clipboard data? */
529 switch (g_ctx.requestHostFormat)
530 {
531 case UTF16:
532 vboxClipboardGetUtf16(pValue, cTextLen / 2, g_ctx.requestBuffer, g_ctx.requestBufferSize,
533 g_ctx.requestActualSize);
534 break;
535 case CTEXT:
536 vboxClipboardGetCText(pValue, cTextLen, g_ctx.requestBuffer, g_ctx.requestBufferSize,
537 g_ctx.requestActualSize);
538 break;
539 case UTF8:
540 {
541 /* If we are given broken Utf-8, we treat it as Latin1. Is this acceptable? */
542 size_t cStringLen;
543 char *pu8SourceText = reinterpret_cast<char *>(pValue);
544
545 if ((g_ctx.requestHostFormat == UTF8)
546 && (RTStrUniLenEx(pu8SourceText, *pcLen, &cStringLen) == VINF_SUCCESS))
547 {
548 vboxClipboardGetUtf8(pValue, cTextLen, g_ctx.requestBuffer, g_ctx.requestBufferSize,
549 g_ctx.requestActualSize);
550 break;
551 }
552 else
553 {
554 vboxClipboardGetLatin1(pValue, cTextLen, g_ctx.requestBuffer, g_ctx.requestBufferSize,
555 g_ctx.requestActualSize);
556 break;
557 }
558 }
559 default:
560 Log (("vboxClipboardGetProc: bad target format\n"));
561 XtFree(reinterpret_cast<char *>(pValue));
562 RTSemMutexRelease(g_ctx.asyncMutex);
563 return;
564 }
565 g_ctx.notifyGuest = true;
566 RTSemMutexRelease(g_ctx.asyncMutex);
567}
568
569/** Callback to handle a reply to a request for the targets the current clipboard holder can
570 handle. We are reading the host clipboard to make it available to the guest. */
571static void vboxClipboardTargetsProc(Widget, XtPointer pClientData, Atom * /* selection */,
572 Atom *atomType, XtPointer pValue, long unsigned int *pcLen,
573 int *piFormat)
574{
575 Atom *atomTargets = reinterpret_cast<Atom *>(pValue);
576 unsigned cAtoms = *pcLen;
577 g_eClipboardFormats eBestTarget = INVALID;
578 Atom atomBestTarget = None;
579
580 Log3 (("vboxClipboardTargetsProc called\n"));
581 if (*atomType == XT_CONVERT_FAIL)
582 {
583 Log (("vboxClipboardTargetsProc: reading clipboard from host, X toolkit failed to convert the selection\n"));
584 return;
585 }
586 /* We grab this mutex whenever an asynchronous clipboard operation completes and while
587 disconnecting a client from the clipboard to stop these operations colliding. */
588 RTSemMutexRequest(g_ctx.asyncMutex, RT_INDEFINITE_WAIT);
589 if (reinterpret_cast<VBOXCLIPBOARDCLIENTDATA *>(pClientData) != g_ctx.pClient)
590 {
591 /* If the client is no longer connected, just return. */
592 LogFlowFunc(("client is no longer connected, returning\n"));
593 RTSemMutexRelease(g_ctx.asyncMutex);
594 return;
595 }
596
597 for (unsigned i = 0; i < cAtoms; ++i)
598 {
599 for (unsigned j = 0; j != g_ctx.formatList.size(); ++j)
600 if (g_ctx.formatList[j].atom == atomTargets[i])
601 {
602 if (eBestTarget < g_ctx.formatList[j].format)
603 {
604 eBestTarget = g_ctx.formatList[j].format;
605 atomBestTarget = g_ctx.formatList[j].atom;
606 }
607 break;
608 }
609#ifdef DEBUG
610 char *szAtomName = XGetAtomName(XtDisplay(g_ctx.widget), atomTargets[i]);
611 if (szAtomName != 0)
612 {
613 Log3 (("vboxClipboardTargetsProc: the host offers target %s\n", szAtomName));
614 XFree(szAtomName);
615 }
616#endif
617 }
618 g_ctx.atomHostTextFormat = atomBestTarget;
619 if ((eBestTarget != g_ctx.hostTextFormat) || (g_ctx.notifyGuest == true))
620 {
621 uint32_t u32Formats = 0;
622#ifdef DEBUG
623 if (atomBestTarget != None)
624 {
625 char *szAtomName = XGetAtomName(XtDisplay(g_ctx.widget), atomBestTarget);
626 Log2 (("vboxClipboardTargetsProc: switching to host text target %s. Available targets are:\n",
627 szAtomName));
628 XFree(szAtomName);
629 }
630 else
631 {
632 Log2(("vboxClipboardTargetsProc: no supported host text target found. Available targets are:\n"));
633 }
634 for (unsigned i = 0; i < cAtoms; ++i)
635 {
636 char *szAtomName = XGetAtomName(XtDisplay(g_ctx.widget), atomTargets[i]);
637 if (szAtomName != 0)
638 {
639 Log2 (("vboxClipboardTargetsProc: %s\n", szAtomName));
640 XFree(szAtomName);
641 }
642 }
643#endif
644 g_ctx.hostTextFormat = eBestTarget;
645 if (eBestTarget != INVALID)
646 u32Formats |= VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT;
647 vboxSvcClipboardReportMsg (g_ctx.pClient, VBOX_SHARED_CLIPBOARD_HOST_MSG_FORMATS,
648 u32Formats);
649 g_ctx.notifyGuest = false;
650 }
651 XtFree(reinterpret_cast<char *>(pValue));
652 RTSemMutexRelease(g_ctx.asyncMutex);
653}
654
655/**
656 * This callback is called every 200ms to check the contents of the host clipboard.
657 */
658static void vboxClipboardTimerProc(XtPointer /* pUserData */, XtIntervalId * /* hTimerId */)
659{
660 Log3 (("vboxClipboardTimerProc called\n"));
661 /* Get the current clipboard contents */
662 if (g_ctx.eOwner == HOST && g_ctx.pClient != 0)
663 {
664 Log3 (("vboxClipboardTimerProc: requesting the targets that the host clipboard offers\n"));
665 XtGetSelectionValue(g_ctx.widget, g_ctx.atomClipboard, g_ctx.atomTargets,
666 vboxClipboardTargetsProc, reinterpret_cast<XtPointer>(g_ctx.pClient),
667 CurrentTime);
668 }
669 /* Re-arm our timer */
670 XtAppAddTimeOut(g_ctx.appContext, 200 /* ms */, vboxClipboardTimerProc, 0);
671}
672
673/** We store information about the target formats we can handle in a global vector for internal
674 use. */
675static void vboxClipboardAddFormat(const char *pszName, g_eClipboardFormats eFormat,
676 unsigned guestFormat)
677{
678 VBOXCLIPBOARDFORMAT sFormat;
679 /* Get an atom from the X server for that target format */
680 Atom atomFormat = XInternAtom(XtDisplay(g_ctx.widget), pszName, false);
681 sFormat.atom = atomFormat;
682 sFormat.format = eFormat;
683 sFormat.guestFormat = guestFormat;
684 g_ctx.formatList.push_back(sFormat);
685 LogFlow (("vboxClipboardAddFormat: added format %s (%d)\n", pszName, eFormat));
686}
687
688/**
689 * The main loop of our clipboard reader.
690 */
691static int vboxClipboardThread(RTTHREAD self, void * /* pvUser */)
692{
693 /* Create a window and make it a clipboard viewer. */
694 int cArgc = 0;
695 char *pcArgv = 0;
696 int rc = VINF_SUCCESS;
697 // static String szFallbackResources[] = { (char*)"*.width: 1", (char*)"*.height: 1", NULL };
698 Display *pDisplay;
699 LogRel (("vboxClipboardThread: starting clipboard thread\n"));
700
701 /* Make sure we are thread safe */
702 XtToolkitThreadInitialize();
703 /* Set up the Clipbard application context and main window. We call all these functions
704 directly instead of calling XtOpenApplication() so that we can fail gracefully if we
705 can't get an X11 display. */
706 XtToolkitInitialize();
707 g_ctx.appContext = XtCreateApplicationContext();
708 // XtAppSetFallbackResources(g_ctx.appContext, szFallbackResources);
709 pDisplay = XtOpenDisplay(g_ctx.appContext, 0, 0, "VBoxClipboard", 0, 0, &cArgc, &pcArgv);
710 if (pDisplay == 0)
711 {
712 LogRel(("vboxClipboardThread: failed to connect to the host clipboard - the window system may not be running.\n"));
713 return VERR_NOT_SUPPORTED;
714 }
715 g_ctx.widget = XtVaAppCreateShell(0, "VBoxClipboard", applicationShellWidgetClass, pDisplay,
716 XtNwidth, 1, XtNheight, 1, NULL);
717 if (g_ctx.widget == 0)
718 {
719 LogRel(("vboxClipboardThread: failed to construct the X11 window for the clipboard manager.\n"));
720 AssertReturn(g_ctx.widget != 0, VERR_ACCESS_DENIED);
721 }
722 RTThreadUserSignal(self);
723 XtSetMappedWhenManaged(g_ctx.widget, false);
724 XtRealizeWidget(g_ctx.widget);
725
726 /* Get hold of the atoms which we need */
727 g_ctx.atomClipboard = XInternAtom(XtDisplay(g_ctx.widget), "CLIPBOARD", false /* only_if_exists */);
728 g_ctx.atomPrimary = XInternAtom(XtDisplay(g_ctx.widget), "PRIMARY", false);
729 g_ctx.atomTargets = XInternAtom(XtDisplay(g_ctx.widget), "TARGETS", false);
730 g_ctx.atomMultiple = XInternAtom(XtDisplay(g_ctx.widget), "MULTIPLE", false);
731 g_ctx.atomTimestamp = XInternAtom(XtDisplay(g_ctx.widget), "TIMESTAMP", false);
732 g_ctx.atomUtf16 = XInternAtom(XtDisplay(g_ctx.widget),
733 "text/plain;charset=ISO-10646-UCS-2", false);
734 g_ctx.atomUtf8 = XInternAtom(XtDisplay(g_ctx.widget), "UTF_STRING", false);
735 /* And build up the vector of supported formats */
736 g_ctx.atomCText = XInternAtom(XtDisplay(g_ctx.widget), "COMPOUND_TEXT", false);
737 /* And build up the vector of supported formats */
738#ifdef USE_UTF16
739 vboxClipboardAddFormat("text/plain;charset=ISO-10646-UCS-2", UTF16,
740 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
741#endif
742#ifdef USE_UTF8
743 vboxClipboardAddFormat("UTF8_STRING", UTF8,
744 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
745 vboxClipboardAddFormat("text/plain;charset=UTF-8", UTF8,
746 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
747 vboxClipboardAddFormat("text/plain;charset=utf-8", UTF8,
748 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
749 vboxClipboardAddFormat("STRING", UTF8,
750 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
751 vboxClipboardAddFormat("TEXT", UTF8,
752 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
753 vboxClipboardAddFormat("text/plain", UTF8,
754 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
755#endif
756#ifdef USE_CTEXT
757 vboxClipboardAddFormat("COMPOUND_TEXT", CTEXT,
758 VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
759#endif
760
761 /* Set up a timer to poll the host clipboard */
762 XtAppAddTimeOut(g_ctx.appContext, 200 /* ms */, vboxClipboardTimerProc, 0);
763
764 XtAppMainLoop(g_ctx.appContext);
765 g_ctx.formatList.clear();
766 RTSemEventDestroy(g_ctx.waitForData);
767 RTSemMutexDestroy(g_ctx.asyncMutex);
768 LogRel (("vboxClipboardThread: clipboard thread terminated successfully with return code %Vrc\n", rc));
769 return rc;
770}
771
772/** Initialise the host side of the shared clipboard - called by the hgcm layer. */
773int vboxClipboardInit (void)
774{
775 int rc;
776
777 LogRel(("vboxClipboardInit: initializing host clipboard\n"));
778 RTSemEventCreate(&g_ctx.waitForData);
779 RTSemMutexCreate(&g_ctx.asyncMutex);
780 rc = RTThreadCreate(&g_ctx.thread, vboxClipboardThread, 0, 0, RTTHREADTYPE_IO,
781 RTTHREADFLAGS_WAITABLE, "SHCLIP");
782 if (RT_FAILURE(rc))
783 {
784 LogRel(("vboxClipboardInit: failed to create the clipboard thread.\n"));
785 RTSemEventDestroy(g_ctx.waitForData);
786 RTSemMutexDestroy(g_ctx.asyncMutex);
787 AssertRCReturn(rc, rc);
788 }
789 return RTThreadUserWait(g_ctx.thread, 1000);
790}
791
792/** Terminate the host side of the shared clipboard - called by the hgcm layer. */
793void vboxClipboardDestroy (void)
794{
795 LogRel(("vboxClipboardDestroy: shutting down host clipboard\n"));
796 int rc, rcThread;
797 unsigned count = 0;
798 XEvent ev;
799
800 /* Set the termination flag. */
801 XtAppSetExitFlag(g_ctx.appContext);
802 /* Wake up the event loop */
803 memset(&ev, 0, sizeof(ev));
804 ev.xclient.type = ClientMessage;
805 ev.xclient.format = 8;
806 XSendEvent(XtDisplay(g_ctx.widget), XtWindow(g_ctx.widget), false, 0, &ev);
807 XFlush(XtDisplay(g_ctx.widget));
808 do
809 {
810 rc = RTThreadWait(g_ctx.thread, 1000, &rcThread);
811 ++count;
812 Assert(RT_SUCCESS(rc) || ((VERR_TIMEOUT == rc) && (count != 5)));
813 } while ((VERR_TIMEOUT == rc) && (count < 300));
814 AssertRC(rcThread);
815 XtCloseDisplay(XtDisplay(g_ctx.widget));
816 LogFlowFunc(("returning.\n"));
817}
818
819/**
820 * Enable the shared clipboard - called by the hgcm clipboard subsystem.
821 *
822 * @param pClient Structure containing context information about the guest system
823 * @returns RT status code
824 */
825int vboxClipboardConnect (VBOXCLIPBOARDCLIENTDATA *pClient)
826{
827 LogFlow(("vboxClipboardConnect\n"));
828
829 /* Only one client is supported for now */
830 if (g_ctx.pClient != 0)
831 {
832 LogRel(("vboxClipboardConnect: attempted to connect, but a client appears to be already running.\n"));
833 AssertReturn(g_ctx.pClient == 0, VERR_NOT_SUPPORTED);
834 }
835
836 pClient->pCtx = &g_ctx;
837 pClient->pCtx->pClient = pClient;
838 g_ctx.eOwner = HOST;
839 g_ctx.notifyGuest = true;
840 return VINF_SUCCESS;
841}
842
843/**
844 * Synchronise the contents of the host clipboard with the guest, called by the HGCM layer
845 * after a save and restore of the guest.
846 */
847int vboxClipboardSync (VBOXCLIPBOARDCLIENTDATA *pClient)
848{
849 /* On a Linux host, the guest should never synchronise/cache its clipboard contents, as
850 we have no way of reliably telling when the host clipboard data changes. So instead
851 of synchronising, we tell the guest to empty its clipboard, and we set the cached
852 flag so that we report formats to the guest next time we poll for them. */
853 vboxSvcClipboardReportMsg (g_ctx.pClient, VBOX_SHARED_CLIPBOARD_HOST_MSG_FORMATS, 0);
854 g_ctx.notifyGuest = true;
855
856 return VINF_SUCCESS;
857}
858
859/**
860 * Shut down the shared clipboard subsystem and "disconnect" the guest.
861 */
862void vboxClipboardDisconnect (VBOXCLIPBOARDCLIENTDATA *pClient)
863{
864 LogFlow(("vboxClipboardDisconnect\n"));
865
866 RTSemMutexRequest(g_ctx.asyncMutex, RT_INDEFINITE_WAIT);
867 g_ctx.pClient = NULL;
868 g_ctx.eOwner = NONE;
869 g_ctx.hostTextFormat = INVALID;
870 g_ctx.hostBitmapFormat = INVALID;
871 RTSemMutexRelease(g_ctx.asyncMutex);
872}
873
874/**
875 * Satisfy a request from the host for available clipboard targets.
876 *
877 * @returns true if we successfully convert the data to the format requested, false otherwise.
878 *
879 * @param atomTypeReturn The type of the data we are returning
880 * @param pValReturn A pointer to the data we are returning. This should be to memory
881 * allocated by XtMalloc, which will be freed by the toolkit later
882 * @param pcLenReturn The length of the data we are returning
883 * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are returning
884 */
885static Boolean vboxClipboardConvertTargets(Atom *atomTypeReturn, XtPointer *pValReturn,
886 unsigned long *pcLenReturn, int *piFormatReturn)
887{
888 unsigned uListSize = g_ctx.formatList.size();
889 Atom *atomTargets = reinterpret_cast<Atom *>(XtMalloc((uListSize + 3) * sizeof(Atom)));
890 unsigned cTargets = 0;
891
892 LogFlow (("vboxClipboardConvertTargets called\n"));
893 for (unsigned i = 0; i < uListSize; ++i)
894 {
895 if ( ((g_ctx.guestFormats & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT) != 0)
896 && (g_ctx.formatList[i].guestFormat == VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT))
897 {
898 atomTargets[cTargets] = g_ctx.formatList[i].atom;
899 ++cTargets;
900 }
901 }
902 atomTargets[cTargets] = g_ctx.atomTargets;
903 atomTargets[cTargets + 1] = g_ctx.atomMultiple;
904 atomTargets[cTargets + 2] = g_ctx.atomTimestamp;
905#ifdef DEBUG
906 for (unsigned i = 0; i < cTargets + 3; i++)
907 {
908 char *szAtomName = XGetAtomName(XtDisplay(g_ctx.widget), atomTargets[i]);
909 if (szAtomName != 0)
910 {
911 Log2 (("vboxClipboardConvertTargets: returning target %s\n", szAtomName));
912 XFree(szAtomName);
913 }
914 else
915 {
916 Log(("vboxClipboardConvertTargets: invalid atom %d in the list!\n", atomTargets[i]));
917 }
918 }
919#endif
920 *atomTypeReturn = XA_ATOM;
921 *pValReturn = reinterpret_cast<XtPointer>(atomTargets);
922 *pcLenReturn = cTargets + 3;
923 *piFormatReturn = 32;
924 return true;
925}
926
927/**
928 * Reset the contents of the buffer used to pass clipboard data from the guest to the host.
929 * This must be done after every clipboard transfer.
930 */
931static void vboxClipboardEmptyGuestBuffer(void)
932{
933 if (g_ctx.pClient->data.pv != 0)
934 RTMemFree(g_ctx.pClient->data.pv);
935 g_ctx.pClient->data.pv = 0;
936 g_ctx.pClient->data.cb = 0;
937 g_ctx.pClient->data.u32Format = 0;
938}
939
940/**
941 * Satisfy a request from the host to convert the clipboard text to Utf16. We return non-zero
942 * terminated text.
943 *
944 * @returns true if we successfully convert the data to the format requested, false otherwise.
945 *
946 * @retval atomTypeReturn The type of the data we are returning
947 * @retval pValReturn A pointer to the data we are returning. This should be to memory
948 * allocated by XtMalloc, which will be freed by the toolkit later
949 * @retval pcLenReturn The length of the data we are returning
950 * @retval piFormatReturn The format (8bit, 16bit, 32bit) of the data we are returning
951 */
952static Boolean vboxClipboardConvertUtf16(Atom *atomTypeReturn, XtPointer *pValReturn,
953 unsigned long *pcLenReturn, int *piFormatReturn)
954{
955 PRTUTF16 pu16SrcText, pu16DestText;
956 size_t cwSrcLen, cwDestLen;
957 int rc;
958
959 LogFlowFunc (("called\n"));
960 rc = vboxClipboardReadDataFromClient(&g_ctx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
961 if ((RT_FAILURE(rc)) || (g_ctx.pClient->data.cb == 0))
962 {
963 /* If vboxClipboardReadDataFromClient fails then pClient may be invalid */
964 LogRelFunc (("vboxClipboardReadDataFromClient returned %Rrc%s\n", rc,
965 RT_SUCCESS(rc) ? ", g_ctx.pClient->data.cb == 0" : ""));
966 vboxClipboardEmptyGuestBuffer();
967 return false;
968 }
969 pu16SrcText = reinterpret_cast<PRTUTF16>(g_ctx.pClient->data.pv);
970 cwSrcLen = g_ctx.pClient->data.cb / 2;
971 /* How long will the converted text be? */
972 rc = vboxClipboardUtf16GetLinSize(pu16SrcText, cwSrcLen, &cwDestLen);
973 if (RT_FAILURE(rc))
974 {
975 LogRel(("vboxClipboardConvertUtf16: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
976 vboxClipboardEmptyGuestBuffer();
977 AssertRCReturn(rc, false);
978 }
979 if (cwDestLen == 0)
980 {
981 LogFlowFunc(("received empty clipboard data from the guest, returning false.\n"));
982 vboxClipboardEmptyGuestBuffer();
983 return false;
984 }
985 pu16DestText = reinterpret_cast<PRTUTF16>(XtMalloc(cwDestLen * 2));
986 if (pu16DestText == 0)
987 {
988 LogRel (("vboxClipboardConvertUtf16: failed to allocate %d bytes\n", cwDestLen * 2));
989 vboxClipboardEmptyGuestBuffer();
990 return false;
991 }
992 /* Convert the text. */
993 rc = vboxClipboardUtf16WinToLin(pu16SrcText, cwSrcLen, pu16DestText, cwDestLen);
994 if (RT_FAILURE(rc))
995 {
996 LogRel(("vboxClipboardConvertUtf16: clipboard conversion failed. vboxClipboardUtf16WinToLin returned %Vrc. Abandoning.\n", rc));
997 XtFree(reinterpret_cast<char *>(pu16DestText));
998 vboxClipboardEmptyGuestBuffer();
999 return false;
1000 }
1001 LogFlowFunc (("converted string is %.*ls. Returning.\n", cwDestLen, pu16DestText));
1002 vboxClipboardEmptyGuestBuffer();
1003 *atomTypeReturn = g_ctx.atomUtf16;
1004 *pValReturn = reinterpret_cast<XtPointer>(pu16DestText);
1005 *pcLenReturn = cwDestLen;
1006 *piFormatReturn = 16;
1007 return true;
1008}
1009
1010/**
1011 * Satisfy a request from the host to convert the clipboard text to Utf8.
1012 *
1013 * @returns true if we successfully convert the data to the format requested, false otherwise.
1014 *
1015 * @param atomTypeReturn The type of the data we are returning
1016 * @param pValReturn A pointer to the data we are returning. This should be to memory
1017 * allocated by XtMalloc, which will be freed by the toolkit later
1018 * @param pcLenReturn The length of the data we are returning
1019 * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are returning
1020 */
1021static Boolean vboxClipboardConvertUtf8(Atom *atomTypeReturn, XtPointer *pValReturn,
1022 unsigned long *pcLenReturn, int *piFormatReturn)
1023{
1024 PRTUTF16 pu16SrcText, pu16DestText;
1025 char *pu8DestText;
1026 size_t cwSrcLen, cwDestLen, cbDestLen;
1027 int rc;
1028
1029 LogFlowFunc (("called\n"));
1030 /* Read the clipboard data from the guest. */
1031 rc = vboxClipboardReadDataFromClient(&g_ctx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
1032 if ((rc != VINF_SUCCESS) || (g_ctx.pClient->data.cb == 0))
1033 {
1034 /* If vboxClipboardReadDataFromClient fails then pClient may be invalid */
1035 LogRelFunc (("vboxClipboardReadDataFromClient returned %Rrc%s\n", rc,
1036 RT_SUCCESS(rc) ? ", g_ctx.pClient->data.cb == 0" : ""));
1037 vboxClipboardEmptyGuestBuffer();
1038 return false;
1039 }
1040 pu16SrcText = reinterpret_cast<PRTUTF16>(g_ctx.pClient->data.pv);
1041 cwSrcLen = g_ctx.pClient->data.cb / 2;
1042 /* How long will the converted text be? */
1043 rc = vboxClipboardUtf16GetLinSize(pu16SrcText, cwSrcLen, &cwDestLen);
1044 if (RT_FAILURE(rc))
1045 {
1046 LogRel(("vboxClipboardConvertUtf8: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
1047 vboxClipboardEmptyGuestBuffer();
1048 AssertRCReturn(rc, false);
1049 }
1050 if (cwDestLen == 0)
1051 {
1052 LogFlowFunc(("received empty clipboard data from the guest, returning false.\n"));
1053 vboxClipboardEmptyGuestBuffer();
1054 return false;
1055 }
1056 pu16DestText = reinterpret_cast<PRTUTF16>(RTMemAlloc(cwDestLen * 2));
1057 if (pu16DestText == 0)
1058 {
1059 LogRel (("vboxClipboardConvertUtf8: failed to allocate %d bytes\n", cwDestLen * 2));
1060 vboxClipboardEmptyGuestBuffer();
1061 return false;
1062 }
1063 /* Convert the text. */
1064 rc = vboxClipboardUtf16WinToLin(pu16SrcText, cwSrcLen, pu16DestText, cwDestLen);
1065 if (RT_FAILURE(rc))
1066 {
1067 LogRel(("vboxClipboardConvertUtf8: clipboard conversion failed. vboxClipboardUtf16WinToLin() returned %Vrc. Abandoning.\n", rc));
1068 RTMemFree(reinterpret_cast<void *>(pu16DestText));
1069 vboxClipboardEmptyGuestBuffer();
1070 return false;
1071 }
1072 /* Allocate enough space, as RTUtf16ToUtf8Ex may fail if the
1073 space is too tightly calculated. */
1074 pu8DestText = XtMalloc(cwDestLen * 4);
1075 if (pu8DestText == 0)
1076 {
1077 LogRel (("vboxClipboardConvertUtf8: failed to allocate %d bytes\n", cwDestLen * 4));
1078 RTMemFree(reinterpret_cast<void *>(pu16DestText));
1079 vboxClipboardEmptyGuestBuffer();
1080 return false;
1081 }
1082 /* Convert the Utf16 string to Utf8. */
1083 rc = RTUtf16ToUtf8Ex(pu16DestText + 1, cwDestLen - 1, &pu8DestText, cwDestLen * 4,
1084 &cbDestLen);
1085 RTMemFree(reinterpret_cast<void *>(pu16DestText));
1086 if (RT_FAILURE(rc))
1087 {
1088 LogRel(("vboxClipboardConvertUtf8: clipboard conversion failed. RTUtf16ToUtf8Ex() returned %Vrc. Abandoning.\n", rc));
1089 XtFree(pu8DestText);
1090 vboxClipboardEmptyGuestBuffer();
1091 return false;
1092 }
1093 LogFlowFunc (("converted string is %.*s. Returning.\n", cbDestLen, pu8DestText));
1094 vboxClipboardEmptyGuestBuffer();
1095 *atomTypeReturn = g_ctx.atomUtf8;
1096 *pValReturn = reinterpret_cast<XtPointer>(pu8DestText);
1097 *pcLenReturn = cbDestLen;
1098 *piFormatReturn = 8;
1099 return true;
1100}
1101
1102/**
1103 * Satisfy a request from the host to convert the clipboard text to COMPOUND_TEXT.
1104 *
1105 * @returns true if we successfully convert the data to the format requested, false otherwise.
1106 *
1107 * @param atomTypeReturn The type of the data we are returning
1108 * @param pValReturn A pointer to the data we are returning. This should be to memory
1109 * allocated by XtMalloc, which will be freed by the toolkit later
1110 * @param pcLenReturn The length of the data we are returning
1111 * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are returning
1112 */
1113static Boolean vboxClipboardConvertCText(Atom *atomTypeReturn, XtPointer *pValReturn,
1114 unsigned long *pcLenReturn, int *piFormatReturn)
1115{
1116 PRTUTF16 pu16SrcText, pu16DestText;
1117 char *pu8DestText = 0;
1118 size_t cwSrcLen, cwDestLen, cbDestLen;
1119 XTextProperty property;
1120 int rc;
1121
1122 LogFlowFunc (("called\n"));
1123 /* Read the clipboard data from the guest. */
1124 rc = vboxClipboardReadDataFromClient(&g_ctx, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
1125 if ((rc != VINF_SUCCESS) || (g_ctx.pClient->data.cb == 0))
1126 {
1127 /* If vboxClipboardReadDataFromClient fails then pClient may be invalid */
1128 LogRelFunc (("vboxClipboardReadDataFromClient returned %Rrc%s\n", rc,
1129 RT_SUCCESS(rc) ? ", g_ctx.pClient->data.cb == 0" : ""));
1130 vboxClipboardEmptyGuestBuffer();
1131 return false;
1132 }
1133 pu16SrcText = reinterpret_cast<PRTUTF16>(g_ctx.pClient->data.pv);
1134 cwSrcLen = g_ctx.pClient->data.cb / 2;
1135 /* How long will the converted text be? */
1136 rc = vboxClipboardUtf16GetLinSize(pu16SrcText, cwSrcLen, &cwDestLen);
1137 if (RT_FAILURE(rc))
1138 {
1139 LogRel(("vboxClipboardConvertCText: clipboard conversion failed. vboxClipboardUtf16GetLinSize returned %Vrc. Abandoning.\n", rc));
1140 vboxClipboardEmptyGuestBuffer();
1141 AssertRCReturn(rc, false);
1142 }
1143 if (cwDestLen == 0)
1144 {
1145 LogFlowFunc(("received empty clipboard data from the guest, returning false.\n"));
1146 vboxClipboardEmptyGuestBuffer();
1147 return false;
1148 }
1149 pu16DestText = reinterpret_cast<PRTUTF16>(RTMemAlloc(cwDestLen * 2));
1150 if (pu16DestText == 0)
1151 {
1152 LogRel (("vboxClipboardConvertCText: failed to allocate %d bytes\n", cwDestLen * 2));
1153 vboxClipboardEmptyGuestBuffer();
1154 return false;
1155 }
1156 /* Convert the text. */
1157 rc = vboxClipboardUtf16WinToLin(pu16SrcText, cwSrcLen, pu16DestText, cwDestLen);
1158 if (RT_FAILURE(rc))
1159 {
1160 LogRel(("vboxClipboardConvertCText: clipboard conversion failed. vboxClipboardUtf16WinToLin() returned %Vrc. Abandoning.\n", rc));
1161 RTMemFree(reinterpret_cast<void *>(pu16DestText));
1162 vboxClipboardEmptyGuestBuffer();
1163 return false;
1164 }
1165 /* Convert the Utf16 string to Utf8. */
1166 rc = RTUtf16ToUtf8Ex(pu16DestText + 1, cwDestLen - 1, &pu8DestText, 0, &cbDestLen);
1167 RTMemFree(reinterpret_cast<void *>(pu16DestText));
1168 if (RT_FAILURE(rc))
1169 {
1170 LogRel(("vboxClipboardConvertCText: clipboard conversion failed. RTUtf16ToUtf8Ex() returned %Vrc. Abandoning.\n", rc));
1171 vboxClipboardEmptyGuestBuffer();
1172 return false;
1173 }
1174 /* And finally (!) convert the Utf8 text to compound text. */
1175#ifdef RT_OS_SOLARIS
1176 rc = XmbTextListToTextProperty(XtDisplay(g_ctx.widget), &pu8DestText, 1,
1177 XCompoundTextStyle, &property);
1178#else
1179 rc = Xutf8TextListToTextProperty(XtDisplay(g_ctx.widget), &pu8DestText, 1,
1180 XCompoundTextStyle, &property);
1181#endif
1182 RTMemFree(pu8DestText);
1183 if (rc < 0)
1184 {
1185 const char *pcReason;
1186 switch(rc)
1187 {
1188 case XNoMemory:
1189 pcReason = "out of memory";
1190 break;
1191 case XLocaleNotSupported:
1192 pcReason = "locale (Utf8) not supported";
1193 break;
1194 case XConverterNotFound:
1195 pcReason = "converter not found";
1196 break;
1197 default:
1198 pcReason = "unknown error";
1199 }
1200 LogRel(("vboxClipboardConvertCText: Xutf8TextListToTextProperty failed. Reason: %s\n",
1201 pcReason));
1202 XFree(property.value);
1203 vboxClipboardEmptyGuestBuffer();
1204 return false;
1205 }
1206 LogFlowFunc (("converted string is %s. Returning.\n", property.value));
1207 vboxClipboardEmptyGuestBuffer();
1208 *atomTypeReturn = property.encoding;
1209 *pValReturn = reinterpret_cast<XtPointer>(property.value);
1210 *pcLenReturn = property.nitems;
1211 *piFormatReturn = property.format;
1212 return true;
1213}
1214
1215/**
1216 * Callback to convert the guests clipboard data for an application on the host. Called by the
1217 * X Toolkit.
1218 * @returns true if we successfully convert the data to the format requested, false otherwise.
1219 *
1220 * @param atomSelection The selection which is being requested. We only handle the clipboard.
1221 * @param atomTarget The format we should convert the data to
1222 * @param atomTypeReturn The type of the data we are returning
1223 * @param pValReturn A pointer to the data we are returning. This should be to memory
1224 * allocated by XtMalloc, which will be freed by the toolkit later
1225 * @param pcLenReturn The length of the data we are returning
1226 * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are returning
1227 */
1228static Boolean vboxClipboardConvertProc(Widget, Atom *atomSelection, Atom *atomTarget,
1229 Atom *atomTypeReturn, XtPointer *pValReturn,
1230 unsigned long *pcLenReturn, int *piFormatReturn)
1231{
1232 g_eClipboardFormats eFormat = INVALID;
1233
1234 LogFlowFunc(("\n"));
1235 if ( (*atomSelection != g_ctx.atomClipboard)
1236 && (*atomSelection != g_ctx.atomPrimary)
1237 )
1238 {
1239 LogFlowFunc(("rc = false\n"));
1240 return false;
1241 }
1242#ifdef DEBUG
1243 char *szAtomName = XGetAtomName(XtDisplay(g_ctx.widget), *atomTarget);
1244 if (szAtomName != 0)
1245 {
1246 Log2 (("vboxClipboardConvertProc: request for format %s\n", szAtomName));
1247 XFree(szAtomName);
1248 }
1249 else
1250 {
1251 Log(("vboxClipboardConvertProc: request for invalid target atom %d!\n", *atomTarget));
1252 }
1253#endif
1254 if (*atomTarget == g_ctx.atomTargets)
1255 {
1256 eFormat = TARGETS;
1257 }
1258 else
1259 {
1260 for (unsigned i = 0; i != g_ctx.formatList.size(); ++i)
1261 {
1262 if (g_ctx.formatList[i].atom == *atomTarget)
1263 {
1264 eFormat = g_ctx.formatList[i].format;
1265 break;
1266 }
1267 }
1268 }
1269 switch (eFormat)
1270 {
1271 case TARGETS:
1272 return vboxClipboardConvertTargets(atomTypeReturn, pValReturn, pcLenReturn,
1273 piFormatReturn);
1274 case UTF16:
1275 return vboxClipboardConvertUtf16(atomTypeReturn, pValReturn, pcLenReturn,
1276 piFormatReturn);
1277 case UTF8:
1278 return vboxClipboardConvertUtf8(atomTypeReturn, pValReturn, pcLenReturn,
1279 piFormatReturn);
1280 case CTEXT:
1281 return vboxClipboardConvertCText(atomTypeReturn, pValReturn, pcLenReturn,
1282 piFormatReturn);
1283 default:
1284 Log(("vboxClipboardConvertProc: bad format\n"));
1285 return false;
1286 }
1287}
1288
1289static void vboxClipboardLoseProc(Widget, Atom *)
1290{
1291 LogFlow (("vboxClipboardLoseProc: called, giving the host clipboard ownership\n"));
1292 g_ctx.eOwner = HOST;
1293 g_ctx.notifyGuest = true;
1294}
1295
1296/**
1297 * The guest is taking possession of the shared clipboard. Called by the HGCM clipboard
1298 * subsystem.
1299 *
1300 * @param pClient Context data for the guest system
1301 * @param u32Formats Clipboard formats the the guest is offering
1302 */
1303void vboxClipboardFormatAnnounce (VBOXCLIPBOARDCLIENTDATA *pClient, uint32_t u32Formats)
1304{
1305 pClient->pCtx->guestFormats = u32Formats;
1306 LogFlowFunc (("u32Formats=%d\n", u32Formats));
1307 if (u32Formats == 0)
1308 {
1309 /* This is just an automatism, not a genuine anouncement */
1310 LogFlowFunc(("returning\n"));
1311 return;
1312 }
1313 if (g_ctx.eOwner == GUEST)
1314 {
1315 /* We already own the clipboard, so no need to grab it, especially as that can lead
1316 to races due to the asynchronous nature of the X11 clipboard. This event may also
1317 have been sent out by the guest to invalidate the Windows clipboard cache. */
1318 LogFlowFunc(("returning\n"));
1319 return;
1320 }
1321 Log2 (("vboxClipboardFormatAnnounce: giving the guest clipboard ownership\n"));
1322 g_ctx.eOwner = GUEST;
1323 g_ctx.hostTextFormat = INVALID;
1324 g_ctx.hostBitmapFormat = INVALID;
1325 if (XtOwnSelection(g_ctx.widget, g_ctx.atomClipboard, CurrentTime, vboxClipboardConvertProc,
1326 vboxClipboardLoseProc, 0) != True)
1327 {
1328 Log2 (("vboxClipboardFormatAnnounce: returning clipboard ownership to the host\n"));
1329 /* We set this so that the guest gets notified when we take the clipboard, even if no
1330 guest formats are found which we understand. */
1331 g_ctx.notifyGuest = true;
1332 g_ctx.eOwner = HOST;
1333 }
1334 XtOwnSelection(g_ctx.widget, g_ctx.atomPrimary, CurrentTime, vboxClipboardConvertProc,
1335 NULL, 0);
1336 LogFlowFunc(("returning\n"));
1337
1338}
1339
1340/**
1341 * Called by the HGCM clipboard subsystem when the guest wants to read the host clipboard.
1342 *
1343 * @param pClient Context information about the guest VM
1344 * @param u32Format The format that the guest would like to receive the data in
1345 * @param pv Where to write the data to
1346 * @param cb The size of the buffer to write the data to
1347 * @param pcbActual Where to write the actual size of the written data
1348 */
1349int vboxClipboardReadData (VBOXCLIPBOARDCLIENTDATA *pClient, uint32_t u32Format, void *pv,
1350 uint32_t cb, uint32_t *pcbActual)
1351{
1352 LogFlow(("vboxClipboardReadData: u32Format = %d, cb = %d\n", u32Format, cb));
1353
1354 /*
1355 * The guest wants to read data in the given format.
1356 */
1357 if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
1358 {
1359 if (g_ctx.hostTextFormat == INVALID)
1360 {
1361 /* No data available. */
1362 *pcbActual = 0;
1363 return VINF_SUCCESS;
1364 }
1365 /* Only one of the host and the guest should ever be waiting. */
1366 if (RT_FAILURE(ASMAtomicCmpXchgU32(&g_ctx.waiter, 1, 0)))
1367 {
1368 LogRel(("vboxClipboardReadData: detected a deadlock situation - the host and the guest are waiting for each other.\n"));
1369 return VERR_DEADLOCK;
1370 }
1371 g_ctx.requestHostFormat = g_ctx.hostTextFormat;
1372 g_ctx.requestBuffer = pv;
1373 g_ctx.requestBufferSize = cb;
1374 g_ctx.requestActualSize = pcbActual;
1375 /* Send out a request for the data to the current clipboard owner */
1376 XtGetSelectionValue(g_ctx.widget, g_ctx.atomClipboard, g_ctx.atomHostTextFormat,
1377 vboxClipboardGetProc, reinterpret_cast<XtPointer>(g_ctx.pClient),
1378 CurrentTime);
1379 /* When the data arrives, the vboxClipboardGetProc callback will be called. The
1380 callback will signal the event semaphore when it has processed the data for us. */
1381 if (RTSemEventWait(g_ctx.waitForData, CLIPBOARDTIMEOUT) != VINF_SUCCESS)
1382 {
1383 LogRel (("vboxClipboardReadDataFromClient: XtGetSelectionValue failed to complete within %d milliseconds\n", CLIPBOARDTIMEOUT));
1384 g_ctx.hostTextFormat = INVALID;
1385 g_ctx.hostBitmapFormat = INVALID;
1386 g_ctx.waiter = 0;
1387 return VERR_TIMEOUT;
1388 }
1389 g_ctx.waiter = 0;
1390 }
1391 else
1392 {
1393 return VERR_NOT_IMPLEMENTED;
1394 }
1395 return VINF_SUCCESS;
1396}
1397
1398/**
1399 * Called by the HGCM clipboard subsystem when we have requested data and that data arrives.
1400 *
1401 * @param pClient Context information about the guest VM
1402 * @param pv Buffer to which the data was written
1403 * @param cb The size of the data written
1404 * @param u32Format The format of the data written
1405 */
1406void vboxClipboardWriteData (VBOXCLIPBOARDCLIENTDATA *pClient, void *pv, uint32_t cb, uint32_t u32Format)
1407{
1408 LogFlow(("vboxClipboardWriteData\n"));
1409
1410 /*
1411 * The guest returns data that was requested in the WM_RENDERFORMAT handler.
1412 */
1413 if (!( pClient->data.pv == NULL
1414 && pClient->data.cb == 0
1415 && pClient->data.u32Format == 0))
1416 {
1417 LogRel(("vboxClipboardWriteData: clipboard data has arrived from the guest, but another transfer is in process or has not cleaned up properly.\n"));
1418 AssertMsgFailed(("vboxClipboardWriteData: clipboard data has arrived from the guest, but another transfer is in process or has not cleaned up properly.\n"));
1419 }
1420
1421 if (cb > 0)
1422 {
1423 pClient->data.pv = RTMemAlloc (cb);
1424
1425 if (pClient->data.pv)
1426 {
1427 memcpy (pClient->data.pv, pv, cb);
1428 pClient->data.cb = cb;
1429 pClient->data.u32Format = u32Format;
1430 }
1431 }
1432
1433 RTSemEventSignal(g_ctx.waitForData);
1434}
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