VirtualBox

source: vbox/trunk/src/VBox/HostServices/SharedClipboard/VBoxClipboard-win.cpp@ 61522

Last change on this file since 61522 was 60784, checked in by vboxsync, 9 years ago

SharedClipboard: vboxOpenClipboard cleanup

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.4 KB
Line 
1/** @file
2 * Shared Clipboard: Win32 host.
3 */
4
5/*
6 * Copyright (C) 2006-2015 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#include <windows.h>
18
19#include <VBox/HostServices/VBoxClipboardSvc.h>
20
21#include <iprt/alloc.h>
22#include <iprt/string.h>
23#include <iprt/asm.h>
24#include <iprt/assert.h>
25#include <iprt/thread.h>
26#include <iprt/ldr.h>
27#include <process.h>
28
29#include "VBoxClipboard.h"
30
31#define dprintf Log
32
33static char gachWindowClassName[] = "VBoxSharedClipboardClass";
34
35enum { CBCHAIN_TIMEOUT = 5000 /* ms */ };
36
37/* Dynamically load clipboard functions from User32.dll. */
38typedef BOOL WINAPI FNADDCLIPBOARDFORMATLISTENER(HWND);
39typedef FNADDCLIPBOARDFORMATLISTENER *PFNADDCLIPBOARDFORMATLISTENER;
40
41typedef BOOL WINAPI FNREMOVECLIPBOARDFORMATLISTENER(HWND);
42typedef FNREMOVECLIPBOARDFORMATLISTENER *PFNREMOVECLIPBOARDFORMATLISTENER;
43
44#ifndef WM_CLIPBOARDUPDATE
45#define WM_CLIPBOARDUPDATE 0x031D
46#endif
47
48struct _VBOXCLIPBOARDCONTEXT
49{
50 HWND hwnd;
51 HWND hwndNextInChain;
52
53 UINT timerRefresh;
54
55 bool fCBChainPingInProcess;
56
57 RTTHREAD thread;
58 bool volatile fTerminate;
59
60 HANDLE hRenderEvent;
61
62 VBOXCLIPBOARDCLIENTDATA *pClient;
63
64 PFNADDCLIPBOARDFORMATLISTENER pfnAddClipboardFormatListener;
65 PFNREMOVECLIPBOARDFORMATLISTENER pfnRemoveClipboardFormatListener;
66
67};
68
69/* Only one client is supported. There seems to be no need for more clients. */
70static VBOXCLIPBOARDCONTEXT g_ctx;
71
72
73#ifdef LOG_ENABLED
74void vboxClipboardDump(const void *pv, size_t cb, uint32_t u32Format)
75{
76 if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
77 {
78 Log(("DUMP: VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT:\n"));
79 if (pv && cb)
80 {
81 Log(("%ls\n", pv));
82 }
83 else
84 {
85 Log(("%p %d\n", pv, cb));
86 }
87 }
88 else if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
89 {
90 dprintf(("DUMP: VBOX_SHARED_CLIPBOARD_FMT_BITMAP\n"));
91 }
92 else if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_HTML)
93 {
94 Log(("DUMP: VBOX_SHARED_CLIPBOARD_FMT_HTML:\n"));
95 if (pv && cb)
96 {
97 Log(("%s\n", pv));
98 }
99 else
100 {
101 Log(("%p %d\n", pv, cb));
102 }
103 }
104 else
105 {
106 dprintf(("DUMP: invalid format %02X\n", u32Format));
107 }
108}
109#else
110#define vboxClipboardDump(__pv, __cb, __format) do { NOREF(__pv); NOREF(__cb); NOREF(__format); } while (0)
111#endif /* LOG_ENABLED */
112
113
114static void vboxClipboardInitNewAPI(VBOXCLIPBOARDCONTEXT *pCtx)
115{
116 RTLDRMOD hUser32 = NIL_RTLDRMOD;
117 int rc = RTLdrLoadSystem("User32.dll", /* fNoUnload = */ true, &hUser32);
118 if (RT_SUCCESS(rc))
119 {
120 rc = RTLdrGetSymbol(hUser32, "AddClipboardFormatListener", (void**)&pCtx->pfnAddClipboardFormatListener);
121 if (RT_SUCCESS(rc))
122 {
123 rc = RTLdrGetSymbol(hUser32, "RemoveClipboardFormatListener", (void**)&pCtx->pfnRemoveClipboardFormatListener);
124 }
125
126 RTLdrClose(hUser32);
127 }
128
129 if (RT_SUCCESS(rc))
130 {
131 Log(("New Clipboard API is enabled\n"));
132 }
133 else
134 {
135 pCtx->pfnAddClipboardFormatListener = NULL;
136 pCtx->pfnRemoveClipboardFormatListener = NULL;
137 Log(("New Clipboard API is not available. rc = %Rrc\n", rc));
138 }
139}
140
141static bool vboxClipboardIsNewAPI(VBOXCLIPBOARDCONTEXT *pCtx)
142{
143 return pCtx->pfnAddClipboardFormatListener != NULL;
144}
145
146
147static int vboxOpenClipboard(HWND hwnd)
148{
149 /* "OpenClipboard fails if another window has the clipboard open."
150 * So try a few times and wait up to 1 second.
151 */
152 BOOL fOpened = FALSE;
153
154 int i = 0;
155 for (;;)
156 {
157 if (OpenClipboard(hwnd))
158 {
159 fOpened = TRUE;
160 break;
161 }
162
163 if (i >= 10) /* sleep interval = [1..512] ms */
164 break;
165
166 RTThreadSleep(1 << i);
167 ++i;
168 }
169
170#ifdef LOG_ENABLED
171 if (i > 0)
172 LogFlowFunc(("%d times tried to open clipboard.\n", i + 1));
173#endif
174
175 int rc;
176 if (fOpened)
177 rc = VINF_SUCCESS;
178 else
179 {
180 const DWORD err = GetLastError();
181 LogFlowFunc(("error %d\n", err));
182 rc = RTErrConvertFromWin32(err);
183 }
184
185 return rc;
186}
187
188
189static void vboxClipboardGetData (uint32_t u32Format, const void *pvSrc, uint32_t cbSrc,
190 void *pvDst, uint32_t cbDst, uint32_t *pcbActualDst)
191{
192 dprintf (("vboxClipboardGetData.\n"));
193
194 *pcbActualDst = cbSrc;
195
196 LogFlow(("vboxClipboardGetData cbSrc = %d, cbDst = %d\n", cbSrc, cbDst));
197
198 if (cbSrc > cbDst)
199 {
200 /* Do not copy data. The dst buffer is not enough. */
201 return;
202 }
203
204 memcpy (pvDst, pvSrc, cbSrc);
205
206 vboxClipboardDump(pvDst, cbSrc, u32Format);
207
208 return;
209}
210
211static int vboxClipboardReadDataFromClient (VBOXCLIPBOARDCONTEXT *pCtx, uint32_t u32Format)
212{
213 Assert(pCtx->pClient);
214 Assert(pCtx->hRenderEvent);
215 Assert(pCtx->pClient->data.pv == NULL && pCtx->pClient->data.cb == 0 && pCtx->pClient->data.u32Format == 0);
216
217 LogFlow(("vboxClipboardReadDataFromClient u32Format = %02X\n", u32Format));
218
219 ResetEvent (pCtx->hRenderEvent);
220
221 vboxSvcClipboardReportMsg (pCtx->pClient, VBOX_SHARED_CLIPBOARD_HOST_MSG_READ_DATA, u32Format);
222
223 DWORD ret = WaitForSingleObject(pCtx->hRenderEvent, INFINITE);
224 LogFlow(("vboxClipboardReadDataFromClient wait completed, ret 0x%08X, err %d\n",
225 ret, GetLastError())); NOREF(ret);
226
227 return VINF_SUCCESS;
228}
229
230static void vboxClipboardChanged (VBOXCLIPBOARDCONTEXT *pCtx)
231{
232 LogFlow(("vboxClipboardChanged\n"));
233
234 if (pCtx->pClient == NULL)
235 {
236 return;
237 }
238
239 /* Query list of available formats and report to host. */
240 int rc = vboxOpenClipboard(pCtx->hwnd);
241 if (RT_SUCCESS(rc))
242 {
243 uint32_t u32Formats = 0;
244
245 UINT format = 0;
246
247 while ((format = EnumClipboardFormats (format)) != 0)
248 {
249 LogFlow(("vboxClipboardChanged format %#x\n", format));
250 switch (format)
251 {
252 case CF_UNICODETEXT:
253 case CF_TEXT:
254 u32Formats |= VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT;
255 break;
256
257 case CF_DIB:
258 case CF_BITMAP:
259 u32Formats |= VBOX_SHARED_CLIPBOARD_FMT_BITMAP;
260 break;
261
262 default:
263 if (format >= 0xC000)
264 {
265 TCHAR szFormatName[256];
266
267 int cActual = GetClipboardFormatName(format, szFormatName, sizeof(szFormatName)/sizeof (TCHAR));
268
269 if (cActual)
270 {
271 if (strcmp (szFormatName, "HTML Format") == 0)
272 {
273 u32Formats |= VBOX_SHARED_CLIPBOARD_FMT_HTML;
274 }
275 }
276 }
277 break;
278 }
279 }
280
281 CloseClipboard ();
282
283 LogFlow(("vboxClipboardChanged u32Formats %02X\n", u32Formats));
284
285 vboxSvcClipboardReportMsg (pCtx->pClient, VBOX_SHARED_CLIPBOARD_HOST_MSG_FORMATS, u32Formats);
286 }
287 else
288 {
289 LogFlow(("vboxClipboardChanged: error in open clipboard. hwnd: %x. err: %Rrc\n", pCtx->hwnd, rc));
290 }
291}
292
293/* Add ourselves into the chain of cliboard listeners */
294static void addToCBChain (VBOXCLIPBOARDCONTEXT *pCtx)
295{
296 if (vboxClipboardIsNewAPI(pCtx))
297 pCtx->pfnAddClipboardFormatListener(pCtx->hwnd);
298 else
299 pCtx->hwndNextInChain = SetClipboardViewer(pCtx->hwnd);
300}
301
302/* Remove ourselves from the chain of cliboard listeners */
303static void removeFromCBChain (VBOXCLIPBOARDCONTEXT *pCtx)
304{
305 if (vboxClipboardIsNewAPI(pCtx))
306 {
307 pCtx->pfnRemoveClipboardFormatListener(pCtx->hwnd);
308 }
309 else
310 {
311 ChangeClipboardChain(pCtx->hwnd, pCtx->hwndNextInChain);
312 pCtx->hwndNextInChain = NULL;
313 }
314}
315
316/* Callback which is invoked when we have successfully pinged ourselves down the
317 * clipboard chain. We simply unset a boolean flag to say that we are responding.
318 * There is a race if a ping returns after the next one is initiated, but nothing
319 * very bad is likely to happen. */
320VOID CALLBACK CBChainPingProc(HWND hwnd, UINT uMsg, ULONG_PTR dwData, LRESULT lResult)
321{
322 (void) hwnd;
323 (void) uMsg;
324 (void) lResult;
325 VBOXCLIPBOARDCONTEXT *pCtx = (VBOXCLIPBOARDCONTEXT *)dwData;
326 pCtx->fCBChainPingInProcess = FALSE;
327}
328
329static LRESULT CALLBACK vboxClipboardWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
330{
331 LRESULT rc = 0;
332
333 VBOXCLIPBOARDCONTEXT *pCtx = &g_ctx;
334
335 switch (msg)
336 {
337 case WM_CLIPBOARDUPDATE:
338 {
339 Log(("WM_CLIPBOARDUPDATE\n"));
340
341 if (GetClipboardOwner() != hwnd)
342 {
343 /* Clipboard was updated by another application. */
344 vboxClipboardChanged(pCtx);
345 }
346 } break;
347
348 case WM_CHANGECBCHAIN:
349 {
350 Log(("WM_CHANGECBCHAIN\n"));
351
352 if (vboxClipboardIsNewAPI(pCtx))
353 {
354 rc = DefWindowProc(hwnd, msg, wParam, lParam);
355 break;
356 }
357
358 HWND hwndRemoved = (HWND)wParam;
359 HWND hwndNext = (HWND)lParam;
360
361 if (hwndRemoved == pCtx->hwndNextInChain)
362 {
363 /* The window that was next to our in the chain is being removed.
364 * Relink to the new next window.
365 */
366 pCtx->hwndNextInChain = hwndNext;
367 }
368 else
369 {
370 if (pCtx->hwndNextInChain)
371 {
372 /* Pass the message further. */
373 DWORD_PTR dwResult;
374 rc = SendMessageTimeout(pCtx->hwndNextInChain, WM_CHANGECBCHAIN, wParam, lParam, 0, CBCHAIN_TIMEOUT, &dwResult);
375 if (!rc)
376 rc = (LRESULT)dwResult;
377 }
378 }
379 } break;
380
381 case WM_DRAWCLIPBOARD:
382 {
383 Log(("WM_DRAWCLIPBOARD\n"));
384
385 if (GetClipboardOwner () != hwnd)
386 {
387 /* Clipboard was updated by another application. */
388 vboxClipboardChanged (pCtx);
389 }
390
391 if (pCtx->hwndNextInChain)
392 {
393 Log(("WM_DRAWCLIPBOARD next %p\n", pCtx->hwndNextInChain));
394 /* Pass the message to next windows in the clipboard chain. */
395 DWORD_PTR dwResult;
396 rc = SendMessageTimeout(pCtx->hwndNextInChain, msg, wParam, lParam, 0, CBCHAIN_TIMEOUT, &dwResult);
397 if (!rc)
398 rc = dwResult;
399 }
400 } break;
401
402 case WM_TIMER:
403 {
404 if (vboxClipboardIsNewAPI(pCtx))
405 break;
406
407 HWND hViewer = GetClipboardViewer();
408
409 /* Re-register ourselves in the clipboard chain if our last ping
410 * timed out or there seems to be no valid chain. */
411 if (!hViewer || pCtx->fCBChainPingInProcess)
412 {
413 removeFromCBChain(pCtx);
414 addToCBChain(pCtx);
415 }
416 /* Start a new ping by passing a dummy WM_CHANGECBCHAIN to be
417 * processed by ourselves to the chain. */
418 pCtx->fCBChainPingInProcess = TRUE;
419 hViewer = GetClipboardViewer();
420 if (hViewer)
421 SendMessageCallback(hViewer, WM_CHANGECBCHAIN, (WPARAM)pCtx->hwndNextInChain, (LPARAM)pCtx->hwndNextInChain, CBChainPingProc, (ULONG_PTR) pCtx);
422 } break;
423
424 case WM_RENDERFORMAT:
425 {
426 /* Insert the requested clipboard format data into the clipboard. */
427 uint32_t u32Format = 0;
428
429 UINT format = (UINT)wParam;
430
431 Log(("WM_RENDERFORMAT %d\n", format));
432
433 switch (format)
434 {
435 case CF_UNICODETEXT:
436 u32Format |= VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT;
437 break;
438
439 case CF_DIB:
440 u32Format |= VBOX_SHARED_CLIPBOARD_FMT_BITMAP;
441 break;
442
443 default:
444 if (format >= 0xC000)
445 {
446 TCHAR szFormatName[256];
447
448 int cActual = GetClipboardFormatName(format, szFormatName, sizeof(szFormatName)/sizeof (TCHAR));
449
450 if (cActual)
451 {
452 if (strcmp (szFormatName, "HTML Format") == 0)
453 {
454 u32Format |= VBOX_SHARED_CLIPBOARD_FMT_HTML;
455 }
456 }
457 }
458 break;
459 }
460
461 if (u32Format == 0 || pCtx->pClient == NULL)
462 {
463 /* Unsupported clipboard format is requested. */
464 Log(("WM_RENDERFORMAT unsupported format requested or client is not active.\n"));
465 EmptyClipboard ();
466 }
467 else
468 {
469 int vboxrc = vboxClipboardReadDataFromClient (pCtx, u32Format);
470
471 dprintf(("vboxClipboardReadDataFromClient vboxrc = %d, pv %p, cb %d, u32Format %d\n",
472 vboxrc, pCtx->pClient->data.pv, pCtx->pClient->data.cb, pCtx->pClient->data.u32Format));
473
474 if ( RT_SUCCESS (vboxrc)
475 && pCtx->pClient->data.pv != NULL
476 && pCtx->pClient->data.cb > 0
477 && pCtx->pClient->data.u32Format == u32Format)
478 {
479 HANDLE hMem = GlobalAlloc (GMEM_DDESHARE | GMEM_MOVEABLE, pCtx->pClient->data.cb);
480
481 dprintf(("hMem %p\n", hMem));
482
483 if (hMem)
484 {
485 void *pMem = GlobalLock (hMem);
486
487 dprintf(("pMem %p, GlobalSize %d\n", pMem, GlobalSize (hMem)));
488
489 if (pMem)
490 {
491 Log(("WM_RENDERFORMAT setting data\n"));
492
493 if (pCtx->pClient->data.pv)
494 {
495 memcpy (pMem, pCtx->pClient->data.pv, pCtx->pClient->data.cb);
496
497 RTMemFree (pCtx->pClient->data.pv);
498 pCtx->pClient->data.pv = NULL;
499 }
500
501 pCtx->pClient->data.cb = 0;
502 pCtx->pClient->data.u32Format = 0;
503
504 /* The memory must be unlocked before inserting to the Clipboard. */
505 GlobalUnlock (hMem);
506
507 /* 'hMem' contains the host clipboard data.
508 * size is 'cb' and format is 'format'.
509 */
510 HANDLE hClip = SetClipboardData (format, hMem);
511
512 dprintf(("vboxClipboardHostEvent hClip %p\n", hClip));
513
514 if (hClip)
515 {
516 /* The hMem ownership has gone to the system. Nothing to do. */
517 break;
518 }
519 }
520
521 GlobalFree (hMem);
522 }
523 }
524
525 RTMemFree (pCtx->pClient->data.pv);
526 pCtx->pClient->data.pv = NULL;
527 pCtx->pClient->data.cb = 0;
528 pCtx->pClient->data.u32Format = 0;
529
530 /* Something went wrong. */
531 EmptyClipboard ();
532 }
533 } break;
534
535 case WM_RENDERALLFORMATS:
536 {
537 Log(("WM_RENDERALLFORMATS\n"));
538
539 /* Do nothing. The clipboard formats will be unavailable now, because the
540 * windows is to be destroyed and therefore the guest side becomes inactive.
541 */
542 int vboxrc = vboxOpenClipboard(hwnd);
543 if (RT_SUCCESS(vboxrc))
544 {
545 EmptyClipboard();
546
547 CloseClipboard();
548 }
549 else
550 {
551 LogFlow(("vboxClipboardWndProc: WM_RENDERALLFORMATS: error in open clipboard. hwnd: %x, rc: %Rrc\n", hwnd, vboxrc));
552 }
553 } break;
554
555 case WM_USER:
556 {
557 if (pCtx->pClient == NULL || pCtx->pClient->fMsgFormats)
558 {
559 /* Host has pending formats message. Ignore the guest announcement,
560 * because host clipboard has more priority.
561 */
562 Log(("WM_USER ignored\n"));
563 break;
564 }
565
566 /* Announce available formats. Do not insert data, they will be inserted in WM_RENDER*. */
567 uint32_t u32Formats = (uint32_t)lParam;
568
569 Log(("WM_USER u32Formats = %02X\n", u32Formats));
570
571 int vboxrc = vboxOpenClipboard(hwnd);
572 if (RT_SUCCESS(vboxrc))
573 {
574 EmptyClipboard();
575
576 Log(("WM_USER emptied clipboard\n"));
577
578 HANDLE hClip = NULL;
579
580 if (u32Formats & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
581 {
582 dprintf(("window proc WM_USER: VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT\n"));
583
584 hClip = SetClipboardData (CF_UNICODETEXT, NULL);
585 }
586
587 if (u32Formats & VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
588 {
589 dprintf(("window proc WM_USER: VBOX_SHARED_CLIPBOARD_FMT_BITMAP\n"));
590
591 hClip = SetClipboardData (CF_DIB, NULL);
592 }
593
594 if (u32Formats & VBOX_SHARED_CLIPBOARD_FMT_HTML)
595 {
596 UINT format = RegisterClipboardFormat ("HTML Format");
597 dprintf(("window proc WM_USER: VBOX_SHARED_CLIPBOARD_FMT_HTML 0x%04X\n", format));
598 if (format != 0)
599 {
600 hClip = SetClipboardData (format, NULL);
601 }
602 }
603
604 CloseClipboard();
605
606 dprintf(("window proc WM_USER: hClip %p, err %d\n", hClip, GetLastError ()));
607 }
608 else
609 {
610 dprintf(("window proc WM_USER: failed to open clipboard. rc: %Rrc\n", vboxrc));
611 }
612 } break;
613
614 case WM_CLOSE:
615 {
616 /* Do nothing. Ignore the message. */
617 } break;
618
619 default:
620 {
621 Log(("WM_ %p\n", msg));
622 rc = DefWindowProc(hwnd, msg, wParam, lParam);
623 }
624 }
625
626 Log(("WM_ rc %d\n", rc));
627 return rc;
628}
629
630DECLCALLBACK(int) VBoxClipboardThread (RTTHREAD ThreadSelf, void *pInstance)
631{
632 /* Create a window and make it a clipboard viewer. */
633 int rc = VINF_SUCCESS;
634
635 LogFlow(("VBoxClipboardThread\n"));
636
637 VBOXCLIPBOARDCONTEXT *pCtx = &g_ctx;
638
639 HINSTANCE hInstance = (HINSTANCE)GetModuleHandle(NULL);
640
641 /* Register the Window Class. */
642 WNDCLASS wc;
643 RT_ZERO(wc);
644
645 wc.style = CS_NOCLOSE;
646 wc.lpfnWndProc = vboxClipboardWndProc;
647 wc.hInstance = hInstance;
648 wc.hbrBackground = (HBRUSH)(COLOR_BACKGROUND + 1);
649 wc.lpszClassName = gachWindowClassName;
650
651 ATOM atomWindowClass = RegisterClass(&wc);
652
653 if (atomWindowClass == 0)
654 {
655 Log(("Failed to register window class\n"));
656 rc = VERR_NOT_SUPPORTED;
657 }
658 else
659 {
660 /* Create the window. */
661 pCtx->hwnd = CreateWindowEx (WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT | WS_EX_TOPMOST,
662 gachWindowClassName, gachWindowClassName,
663 WS_POPUPWINDOW,
664 -200, -200, 100, 100, NULL, NULL, hInstance, NULL);
665
666 if (pCtx->hwnd == NULL)
667 {
668 Log(("Failed to create window\n"));
669 rc = VERR_NOT_SUPPORTED;
670 }
671 else
672 {
673 SetWindowPos(pCtx->hwnd, HWND_TOPMOST, -200, -200, 0, 0,
674 SWP_NOACTIVATE | SWP_HIDEWINDOW | SWP_NOCOPYBITS | SWP_NOREDRAW | SWP_NOSIZE);
675
676 addToCBChain(pCtx);
677 if (!vboxClipboardIsNewAPI(pCtx))
678 pCtx->timerRefresh = SetTimer(pCtx->hwnd, 0, 10 * 1000, NULL);
679
680 MSG msg;
681 while (GetMessage(&msg, NULL, 0, 0) && !pCtx->fTerminate)
682 {
683 TranslateMessage(&msg);
684 DispatchMessage(&msg);
685 }
686 }
687 }
688
689 if (pCtx->hwnd)
690 {
691 removeFromCBChain(pCtx);
692 if (pCtx->timerRefresh)
693 KillTimer(pCtx->hwnd, 0);
694
695 DestroyWindow (pCtx->hwnd);
696 pCtx->hwnd = NULL;
697 }
698
699 if (atomWindowClass != 0)
700 {
701 UnregisterClass (gachWindowClassName, hInstance);
702 atomWindowClass = 0;
703 }
704
705 return 0;
706}
707
708/*
709 * Public platform dependent functions.
710 */
711int vboxClipboardInit (void)
712{
713 int rc = VINF_SUCCESS;
714
715 RT_ZERO(g_ctx);
716
717 /* Check that new Clipboard API is available */
718 vboxClipboardInitNewAPI(&g_ctx);
719
720 g_ctx.hRenderEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
721
722 rc = RTThreadCreate (&g_ctx.thread, VBoxClipboardThread, NULL, 65536,
723 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "SHCLIP");
724
725 if (RT_FAILURE (rc))
726 {
727 CloseHandle (g_ctx.hRenderEvent);
728 }
729
730 return rc;
731}
732
733void vboxClipboardDestroy (void)
734{
735 Log(("vboxClipboardDestroy\n"));
736
737 /* Set the termination flag and ping the window thread. */
738 ASMAtomicWriteBool (&g_ctx.fTerminate, true);
739
740 if (g_ctx.hwnd)
741 {
742 PostMessage (g_ctx.hwnd, WM_CLOSE, 0, 0);
743 }
744
745 CloseHandle (g_ctx.hRenderEvent);
746
747 /* Wait for the window thread to terminate. */
748 RTThreadWait (g_ctx.thread, RT_INDEFINITE_WAIT, NULL);
749
750 g_ctx.thread = NIL_RTTHREAD;
751}
752
753int vboxClipboardConnect (VBOXCLIPBOARDCLIENTDATA *pClient, bool)
754{
755 Log(("vboxClipboardConnect\n"));
756
757 if (g_ctx.pClient != NULL)
758 {
759 /* One client only. */
760 return VERR_NOT_SUPPORTED;
761 }
762
763 pClient->pCtx = &g_ctx;
764
765 pClient->pCtx->pClient = pClient;
766
767 /* Sync the host clipboard content with the client. */
768 vboxClipboardSync (pClient);
769
770 return VINF_SUCCESS;
771}
772
773int vboxClipboardSync (VBOXCLIPBOARDCLIENTDATA *pClient)
774{
775 /* Sync the host clipboard content with the client. */
776 vboxClipboardChanged (pClient->pCtx);
777
778 return VINF_SUCCESS;
779}
780
781void vboxClipboardDisconnect (VBOXCLIPBOARDCLIENTDATA *pClient)
782{
783 Log(("vboxClipboardDisconnect\n"));
784
785 g_ctx.pClient = NULL;
786}
787
788void vboxClipboardFormatAnnounce (VBOXCLIPBOARDCLIENTDATA *pClient, uint32_t u32Formats)
789{
790 /*
791 * The guest announces formats. Forward to the window thread.
792 */
793 PostMessage (pClient->pCtx->hwnd, WM_USER, 0, u32Formats);
794}
795
796int vboxClipboardReadData (VBOXCLIPBOARDCLIENTDATA *pClient, uint32_t u32Format, void *pv, uint32_t cb, uint32_t *pcbActual)
797{
798 LogFlow(("vboxClipboardReadData: u32Format = %02X\n", u32Format));
799
800 HANDLE hClip = NULL;
801
802 /*
803 * The guest wants to read data in the given format.
804 */
805 int rc = vboxOpenClipboard(pClient->pCtx->hwnd);
806 if (RT_SUCCESS(rc))
807 {
808 dprintf(("Clipboard opened.\n"));
809
810 if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
811 {
812 hClip = GetClipboardData (CF_DIB);
813
814 if (hClip != NULL)
815 {
816 LPVOID lp = GlobalLock (hClip);
817
818 if (lp != NULL)
819 {
820 dprintf(("CF_DIB\n"));
821
822 vboxClipboardGetData (VBOX_SHARED_CLIPBOARD_FMT_BITMAP, lp, GlobalSize (hClip),
823 pv, cb, pcbActual);
824
825 GlobalUnlock(hClip);
826 }
827 else
828 {
829 hClip = NULL;
830 }
831 }
832 }
833 else if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
834 {
835 hClip = GetClipboardData(CF_UNICODETEXT);
836
837 if (hClip != NULL)
838 {
839 LPWSTR uniString = (LPWSTR)GlobalLock (hClip);
840
841 if (uniString != NULL)
842 {
843 dprintf(("CF_UNICODETEXT\n"));
844
845 vboxClipboardGetData (VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT, uniString, (lstrlenW (uniString) + 1) * 2,
846 pv, cb, pcbActual);
847
848 GlobalUnlock(hClip);
849 }
850 else
851 {
852 hClip = NULL;
853 }
854 }
855 }
856 else if (u32Format & VBOX_SHARED_CLIPBOARD_FMT_HTML)
857 {
858 UINT format = RegisterClipboardFormat ("HTML Format");
859
860 if (format != 0)
861 {
862 hClip = GetClipboardData (format);
863
864 if (hClip != NULL)
865 {
866 LPVOID lp = GlobalLock (hClip);
867
868 if (lp != NULL)
869 {
870 dprintf(("CF_HTML\n"));
871
872 vboxClipboardGetData (VBOX_SHARED_CLIPBOARD_FMT_HTML, lp, GlobalSize (hClip),
873 pv, cb, pcbActual);
874
875 GlobalUnlock(hClip);
876 }
877 else
878 {
879 hClip = NULL;
880 }
881 }
882 }
883 }
884
885 CloseClipboard ();
886 }
887 else
888 {
889 dprintf(("vboxClipboardReadData: failed to open clipboard, rc: %Rrc\n", rc));
890 }
891
892 if (hClip == NULL)
893 {
894 /* Reply with empty data. */
895 vboxClipboardGetData (0, NULL, 0,
896 pv, cb, pcbActual);
897 }
898
899 return VINF_SUCCESS;
900}
901
902void vboxClipboardWriteData (VBOXCLIPBOARDCLIENTDATA *pClient, void *pv, uint32_t cb, uint32_t u32Format)
903{
904 LogFlow(("vboxClipboardWriteData\n"));
905
906 /*
907 * The guest returns data that was requested in the WM_RENDERFORMAT handler.
908 */
909 Assert(pClient->data.pv == NULL && pClient->data.cb == 0 && pClient->data.u32Format == 0);
910
911 vboxClipboardDump(pv, cb, u32Format);
912
913 if (cb > 0)
914 {
915 pClient->data.pv = RTMemAlloc (cb);
916
917 if (pClient->data.pv)
918 {
919 memcpy (pClient->data.pv, pv, cb);
920 pClient->data.cb = cb;
921 pClient->data.u32Format = u32Format;
922 }
923 }
924
925 SetEvent(pClient->pCtx->hRenderEvent);
926}
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