VirtualBox

source: vbox/trunk/src/VBox/Additions/darwin/VBoxClient/VBoxClientClipboardHostToGuest.cpp@ 78003

Last change on this file since 78003 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.1 KB
Line 
1/** $Id: VBoxClientClipboardHostToGuest.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * VBoxClient - Shared Clipboard Host -> Guest copying, Darwin.
4 */
5
6/*
7 * Copyright (C) 2007-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <Carbon/Carbon.h>
23#include <signal.h>
24#include <stdlib.h>
25
26#include <iprt/initterm.h>
27#include <iprt/mem.h>
28#include <iprt/message.h>
29#include <iprt/stream.h>
30#include <iprt/thread.h>
31#include <iprt/utf16.h>
32#include <VBox/VBoxGuestLib.h>
33#include <VBox/HostServices/VBoxClipboardSvc.h>
34#include <VBox/GuestHost/clipboard-helper.h>
35#include "VBoxClientInternal.h"
36
37/**
38 * Allocate memory for host buffer and receive it.
39 *
40 * @param u32ClientId Host connection.
41 * @param fFormat Buffer data format.
42 * @param pData Where to store received data.
43 * @param cbDataSize The size of the received data.
44 * @param cbMemSize The actual size of memory occupied by *pData.
45 *
46 * @returns IPRT status code.
47 */
48static int vbclClipboardReadHostData(uint32_t u32ClientId, uint32_t fFormat, void **pData, uint32_t *cbDataSize, uint32_t *cbMemSize)
49{
50 int rc;
51
52 AssertReturn(pData && cbDataSize && cbMemSize, VERR_INVALID_PARAMETER);
53
54 uint32_t cbDataSizeInternal = _4K;
55 uint32_t cbMemSizeInternal = cbDataSizeInternal;
56 void *pDataInternal = RTMemPageAllocZ(cbDataSizeInternal);
57
58 if (!pDataInternal)
59 return VERR_NO_MEMORY;
60
61 rc = VbglR3ClipboardReadData(u32ClientId, fFormat, pDataInternal, cbMemSizeInternal, &cbDataSizeInternal);
62 if (rc == VINF_BUFFER_OVERFLOW)
63 {
64 /* Reallocate bigger buffer and receive all the data */
65 RTMemPageFree(pDataInternal, cbMemSizeInternal);
66 cbDataSizeInternal = cbMemSizeInternal = RT_ALIGN_32(cbDataSizeInternal, PAGE_SIZE);
67 pDataInternal = RTMemPageAllocZ(cbMemSizeInternal);
68 if (!pDataInternal)
69 return VERR_NO_MEMORY;
70
71 rc = VbglR3ClipboardReadData(u32ClientId, fFormat, pDataInternal, cbMemSizeInternal, &cbDataSizeInternal);
72 }
73
74 /* Error occurred of zero-sized buffer */
75 if (RT_FAILURE(rc))
76 {
77 RTMemPageFree(pDataInternal, cbMemSizeInternal);
78 return VERR_NO_MEMORY;
79 }
80
81 *pData = pDataInternal;
82 *cbDataSize = cbDataSizeInternal;
83 *cbMemSize = cbMemSizeInternal;
84
85 return rc;
86}
87
88/**
89 * Release memory occupied by host buffer.
90 *
91 * @param pData Pointer to memory occupied by host buffer.
92 * @param cbMemSize The actual size of memory occupied by *pData.
93 */
94static void vbclClipboardReleaseHostData(void **pData, uint32_t cbMemSize)
95{
96 AssertReturnVoid(pData && cbMemSize > 0);
97 RTMemPageFree(*pData, cbMemSize);
98}
99
100/**
101 * Paste buffer into guest clipboard.
102 *
103 * @param pPasteboard Guest PasteBoard reference.
104 * @param pData Data to be pasted.
105 * @param cbDataSize The size of *pData.
106 * @param fFormat Buffer data format.
107 * @param fClear Whether or not clear guest clipboard before insert data.
108 *
109 * @returns IPRT status code.
110 */
111static int vbclClipboardGuestPasteData(PasteboardRef pPasteboard, UInt8 *pData, CFIndex cbDataSize, CFStringRef sFormat, bool fClear)
112{
113 PasteboardItemID itemId = (PasteboardItemID)1;
114 CFDataRef textData = NULL;
115 OSStatus rc;
116
117 /* Ignoring sunchronization flags here */
118 PasteboardSynchronize(pPasteboard);
119
120 if (fClear)
121 {
122 rc = PasteboardClear(pPasteboard);
123 AssertReturn(rc == noErr, VERR_NOT_SUPPORTED);
124 }
125
126 /* Create a CData object which we could pass to the pasteboard */
127 if ((textData = CFDataCreate(kCFAllocatorDefault, pData, cbDataSize)))
128 {
129 /* Put the Utf-8 version to the pasteboard */
130 rc = PasteboardPutItemFlavor(pPasteboard, itemId, sFormat, textData, 0);
131 CFRelease(textData);
132 if (rc != noErr)
133 {
134 VBoxClientVerbose(3, "unable to put data into guest's clipboard: %d\n", rc);
135 return VERR_GENERAL_FAILURE;
136 }
137 }
138 else
139 return VERR_NO_MEMORY;
140
141 /* Synchronize updated content */
142 PasteboardSynchronize(pPasteboard);
143
144 return VINF_SUCCESS;
145}
146
147/**
148 * Paste text data into guest clipboard.
149 *
150 * @param pPasteboard Guest PasteBoard reference.
151 * @param pData Data to be pasted.
152 * @param cbDataSize Size of *pData.
153 */
154static int vbclClipboardGuestPasteText(PasteboardRef pPasteboard, void *pData, uint32_t cbDataSize)
155{
156 size_t cbActualLen;
157 int rc;
158 char *pszUtf8Buf;
159 RTUTF16 *pDataInternal;
160
161 AssertReturn(pData, VERR_INVALID_PARAMETER);
162
163 /* Skip zero-sized buffer */
164 AssertReturn(cbDataSize > 0, VINF_SUCCESS);
165
166 /* If buffer content is Unicode text, then deliver
167 it in both formats UTF16 (original) and UTF8. */
168
169 /* Convert END-OF-LINE */
170 rc = vboxClipboardUtf16GetLinSize((RTUTF16 *)pData, cbDataSize / 2, &cbActualLen);
171 AssertReturn(RT_SUCCESS(rc), rc);
172 pDataInternal = (RTUTF16 *)RTMemAlloc(cbActualLen * 2);
173 AssertReturn(pDataInternal, VERR_NO_MEMORY);
174 rc = vboxClipboardUtf16WinToLin((RTUTF16 *)pData, cbDataSize / 2, pDataInternal, cbActualLen);
175
176 /* Do actual paste */
177 if (RT_SUCCESS(rc))
178 {
179 /* Paste UTF16 */
180 rc = vbclClipboardGuestPasteData(pPasteboard, (UInt8 *)pDataInternal, cbActualLen * 2, kUTTypeUTF16PlainText, true);
181 if (RT_SUCCESS(rc))
182 {
183 /* Paste UTF8 */
184 rc = RTUtf16ToUtf8((RTUTF16 *)pDataInternal, &pszUtf8Buf);
185 if (RT_SUCCESS(rc))
186 {
187 rc = vbclClipboardGuestPasteData(pPasteboard, (UInt8 *)pszUtf8Buf, strlen(pszUtf8Buf), kUTTypeUTF8PlainText, false);
188 RTStrFree(pszUtf8Buf);
189 }
190 }
191
192 }
193
194 RTMemFree(pDataInternal);
195
196 return rc;
197}
198
199/**
200 * Paste picture data into guest clipboard.
201 *
202 * @param pPasteboard Guest PasteBoard reference.
203 * @param pData Data to be pasted.
204 * @param cbDataSize The size of *pData.
205 *
206 * @returns IPRT status code.
207 */
208static int vbclClipboardGuestPastePicture(PasteboardRef pPasteboard, void *pData, uint32_t cbDataSize)
209{
210 int rc;
211 void *pBmp;
212 size_t cbBmpSize;
213
214 AssertReturn(pData, VERR_INVALID_PARAMETER);
215 /* Skip zero-sized buffer */
216 AssertReturn(cbDataSize > 0, VINF_SUCCESS);
217
218 rc = vboxClipboardDibToBmp(pData, cbDataSize, &pBmp, &cbBmpSize);
219 AssertReturn(RT_SUCCESS(rc), rc);
220
221 rc = vbclClipboardGuestPasteData(pPasteboard, (UInt8 *)pBmp, cbBmpSize, kUTTypeBMP, true);
222 RTMemFree(pBmp);
223
224 return rc;
225}
226
227/**
228 * Read host's clipboard buffer and put its content to guest clipboard.
229 *
230 * @param u32ClientId Host connection.
231 * @param pPasteboard Guest PasteBoard reference.
232 * @param fFormats List of data formats (bit field) received from host.
233 *
234 * @returns IPRT status code.
235 */
236int vbclClipboardForwardToGuest(uint32_t u32ClientId, PasteboardRef pPasteboard, uint32_t fFormats)
237{
238 int rc = VERR_INVALID_PARAMETER;
239 void *pData;
240 uint32_t cbDataSize, cbMemSize;
241 uint32_t fFormatsInternal = fFormats;
242
243 /* Walk across all item(s) formats */
244 while (fFormatsInternal)
245 {
246 if (fFormatsInternal & VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT)
247 {
248 VBoxClientVerbose(3, "found VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT: %d\n", fFormatsInternal);
249
250 rc = vbclClipboardReadHostData(u32ClientId, VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT, &pData, &cbDataSize, &cbMemSize);
251 if (RT_SUCCESS(rc))
252 {
253 /* Store data in guest buffer */
254 rc = vbclClipboardGuestPasteText(pPasteboard, pData, cbDataSize);
255
256 /* Release occupied resources */
257 vbclClipboardReleaseHostData(&pData, cbMemSize);
258 }
259
260 fFormatsInternal &= ~((uint32_t)VBOX_SHARED_CLIPBOARD_FMT_UNICODETEXT);
261 }
262
263 else if (fFormatsInternal & VBOX_SHARED_CLIPBOARD_FMT_BITMAP)
264 {
265 VBoxClientVerbose(3, "found VBOX_SHARED_CLIPBOARD_FMT_BITMAP: %d\n", fFormatsInternal);
266
267 rc = vbclClipboardReadHostData(u32ClientId, VBOX_SHARED_CLIPBOARD_FMT_BITMAP, &pData, &cbDataSize, &cbMemSize);
268 if (RT_SUCCESS(rc))
269 {
270 /* Store data in guest buffer */
271 rc = vbclClipboardGuestPastePicture(pPasteboard, pData, cbDataSize);
272
273 /* Release occupied resources */
274 vbclClipboardReleaseHostData(&pData, cbMemSize);
275 }
276
277 fFormatsInternal &= ~((uint32_t)VBOX_SHARED_CLIPBOARD_FMT_BITMAP);
278 }
279
280 else if (fFormatsInternal & VBOX_SHARED_CLIPBOARD_FMT_HTML)
281 {
282 VBoxClientVerbose(3, "found VBOX_SHARED_CLIPBOARD_FMT_HTML: %d\n", fFormatsInternal);
283
284 rc = vbclClipboardReadHostData(u32ClientId, VBOX_SHARED_CLIPBOARD_FMT_HTML, &pData, &cbDataSize, &cbMemSize);
285 if (RT_SUCCESS(rc))
286 {
287 /* Store data in guest buffer */
288 rc = vbclClipboardGuestPasteData(pPasteboard, (UInt8 *)pData, cbDataSize, kUTTypeHTML, true);
289
290 /* Release occupied resources */
291 vbclClipboardReleaseHostData(&pData, cbMemSize);
292 }
293
294 fFormatsInternal &= ~((uint32_t)VBOX_SHARED_CLIPBOARD_FMT_HTML);
295 }
296
297 else
298 {
299 VBoxClientVerbose(3, "received data in unsupported format: %d\n", fFormats);
300 break;
301 }
302 }
303
304 return rc;
305}
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