/** @file * Shared Clipboard: Common X11 code. */ /* * Copyright (C) 2006-2019 Oracle Corporation * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. */ /* Note: to automatically run regression tests on the shared clipboard, * execute the tstClipboardX11 testcase. If you often make changes to the * clipboard code, adding the line * OTHERS += $(PATH_tstClipboardX11)/tstClipboardX11.run * to LocalConfig.kmk will cause the tests to be run every time the code is * changed. */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_SHARED_CLIPBOARD #include #include #include #include #ifdef RT_OS_SOLARIS #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS # include # include #endif #include #include #include #include #include #include /********************************************************************************************************************************* * Internal Functions * *********************************************************************************************************************************/ class formats; static Atom clipGetAtom(PSHCLX11CTX pCtx, const char *pszName); /********************************************************************************************************************************* * Global Variables * *********************************************************************************************************************************/ /** * The table maps X11 names to data formats * and to the corresponding VBox clipboard formats. */ static struct _SHCLX11FMTTABLE { /** The X11 atom name of the format (several names can match one format). */ const char *pcszAtom; /** The format corresponding to the name. */ SHCLX11FMT enmFmtX11; /** The corresponding VBox clipboard format. */ SHCLFORMAT uFmtVBox; } g_aFormats[] = { { "INVALID", SHCLX11FMT_INVALID, VBOX_SHCL_FMT_NONE }, { "UTF8_STRING", SHCLX11FMT_UTF8, VBOX_SHCL_FMT_UNICODETEXT }, { "text/plain;charset=UTF-8", SHCLX11FMT_UTF8, VBOX_SHCL_FMT_UNICODETEXT }, { "text/plain;charset=utf-8", SHCLX11FMT_UTF8, VBOX_SHCL_FMT_UNICODETEXT }, { "STRING", SHCLX11FMT_TEXT, VBOX_SHCL_FMT_UNICODETEXT }, { "TEXT", SHCLX11FMT_TEXT, VBOX_SHCL_FMT_UNICODETEXT }, { "text/plain", SHCLX11FMT_TEXT, VBOX_SHCL_FMT_UNICODETEXT }, { "text/html", SHCLX11FMT_HTML, VBOX_SHCL_FMT_HTML }, { "text/html;charset=utf-8", SHCLX11FMT_HTML, VBOX_SHCL_FMT_HTML }, { "image/bmp", SHCLX11FMT_BMP, VBOX_SHCL_FMT_BITMAP }, { "image/x-bmp", SHCLX11FMT_BMP, VBOX_SHCL_FMT_BITMAP }, { "image/x-MS-bmp", SHCLX11FMT_BMP, VBOX_SHCL_FMT_BITMAP }, /** @todo Inkscape exports image/png but not bmp... */ #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS { "text/uri-list", SHCLX11FMT_URI_LIST, VBOX_SHCL_FMT_URI_LIST }, { "x-special/gnome-copied-files", SHCLX11FMT_URI_LIST, VBOX_SHCL_FMT_URI_LIST }, { "x-special/nautilus-clipboard", SHCLX11FMT_URI_LIST, VBOX_SHCL_FMT_URI_LIST }, { "application/x-kde-cutselection", SHCLX11FMT_URI_LIST, VBOX_SHCL_FMT_URI_LIST }, /** @todo Anything else we need to add here? */ /** @todo Add Wayland / Weston support. */ #endif }; enum { NIL_CLIPX11FORMAT = 0, MAX_CLIP_X11_FORMATS = RT_ELEMENTS(g_aFormats) }; /** * Returns the atom corresponding to a supported X11 format. * * @returns Found atom to the corresponding X11 format. * @param pCtx The X11 clipboard context to use. * @param uFmtIdx Format index to look up atom for. */ static Atom clipAtomForX11Format(PSHCLX11CTX pCtx, SHCLX11FMTIDX uFmtIdx) { LogFlowFunc(("format=%u -> pcszAtom=%s\n", uFmtIdx, g_aFormats[uFmtIdx].pcszAtom)); AssertReturn(uFmtIdx <= RT_ELEMENTS(g_aFormats), 0); return clipGetAtom(pCtx, g_aFormats[uFmtIdx].pcszAtom); } /** * Returns the SHCLX11FMT corresponding to a supported X11 format. * * @return SHCLX11FMT for a specific format index. * @param uFmtIdx Format index to look up SHCLX11CLIPFMT for. */ static SHCLX11FMT clipRealFormatForX11Format(SHCLX11FMTIDX uFmtIdx) { AssertReturn(uFmtIdx <= RT_ELEMENTS(g_aFormats), SHCLX11FMT_INVALID); return g_aFormats[uFmtIdx].enmFmtX11; } /** * Returns the VBox format corresponding to a supported X11 format. * * @return SHCLFORMAT for a specific format index. * @param uFmtIdx Format index to look up VBox format for. */ static SHCLFORMAT clipVBoxFormatForX11Format(SHCLX11FMTIDX uFmtIdx) { AssertReturn(uFmtIdx <= RT_ELEMENTS(g_aFormats), VBOX_SHCL_FMT_NONE); return g_aFormats[uFmtIdx].uFmtVBox; } /** * Looks up the X11 format matching a given X11 atom. * * @returns The format on success, NIL_CLIPX11FORMAT on failure. * @param pCtx The X11 clipboard context to use. * @param atomFormat Atom to look up X11 format for. */ static SHCLX11FMTIDX clipFindX11FormatByAtom(PSHCLX11CTX pCtx, Atom atomFormat) { for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i) if (clipAtomForX11Format(pCtx, i) == atomFormat) return i; return NIL_CLIPX11FORMAT; } #ifdef TESTCASE /** * Looks up the X11 format matching a given X11 atom text. * * @returns the format on success, NIL_CLIPX11FORMAT on failure * @param pcszAtom Atom text to look up format for. */ static SHCLX11FMTIDX clipFindX11FormatByAtomText(const char *pcszAtom) { for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i) if (!strcmp(g_aFormats[i].pcszAtom, pcszAtom)) return i; return NIL_CLIPX11FORMAT; } #endif /** * Enumerates supported X11 clipboard formats corresponding to given VBox formats. * * @returns The next matching X11 format index in the list, or NIL_CLIPX11FORMAT if there are no more. * @param uFormatsVBox VBox formats to enumerate supported X11 clipboard formats for. * @param lastFmtIdx The value returned from the last call of this function. * Use NIL_CLIPX11FORMAT to start the enumeration. */ static SHCLX11FMTIDX clipEnumX11Formats(SHCLFORMATS uFormatsVBox, SHCLX11FMTIDX lastFmtIdx) { for (unsigned i = lastFmtIdx + 1; i < RT_ELEMENTS(g_aFormats); ++i) { if (uFormatsVBox & clipVBoxFormatForX11Format(i)) return i; } return NIL_CLIPX11FORMAT; } /** * The number of simultaneous instances we support. For all normal purposes * we should never need more than one. For the testcase it is convenient to * have a second instance that the first can interact with in order to have * a more controlled environment. */ enum { CLIP_MAX_CONTEXTS = 20 }; /** * Array of structures for mapping Xt widgets to context pointers. We * need this because the widget clipboard callbacks do not pass user data. */ static struct { /** Pointer to widget we want to associate the context with. */ Widget pWidget; /** Pointer to X11 context associated with the widget. */ PSHCLX11CTX pCtx; } g_aContexts[CLIP_MAX_CONTEXTS]; /** * Registers a new X11 clipboard context. * * @returns VBox status code. * @param pCtx The X11 clipboard context to use. */ static int clipRegisterContext(PSHCLX11CTX pCtx) { AssertPtrReturn(pCtx, VERR_INVALID_PARAMETER); bool fFound = false; Widget pWidget = pCtx->pWidget; AssertReturn(pWidget != NULL, VERR_INVALID_PARAMETER); for (unsigned i = 0; i < RT_ELEMENTS(g_aContexts); ++i) { AssertReturn( (g_aContexts[i].pWidget != pWidget) && (g_aContexts[i].pCtx != pCtx), VERR_WRONG_ORDER); if (g_aContexts[i].pWidget == NULL && !fFound) { AssertReturn(g_aContexts[i].pCtx == NULL, VERR_INTERNAL_ERROR); g_aContexts[i].pWidget = pWidget; g_aContexts[i].pCtx = pCtx; fFound = true; } } return fFound ? VINF_SUCCESS : VERR_OUT_OF_RESOURCES; } /** * Unregister an X11 clipboard context. * * @param pCtx The X11 clipboard context to use. */ static void clipUnregisterContext(PSHCLX11CTX pCtx) { AssertPtrReturnVoid(pCtx); Widget widget = pCtx->pWidget; AssertPtrReturnVoid(widget); bool fFound = false; for (unsigned i = 0; i < RT_ELEMENTS(g_aContexts); ++i) { Assert(!fFound || g_aContexts[i].pWidget != widget); if (g_aContexts[i].pWidget == widget) { Assert(g_aContexts[i].pCtx != NULL); g_aContexts[i].pWidget = NULL; g_aContexts[i].pCtx = NULL; fFound = true; } } } /** * Finds a X11 clipboard context for a specific X11 widget. * * @returns Pointer to associated X11 clipboard context if found, or NULL if not found. * @param pWidget X11 widget to return X11 clipboard context for. */ static PSHCLX11CTX clipLookupContext(Widget pWidget) { AssertPtrReturn(pWidget, NULL); for (unsigned i = 0; i < RT_ELEMENTS(g_aContexts); ++i) { if (g_aContexts[i].pWidget == pWidget) { Assert(g_aContexts[i].pCtx != NULL); return g_aContexts[i].pCtx; } } return NULL; } /** * Converts an atom name string to an X11 atom, looking it up in a cache before asking the server. * * @returns Found X11 atom. * @param pCtx The X11 clipboard context to use. * @param pcszName Name of atom to return atom for. */ static Atom clipGetAtom(PSHCLX11CTX pCtx, const char *pcszName) { AssertPtrReturn(pcszName, None); return XInternAtom(XtDisplay(pCtx->pWidget), pcszName, False); } #ifdef TESTCASE static void tstClipQueueToEventThread(void (*proc)(void *, void *), void *client_data); #endif /** String written to the wakeup pipe. */ #define WAKE_UP_STRING "WakeUp!" /** Length of the string written. */ #define WAKE_UP_STRING_LEN ( sizeof(WAKE_UP_STRING) - 1 ) /** * Schedules a function call to run on the Xt event thread by passing it to * the application context as a 0ms timeout and waking up the event loop by * writing to the wakeup pipe which it monitors. */ static int clipQueueToEventThread(PSHCLX11CTX pCtx, void (*proc)(void *, void *), void *client_data) { LogFlowFunc(("proc=%p, client_data=%p\n", proc, client_data)); int rc = VINF_SUCCESS; #ifndef TESTCASE XtAppAddTimeOut(pCtx->appContext, 0, (XtTimerCallbackProc)proc, (XtPointer)client_data); ssize_t cbWritten = write(pCtx->wakeupPipeWrite, WAKE_UP_STRING, WAKE_UP_STRING_LEN); RT_NOREF(cbWritten); #else RT_NOREF(pCtx); tstClipQueueToEventThread(proc, client_data); #endif LogFlowFuncLeaveRC(rc); return rc; } /** * Reports the formats currently supported by the X11 clipboard to VBox. * * @note Runs in Xt event thread. * * @param pCtx The X11 clipboard context to use. */ static void clipReportFormatsToVBox(PSHCLX11CTX pCtx) { uint32_t fFormats = clipVBoxFormatForX11Format(pCtx->X11TextFormat); fFormats |= clipVBoxFormatForX11Format(pCtx->X11BitmapFormat); fFormats |= clipVBoxFormatForX11Format(pCtx->X11HTMLFormat); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS fFormats |= clipVBoxFormatForX11Format(pCtx->X11URIListFormat); #endif LogFlowFunc(("fFormats=0x%x\n", fFormats)); LogFlowFunc(("txt: %u, bmp: %u, html: %u\n", pCtx->X11TextFormat, pCtx->X11BitmapFormat, pCtx->X11HTMLFormat)); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS LogFlowFunc(("uri list: %u\n", pCtx->X11URIListFormat)); #endif LogRel2(("Shared Clipboard: X11 reported available formats 0x%x\n", fFormats)); ShClX11ReportFormatsCallback(pCtx->pFrontend, fFormats); } /** * Forgets which formats were previously in the X11 clipboard. Called when we * grab the clipboard. * * @param pCtx The X11 clipboard context to use. */ static void clipResetX11Formats(PSHCLX11CTX pCtx) { LogFlowFuncEnter(); pCtx->X11TextFormat = SHCLX11FMT_INVALID; pCtx->X11BitmapFormat = SHCLX11FMT_INVALID; pCtx->X11HTMLFormat = SHCLX11FMT_INVALID; #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS pCtx->X11URIListFormat = SHCLX11FMT_INVALID; #endif } /** * Tells VBox that X11 currently has nothing in its clipboard. * * @param pCtx The X11 clipboard context to use. */ static void clipReportEmptyX11CB(PSHCLX11CTX pCtx) { clipResetX11Formats(pCtx); clipReportFormatsToVBox(pCtx); } /** * Go through an array of X11 clipboard targets to see if they contain a text * format we can support, and if so choose the ones we prefer (e.g. we like * UTF-8 better than plain text). * * @return Index to supported X clipboard format. * @param pCtx The X11 clipboard context to use. * @param pTargets The list of targets. * @param cTargets The size of the list in @a pTargets. */ static SHCLX11FMTIDX clipGetTextFormatFromTargets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT); AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT); SHCLX11FMTIDX bestTextFormat = NIL_CLIPX11FORMAT; SHCLX11FMT enmBestTextTarget = SHCLX11FMT_INVALID; for (unsigned i = 0; i < cTargets; ++i) { SHCLX11FMTIDX format = pTargets[i]; if (format != NIL_CLIPX11FORMAT) { if ( (clipVBoxFormatForX11Format(format) == VBOX_SHCL_FMT_UNICODETEXT) && enmBestTextTarget < clipRealFormatForX11Format(format)) { enmBestTextTarget = clipRealFormatForX11Format(format); bestTextFormat = format; } } } return bestTextFormat; } #ifdef TESTCASE static bool tstClipTextFormatConversion(PSHCLX11CTX pCtx) { bool fSuccess = true; SHCLX11FMTIDX targets[2]; SHCLX11FMTIDX x11Format; targets[0] = clipFindX11FormatByAtomText("text/plain"); targets[1] = clipFindX11FormatByAtomText("image/bmp"); x11Format = clipGetTextFormatFromTargets(pCtx, targets, 2); if (clipRealFormatForX11Format(x11Format) != SHCLX11FMT_TEXT) fSuccess = false; targets[0] = clipFindX11FormatByAtomText("UTF8_STRING"); targets[1] = clipFindX11FormatByAtomText("text/plain"); x11Format = clipGetTextFormatFromTargets(pCtx, targets, 2); if (clipRealFormatForX11Format(x11Format) != SHCLX11FMT_UTF8) fSuccess = false; return fSuccess; } #endif /** * Goes through an array of X11 clipboard targets to see if they contain a bitmap * format we can support, and if so choose the ones we prefer (e.g. we like * BMP better than PNG because we don't have to convert). * * @return Supported X clipboard format. * @param pCtx The X11 clipboard context to use. * @param pTargets The list of targets. * @param cTargets The size of the list in @a pTargets. */ static SHCLX11FMTIDX clipGetBitmapFormatFromTargets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT); AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT); SHCLX11FMTIDX bestBitmapFormat = NIL_CLIPX11FORMAT; SHCLX11FMT enmBestBitmapTarget = SHCLX11FMT_INVALID; for (unsigned i = 0; i < cTargets; ++i) { SHCLX11FMTIDX format = pTargets[i]; if (format != NIL_CLIPX11FORMAT) { if ( (clipVBoxFormatForX11Format(format) == VBOX_SHCL_FMT_BITMAP) && enmBestBitmapTarget < clipRealFormatForX11Format(format)) { enmBestBitmapTarget = clipRealFormatForX11Format(format); bestBitmapFormat = format; } } } return bestBitmapFormat; } /** * Goes through an array of X11 clipboard targets to see if they contain a HTML * format we can support, and if so choose the ones we prefer. * * @return Supported X clipboard format. * @param pCtx The X11 clipboard context to use. * @param pTargets The list of targets. * @param cTargets The size of the list in @a pTargets. */ static SHCLX11FMTIDX clipGetHtmlFormatFromTargets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT); AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT); SHCLX11FMTIDX bestHTMLFormat = NIL_CLIPX11FORMAT; SHCLX11FMT enmBestHtmlTarget = SHCLX11FMT_INVALID; for (unsigned i = 0; i < cTargets; ++i) { SHCLX11FMTIDX format = pTargets[i]; if (format != NIL_CLIPX11FORMAT) { if ( (clipVBoxFormatForX11Format(format) == VBOX_SHCL_FMT_HTML) && enmBestHtmlTarget < clipRealFormatForX11Format(format)) { enmBestHtmlTarget = clipRealFormatForX11Format(format); bestHTMLFormat = format; } } } return bestHTMLFormat; } #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS /** * Goes through an array of X11 clipboard targets to see if they contain an URI list * format we can support, and if so choose the ones we prefer. * * @return Supported X clipboard format. * @param pCtx The X11 clipboard context to use. * @param pTargets The list of targets. * @param cTargets The size of the list in @a pTargets. */ static SHCLX11FMTIDX clipGetURIListFormatFromTargets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { AssertPtrReturn(pCtx, NIL_CLIPX11FORMAT); AssertReturn(VALID_PTR(pTargets) || cTargets == 0, NIL_CLIPX11FORMAT); SHCLX11FMTIDX bestURIListFormat = NIL_CLIPX11FORMAT; SHCLX11FMT enmBestURIListTarget = SHCLX11FMT_INVALID; for (unsigned i = 0; i < cTargets; ++i) { SHCLX11FMTIDX format = pTargets[i]; if (format != NIL_CLIPX11FORMAT) { if ( (clipVBoxFormatForX11Format(format) == VBOX_SHCL_FMT_URI_LIST) && enmBestURIListTarget < clipRealFormatForX11Format(format)) { enmBestURIListTarget = clipRealFormatForX11Format(format); bestURIListFormat = format; } } } return bestURIListFormat; } #endif /* VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS */ /** * Goes through an array of X11 clipboard targets to see if we can support any * of them and if relevant to choose the ones we prefer (e.g. we like Utf8 * better than plain text). * * @param pCtx The X11 clipboard context to use. * @param pTargets The list of targets. * @param cTargets The size of the list in @a pTargets. */ static void clipGetFormatsFromTargets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { AssertPtrReturnVoid(pCtx); AssertPtrReturnVoid(pTargets); SHCLX11FMTIDX bestTextFormat = clipGetTextFormatFromTargets(pCtx, pTargets, cTargets); if (pCtx->X11TextFormat != bestTextFormat) pCtx->X11TextFormat = bestTextFormat; pCtx->X11BitmapFormat = SHCLX11FMT_INVALID; /* not yet supported */ /** @todo r=andy Check this. */ SHCLX11FMTIDX bestBitmapFormat = clipGetBitmapFormatFromTargets(pCtx, pTargets, cTargets); if (pCtx->X11BitmapFormat != bestBitmapFormat) pCtx->X11BitmapFormat = bestBitmapFormat; SHCLX11FMTIDX bestHtmlFormat = clipGetHtmlFormatFromTargets(pCtx, pTargets, cTargets); if (pCtx->X11HTMLFormat != bestHtmlFormat) pCtx->X11HTMLFormat = bestHtmlFormat; #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS SHCLX11FMTIDX bestURIListFormat = clipGetURIListFormatFromTargets(pCtx, pTargets, cTargets); if (pCtx->X11URIListFormat != bestURIListFormat) pCtx->X11URIListFormat = bestURIListFormat; #endif } static DECLCALLBACK(void) clipQueryX11FormatsCallback(PSHCLX11CTX pCtx); /** * Updates the context's information about targets currently supported by X11, * based on an array of X11 atoms. * * @param pCtx The X11 clipboard context to use. * @param pTargets The array of atoms describing the targets supported. * @param cTargets The size of the array @a pTargets. */ static void clipUpdateX11Targets(PSHCLX11CTX pCtx, SHCLX11FMTIDX *pTargets, size_t cTargets) { LogFlowFuncEnter(); if (pTargets == NULL) { /* No data available */ clipReportEmptyX11CB(pCtx); return; } clipGetFormatsFromTargets(pCtx, pTargets, cTargets); clipReportFormatsToVBox(pCtx); } /** * Notifies the VBox clipboard about available data formats, based on the * "targets" information obtained from the X11 clipboard. * * @note Callback for XtGetSelectionValue(). * @note This function is treated as API glue, and as such is not part of any * unit test. So keep it simple, be paranoid and log everything. */ static void clipConvertX11TargetsCallback(Widget widget, XtPointer pClient, Atom * /* selection */, Atom *atomType, XtPointer pValue, long unsigned int *pcLen, int *piFormat) { RT_NOREF(piFormat); PSHCLX11CTX pCtx = reinterpret_cast(pClient); #ifndef TESTCASE LogFlowFunc(("fXtNeedsUpdate=%RTbool, fXtBusy=%RTbool\n", pCtx->fXtNeedsUpdate, pCtx->fXtBusy)); if (pCtx->fXtNeedsUpdate) { // The data from this callback is already out of date. Refresh it. pCtx->fXtNeedsUpdate = false; XtGetSelectionValue(pCtx->pWidget, clipGetAtom(pCtx, "CLIPBOARD"), clipGetAtom(pCtx, "TARGETS"), clipConvertX11TargetsCallback, pCtx, CurrentTime); return; } else { pCtx->fXtBusy = false; } #endif Atom *pAtoms = (Atom *)pValue; unsigned i, j; LogFlowFunc(("pValue=%p, *pcLen=%u, *atomType=%d%s\n", pValue, *pcLen, *atomType, *atomType == XT_CONVERT_FAIL ? " (XT_CONVERT_FAIL)" : "")); SHCLX11FMTIDX *pFormats = NULL; if ( *pcLen && pValue && (*atomType != XT_CONVERT_FAIL /* time out */)) { pFormats = (SHCLX11FMTIDX *)RTMemAllocZ(*pcLen * sizeof(SHCLX11FMTIDX)); } #if !defined(TESTCASE) if (pValue) { for (i = 0; i < *pcLen; ++i) { if (pAtoms[i]) { char *pszName = XGetAtomName(XtDisplay(widget), pAtoms[i]); LogRel2(("Shared Clipboard: Found target '%s'\n", pszName)); XFree(pszName); } else LogFunc(("Found empty target\n")); } } #endif if (pFormats) { for (i = 0; i < *pcLen; ++i) { for (j = 0; j < RT_ELEMENTS(g_aFormats); ++j) { Atom target = XInternAtom(XtDisplay(widget), g_aFormats[j].pcszAtom, False); if (*(pAtoms + i) == target) pFormats[i] = j; } #if defined(DEBUG) && !defined(TESTCASE) LogRel2(("%s: reporting format %d (%s)\n", __FUNCTION__, pFormats[i], g_aFormats[pFormats[i]].pcszAtom)); #endif } } else LogFunc(("Reporting empty targets (none reported or allocation failure)\n")); clipUpdateX11Targets(pCtx, pFormats, *pcLen); RTMemFree(pFormats); XtFree(reinterpret_cast(pValue)); } #ifdef TESTCASE void tstRequestTargets(PSHCLX11CTX pCtx); #endif /** * Callback to notify us when the contents of the X11 clipboard change. * * @param pCtx The X11 clipboard context to use. */ static DECLCALLBACK(void) clipQueryX11FormatsCallback(PSHCLX11CTX pCtx) { LogFlowFuncEnter(); #ifndef TESTCASE LogFlowFunc(("fXtBusy=%RTbool\n", pCtx->fXtBusy)); if (pCtx->fXtBusy) { pCtx->fXtNeedsUpdate = true; } else { pCtx->fXtBusy = true; XtGetSelectionValue(pCtx->pWidget, clipGetAtom(pCtx, "CLIPBOARD"), clipGetAtom(pCtx, "TARGETS"), clipConvertX11TargetsCallback, pCtx, CurrentTime); } #else tstRequestTargets(pCtx); #endif } #ifndef TESTCASE typedef struct { int type; /* event base */ unsigned long serial; Bool send_event; Display *display; Window window; int subtype; Window owner; Atom selection; Time timestamp; Time selection_timestamp; } XFixesSelectionNotifyEvent; /** * Waits until an event arrives and handle it if it is an XFIXES selection * event, which Xt doesn't know about. * * @param pCtx The X11 clipboard context to use. */ void clipPeekEventAndDoXFixesHandling(PSHCLX11CTX pCtx) { union { XEvent event; XFixesSelectionNotifyEvent fixes; } event = { { 0 } }; if (XtAppPeekEvent(pCtx->appContext, &event.event)) { if ( (event.event.type == pCtx->fixesEventBase) && (event.fixes.owner != XtWindow(pCtx->pWidget))) { if ( (event.fixes.subtype == 0 /* XFixesSetSelectionOwnerNotify */) && (event.fixes.owner != 0)) clipQueryX11FormatsCallback(pCtx); else clipReportEmptyX11CB(pCtx); } } } /** * The main loop of our X11 event thread. * * @returns VBox status code. * @param hThreadSelf Associated thread handle. * @param pvUser Pointer to user-provided thread data. */ static DECLCALLBACK(int) clipEventThread(RTTHREAD hThreadSelf, void *pvUser) { RT_NOREF(hThreadSelf); LogRel(("Shared Clipboard: Starting X11 event thread\n")); PSHCLX11CTX pCtx = (SHCLX11CTX *)pvUser; if (pCtx->fGrabClipboardOnStart) clipQueryX11FormatsCallback(pCtx); while (XtAppGetExitFlag(pCtx->appContext) == FALSE) { clipPeekEventAndDoXFixesHandling(pCtx); XtAppProcessEvent(pCtx->appContext, XtIMAll); } LogRel(("Shared Clipboard: X11 event thread terminated successfully\n")); return VINF_SUCCESS; } #endif /* !TESTCASE */ /** * X11 specific uninitialisation for the shared clipboard. * * @param pCtx The X11 clipboard context to use. */ static void clipUninit(PSHCLX11CTX pCtx) { AssertPtrReturnVoid(pCtx); if (pCtx->pWidget) { /* Valid widget + invalid appcontext = bug. But don't return yet. */ AssertPtr(pCtx->appContext); clipUnregisterContext(pCtx); XtDestroyWidget(pCtx->pWidget); } pCtx->pWidget = NULL; if (pCtx->appContext) XtDestroyApplicationContext(pCtx->appContext); pCtx->appContext = NULL; if (pCtx->wakeupPipeRead != 0) close(pCtx->wakeupPipeRead); if (pCtx->wakeupPipeWrite != 0) close(pCtx->wakeupPipeWrite); pCtx->wakeupPipeRead = 0; pCtx->wakeupPipeWrite = 0; } /** Worker function for stopping the clipboard which runs on the event * thread. */ static void clipStopEventThreadWorker(void *pUserData, void *) { PSHCLX11CTX pCtx = (SHCLX11CTX *)pUserData; /* This might mean that we are getting stopped twice. */ Assert(pCtx->pWidget != NULL); /* Set the termination flag to tell the Xt event loop to exit. We * reiterate that any outstanding requests from the X11 event loop to * the VBox part *must* have returned before we do this. */ XtAppSetExitFlag(pCtx->appContext); } #ifndef TESTCASE /** * Sets up the XFixes library and load the XFixesSelectSelectionInput symbol. */ static int clipLoadXFixes(Display *pDisplay, PSHCLX11CTX pCtx) { int rc; void *hFixesLib = dlopen("libXfixes.so.1", RTLD_LAZY); if (!hFixesLib) hFixesLib = dlopen("libXfixes.so.2", RTLD_LAZY); if (!hFixesLib) hFixesLib = dlopen("libXfixes.so.3", RTLD_LAZY); if (!hFixesLib) hFixesLib = dlopen("libXfixes.so.4", RTLD_LAZY); if (hFixesLib) { /* For us, a NULL function pointer is a failure */ pCtx->fixesSelectInput = (void (*)(Display *, Window, Atom, long unsigned int)) (uintptr_t)dlsym(hFixesLib, "XFixesSelectSelectionInput"); if (pCtx->fixesSelectInput) { int dummy1 = 0; int dummy2 = 0; if (XQueryExtension(pDisplay, "XFIXES", &dummy1, &pCtx->fixesEventBase, &dummy2) != 0) { if (pCtx->fixesEventBase >= 0) { rc = VINF_SUCCESS; } else { LogRel(("Shared Clipboard: fixesEventBase is less than zero: %d\n", pCtx->fixesEventBase)); rc = VERR_NOT_SUPPORTED; } } else { LogRel(("Shared Clipboard: XQueryExtension failed\n")); rc = VERR_NOT_SUPPORTED; } } else { LogRel(("Shared Clipboard: Symbol XFixesSelectSelectionInput not found!\n")); rc = VERR_NOT_SUPPORTED; } } else { LogRel(("Shared Clipboard: libxFixes.so.* not found!\n")); rc = VERR_NOT_SUPPORTED; } return rc; } #endif /* !TESTCASE */ /** * This is the callback which is scheduled when data is available on the * wakeup pipe. It simply reads all data from the pipe. */ static void clipDrainWakeupPipe(XtPointer pUserData, int *, XtInputId *) { LogFlowFuncEnter(); PSHCLX11CTX pCtx = (SHCLX11CTX *)pUserData; char acBuf[WAKE_UP_STRING_LEN]; while (read(pCtx->wakeupPipeRead, acBuf, sizeof(acBuf)) > 0) {} } /** * X11 specific initialisation for the shared clipboard. * * @returns VBox status code. * @param pCtx The X11 clipboard context to use. */ static int clipInit(PSHCLX11CTX pCtx) { /* Create a window and make it a clipboard viewer. */ int cArgc = 0; char *pcArgv = 0; int rc = VINF_SUCCESS; Display *pDisplay; /* Make sure we are thread safe. */ XtToolkitThreadInitialize(); /* * Set up the Clipboard application context and main window. We call all * these functions directly instead of calling XtOpenApplication() so * that we can fail gracefully if we can't get an X11 display. */ XtToolkitInitialize(); pCtx->appContext = XtCreateApplicationContext(); pDisplay = XtOpenDisplay(pCtx->appContext, 0, 0, "VBoxShCl", 0, 0, &cArgc, &pcArgv); if (NULL == pDisplay) { LogRel(("Shared Clipboard: Failed to connect to the X11 clipboard - the window system may not be running\n")); rc = VERR_NOT_SUPPORTED; } #ifndef TESTCASE if (RT_SUCCESS(rc)) { rc = clipLoadXFixes(pDisplay, pCtx); if (RT_FAILURE(rc)) LogRel(("Shared Clipboard: Failed to load the XFIXES extension\n")); } #endif if (RT_SUCCESS(rc)) { pCtx->pWidget = XtVaAppCreateShell(0, "VBoxShCl", applicationShellWidgetClass, pDisplay, XtNwidth, 1, XtNheight, 1, NULL); if (NULL == pCtx->pWidget) { LogRel(("Shared Clipboard: Failed to construct the X11 window for the shared clipboard manager\n")); rc = VERR_NO_MEMORY; } else rc = clipRegisterContext(pCtx); } if (RT_SUCCESS(rc)) { XtSetMappedWhenManaged(pCtx->pWidget, false); XtRealizeWidget(pCtx->pWidget); #ifndef TESTCASE /* Enable clipboard update notification. */ pCtx->fixesSelectInput(pDisplay, XtWindow(pCtx->pWidget), clipGetAtom(pCtx, "CLIPBOARD"), 7 /* All XFixes*Selection*NotifyMask flags */); #endif } /* * Create the pipes. */ int pipes[2]; if (!pipe(pipes)) { pCtx->wakeupPipeRead = pipes[0]; pCtx->wakeupPipeWrite = pipes[1]; if (!XtAppAddInput(pCtx->appContext, pCtx->wakeupPipeRead, (XtPointer) XtInputReadMask, clipDrainWakeupPipe, (XtPointer) pCtx)) rc = VERR_NO_MEMORY; /* What failure means is not doc'ed. */ if ( RT_SUCCESS(rc) && (fcntl(pCtx->wakeupPipeRead, F_SETFL, O_NONBLOCK) != 0)) rc = RTErrConvertFromErrno(errno); if (RT_FAILURE(rc)) LogRel(("Shared Clipboard: Failed to setup the termination mechanism\n")); } else rc = RTErrConvertFromErrno(errno); if (RT_FAILURE(rc)) clipUninit(pCtx); if (RT_FAILURE(rc)) LogRel(("Shared Clipboard: Initialisation failed: %Rrc\n", rc)); return rc; } /** * Initializes the X11 context of the Shared Clipboard. * * @returns VBox status code. * @param pCtx The clipboard context to initialize. * @param pParent Parent context to use. * @param fHeadless Whether the code runs in a headless environment or not. */ int ShClX11Init(PSHCLX11CTX pCtx, PSHCLCONTEXT pParent, bool fHeadless) { AssertPtrReturn(pCtx, VERR_INVALID_POINTER); #if !defined(SMOKETEST) && !defined(TESTCASE) /* Smoktests / Testcases don't have a (valid) parent. */ AssertPtrReturn(pParent, VERR_INVALID_POINTER); #endif RT_BZERO(pCtx, sizeof(SHCLX11CTX)); if (fHeadless) { /* * If we don't find the DISPLAY environment variable we assume that * we are not connected to an X11 server. Don't actually try to do * this then, just fail silently and report success on every call. * This is important for VBoxHeadless. */ LogRel(("Shared Clipboard: X11 DISPLAY variable not set -- disabling clipboard sharing\n")); } pCtx->fHaveX11 = !fHeadless; pCtx->pFrontend = pParent; pCtx->fXtBusy = false; pCtx->fXtNeedsUpdate = false; LogFlowFuncLeaveRC(VINF_SUCCESS); return VINF_SUCCESS; } /** * Destructs the Shared Clipboard X11 context. * * @param pCtx The X11 clipboard context to destroy. */ void ShClX11Destroy(PSHCLX11CTX pCtx) { if (!pCtx) return; if (pCtx->fHaveX11) { /* We set this to NULL when the event thread exits. It really should * have exited at this point, when we are about to unload the code from * memory. */ Assert(pCtx->pWidget == NULL); } } /** * Announces to the X11 backend that we are ready to start. * * @returns VBox status code. * @param pCtx The X11 clipboard context to use. * @param fGrab Whether we should try to grab the shared clipboard at once. */ int ShClX11ThreadStart(PSHCLX11CTX pCtx, bool fGrab) { int rc = VINF_SUCCESS; /* * Immediately return if we are not connected to the X server. */ if (!pCtx->fHaveX11) return VINF_SUCCESS; rc = clipInit(pCtx); if (RT_SUCCESS(rc)) { clipResetX11Formats(pCtx); pCtx->fGrabClipboardOnStart = fGrab; } #ifndef TESTCASE if (RT_SUCCESS(rc)) { LogRel2(("Shared Clipboard: Starting X11 event thread ...\n")); rc = RTThreadCreate(&pCtx->Thread, clipEventThread, pCtx, 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "SHCLX11"); if (RT_FAILURE(rc)) { LogRel(("Shared Clipboard: Failed to start the X11 event thread with %Rrc\n", rc)); clipUninit(pCtx); } } #endif return rc; } /** * Shuts down the shared clipboard X11 backend. * * @note Any requests from this object to get clipboard data from VBox * *must* have completed or aborted before we are called, as * otherwise the X11 event loop will still be waiting for the request * to return and will not be able to terminate. * * @returns VBox status code. * @param pCtx The X11 clipboard context to use. */ int ShClX11ThreadStop(PSHCLX11CTX pCtx) { int rc, rcThread; unsigned count = 0; /* * Immediately return if we are not connected to the X server. */ if (!pCtx->fHaveX11) return VINF_SUCCESS; LogRel(("Shared Clipboard: Stopping X11 event thread ...\n")); /* Write to the "stop" pipe. */ clipQueueToEventThread(pCtx, clipStopEventThreadWorker, (XtPointer)pCtx); #ifndef TESTCASE do { rc = RTThreadWait(pCtx->Thread, 1000, &rcThread); ++count; Assert(RT_SUCCESS(rc) || ((VERR_TIMEOUT == rc) && (count != 5))); } while ((VERR_TIMEOUT == rc) && (count < 300)); #else rc = VINF_SUCCESS; rcThread = VINF_SUCCESS; RT_NOREF_PV(count); #endif if (RT_SUCCESS(rc)) { AssertRC(rcThread); } else LogRel(("Shared Clipboard: Stopping X11 event thread failed with %Rrc\n", rc)); clipUninit(pCtx); RT_NOREF_PV(rcThread); return rc; } /** * Satisfies a request from X11 for clipboard targets supported by VBox. * * @returns VBox status code. * @param pCtx The X11 clipboard context to use. * @param atomTypeReturn The type of the data we are returning. * @param pValReturn A pointer to the data we are returning. This * should be set to memory allocated by XtMalloc, * which will be freed later by the Xt toolkit. * @param pcLenReturn The length of the data we are returning. * @param piFormatReturn The format (8bit, 16bit, 32bit) of the data we are * returning. * @note X11 backend code, called by the XtOwnSelection callback. */ static int clipCreateX11Targets(PSHCLX11CTX pCtx, Atom *atomTypeReturn, XtPointer *pValReturn, unsigned long *pcLenReturn, int *piFormatReturn) { LogFlowFuncEnter(); const unsigned cFixedTargets = 3; Atom *atomTargets = (Atom *)XtMalloc((MAX_CLIP_X11_FORMATS + cFixedTargets) * sizeof(Atom)); unsigned cTargets = 0; SHCLX11FMTIDX format = NIL_CLIPX11FORMAT; do { format = clipEnumX11Formats(pCtx->vboxFormats, format); if (format != NIL_CLIPX11FORMAT) { atomTargets[cTargets] = clipAtomForX11Format(pCtx, format); ++cTargets; } } while (format != NIL_CLIPX11FORMAT); /* We always offer these fixed targets. */ atomTargets[cTargets] = clipGetAtom(pCtx, "TARGETS"); atomTargets[cTargets + 1] = clipGetAtom(pCtx, "MULTIPLE"); atomTargets[cTargets + 2] = clipGetAtom(pCtx, "TIMESTAMP"); *atomTypeReturn = XA_ATOM; *pValReturn = (XtPointer)atomTargets; *pcLenReturn = cTargets + cFixedTargets; *piFormatReturn = 32; return VINF_SUCCESS; } /** * This is a wrapper around ShClX11RequestDataForX11Callback that will cache the * data returned. */ static int clipReadVBoxShCl(PSHCLX11CTX pCtx, SHCLFORMAT Format, void **ppv, uint32_t *pcb) { LogFlowFunc(("pCtx=%p, Format=%02X, ppv=%p, pcb=%p\n", pCtx, Format, ppv, pcb)); int rc = VINF_SUCCESS; if (Format == VBOX_SHCL_FMT_UNICODETEXT) { if (pCtx->pvUnicodeCache == NULL) rc = ShClX11RequestDataForX11Callback(pCtx->pFrontend, Format, &pCtx->pvUnicodeCache, &pCtx->cbUnicodeCache); if (RT_SUCCESS(rc)) { AssertPtrReturn(pCtx->pvUnicodeCache, VERR_INVALID_POINTER); AssertReturn (pCtx->cbUnicodeCache, VERR_INVALID_PARAMETER); *ppv = RTMemDup(pCtx->pvUnicodeCache, pCtx->cbUnicodeCache); *pcb = pCtx->cbUnicodeCache; if (*ppv == NULL) rc = VERR_NO_MEMORY; } } else rc = ShClX11RequestDataForX11Callback(pCtx->pFrontend, Format, ppv, pcb); if (RT_SUCCESS(rc)) LogFlowFunc(("*ppv=%.*ls, *pcb=%u\n", *pcb, *ppv, *pcb)); LogFlowFuncLeaveRC(rc); return rc; } /** * Calculates a buffer size large enough to hold the source Windows format * text converted into Unix Utf8, including the null terminator. * * @returns VBox status code. * @param pwsz The source text in UCS-2 with Windows EOLs. * @param cwc The size in USC-2 elements of the source text, with or * without the terminator. * @param pcbActual Where to store the buffer size needed. */ static int clipWinTxtBufSizeForUtf8(PRTUTF16 pwsz, size_t cwc, size_t *pcbActual) { size_t cbRet = 0; int rc = RTUtf16CalcUtf8LenEx(pwsz, cwc, &cbRet); if (RT_SUCCESS(rc)) *pcbActual = cbRet + 1; /* null terminator */ return rc; } /** * Converts text from Windows format (UCS-2 with CRLF line endings) to standard UTF-8. * * @returns VBox status code. * @param pwszSrc The text to be converted. * @param cbSrc The length of @a pwszSrc in bytes. * @param pszBuf Where to write the converted string. * @param cbBuf The size of the buffer pointed to by @a pszBuf. * @param pcbActual Where to store the size of the converted string. * optional. */ static int clipWinTxtToUtf8(PRTUTF16 pwszSrc, size_t cbSrc, char *pszBuf, size_t cbBuf, size_t *pcbActual) { PRTUTF16 pwszTmp = NULL; size_t cwSrc = cbSrc / 2, cwTmp = 0, cbDest = 0; int rc = VINF_SUCCESS; LogFlowFunc (("pwszSrc=%.*ls, cbSrc=%u\n", cbSrc, pwszSrc, cbSrc)); /* How long will the converted text be? */ AssertPtr(pwszSrc); AssertPtr(pszBuf); rc = ShClUtf16GetLinSize(pwszSrc, cwSrc, &cwTmp); if (RT_SUCCESS(rc) && cwTmp == 0) rc = VERR_NO_DATA; if (RT_SUCCESS(rc)) pwszTmp = (PRTUTF16)RTMemAlloc(cwTmp * 2); if (!pwszTmp) rc = VERR_NO_MEMORY; /* Convert the text. */ if (RT_SUCCESS(rc)) rc = ShClUtf16WinToLin(pwszSrc, cwSrc, pwszTmp, cwTmp); if (RT_SUCCESS(rc)) { /* Convert the UTF-16 string to Utf8. */ rc = RTUtf16ToUtf8Ex(pwszTmp + 1, cwTmp - 1, &pszBuf, cbBuf, &cbDest); } RTMemFree(reinterpret_cast(pwszTmp)); if (pcbActual) *pcbActual = cbDest + 1; if (RT_SUCCESS(rc)) LogFlowFunc (("converted string is %.*s. Returning.\n", cbDest, pszBuf)); LogFlowFuncLeaveRC(rc); return rc; } /** * Satisfies a request from X11 to convert the clipboard text to UTF-8. We * return null-terminated text, but can cope with non-null-terminated input. * * @returns VBox status code. * @param pDisplay An X11 display structure, needed for conversions * performed by Xlib. * @param pv The text to be converted (UCS-2 with Windows EOLs). * @param cb The length of the text in @cb in bytes. * @param atomTypeReturn Where to store the atom for the type of the data * we are returning. * @param pValReturn Where to store the pointer to the data we are * returning. This should be to memory allocated by * XtMalloc, which will be freed by the Xt toolkit * later. * @param pcLenReturn Where to store the length of the data we are * returning. * @param piFormatReturn Where to store the bit width (8, 16, 32) of the * data we are returning. */ static int clipWinTxtToUtf8ForX11CB(Display *pDisplay, PRTUTF16 pwszSrc, size_t cbSrc, Atom *atomTarget, Atom *atomTypeReturn, XtPointer *pValReturn, unsigned long *pcLenReturn, int *piFormatReturn) { RT_NOREF(pDisplay, pcLenReturn); /* This may slightly overestimate the space needed. */ size_t cbDest = 0; int rc = clipWinTxtBufSizeForUtf8(pwszSrc, cbSrc / 2, &cbDest); if (RT_SUCCESS(rc)) { char *pszDest = (char *)XtMalloc(cbDest); size_t cbActual = 0; if (pszDest) { rc = clipWinTxtToUtf8(pwszSrc, cbSrc, pszDest, cbDest, &cbActual); } else rc = VERR_NO_MEMORY; if (RT_SUCCESS(rc)) { *atomTypeReturn = *atomTarget; *pValReturn = (XtPointer)pszDest; *pcLenReturn = cbActual; *piFormatReturn = 8; } } return rc; } /** * Satisfies a request from X11 to convert the clipboard HTML fragment to UTF-8. We * return null-terminated text, but can cope with non-null-terminated input. * * @returns VBox status code. * @param pDisplay An X11 display structure, needed for conversions * performed by Xlib. * @param pv The text to be converted (UTF8 with Windows EOLs). * @param cb The length of the text in @cb in bytes. * @param atomTypeReturn Where to store the atom for the type of the data * we are returning. * @param pValReturn Where to store the pointer to the data we are * returning. This should be to memory allocated by * XtMalloc, which will be freed by the Xt toolkit later. * @param pcLenReturn Where to store the length of the data we are returning. * @param piFormatReturn Where to store the bit width (8, 16, 32) of the * data we are returning. */ static int clipWinHTMLToUtf8ForX11CB(Display *pDisplay, const char *pszSrc, size_t cbSrc, Atom *atomTarget, Atom *atomTypeReturn, XtPointer *pValReturn, unsigned long *pcLenReturn, int *piFormatReturn) { RT_NOREF(pDisplay, pValReturn); /* This may slightly overestimate the space needed. */ LogFlowFunc(("Source: %s", pszSrc)); char *pszDest = (char *)XtMalloc(cbSrc); if (pszDest == NULL) return VERR_NO_MEMORY; memcpy(pszDest, pszSrc, cbSrc); *atomTypeReturn = *atomTarget; *pValReturn = (XtPointer)pszDest; *pcLenReturn = cbSrc; *piFormatReturn = 8; return VINF_SUCCESS; } /** * Does this atom correspond to one of the two selection types we support? * * @param pCtx The X11 clipboard context to use. * @param selType The atom in question. */ static bool clipIsSupportedSelectionType(PSHCLX11CTX pCtx, Atom selType) { return( (selType == clipGetAtom(pCtx, "CLIPBOARD")) || (selType == clipGetAtom(pCtx, "PRIMARY"))); } /** * Removes a trailing nul character from a string by adjusting the string * length. Some X11 applications don't like zero-terminated text... * * @param pText The text in question. * @param pcText The length of the text, adjusted on return. * @param format The format of the text. */ static void clipTrimTrailingNul(XtPointer pText, unsigned long *pcText, SHCLX11FMT format) { AssertPtrReturnVoid(pText); AssertPtrReturnVoid(pcText); AssertReturnVoid((format == SHCLX11FMT_UTF8) || (format == SHCLX11FMT_TEXT) || (format == SHCLX11FMT_HTML)); if (((char *)pText)[*pcText - 1] == '\0') --(*pcText); } static int clipConvertVBoxCBForX11(PSHCLX11CTX pCtx, Atom *atomTarget, Atom *atomTypeReturn, XtPointer *pValReturn, unsigned long *pcLenReturn, int *piFormatReturn) { int rc = VINF_SUCCESS; SHCLX11FMTIDX x11Format = clipFindX11FormatByAtom(pCtx, *atomTarget); SHCLX11FMT clipFormat = clipRealFormatForX11Format(x11Format); LogFlowFunc(("fFormats=0x%x, x11Format=%u, clipFormat=%u\n", pCtx->vboxFormats, x11Format, clipFormat)); if ( ((clipFormat == SHCLX11FMT_UTF8) || (clipFormat == SHCLX11FMT_TEXT)) && (pCtx->vboxFormats & VBOX_SHCL_FMT_UNICODETEXT)) { void *pv = NULL; uint32_t cb = 0; rc = clipReadVBoxShCl(pCtx, VBOX_SHCL_FMT_UNICODETEXT, &pv, &cb); if (RT_SUCCESS(rc) && (cb == 0)) rc = VERR_NO_DATA; if (RT_SUCCESS(rc) && ((clipFormat == SHCLX11FMT_UTF8) || (clipFormat == SHCLX11FMT_TEXT))) rc = clipWinTxtToUtf8ForX11CB(XtDisplay(pCtx->pWidget), (PRTUTF16)pv, cb, atomTarget, atomTypeReturn, pValReturn, pcLenReturn, piFormatReturn); if (RT_SUCCESS(rc)) clipTrimTrailingNul(*(XtPointer *)pValReturn, pcLenReturn, clipFormat); RTMemFree(pv); } else if ( (clipFormat == SHCLX11FMT_BMP) && (pCtx->vboxFormats & VBOX_SHCL_FMT_BITMAP)) { void *pv = NULL; uint32_t cb = 0; rc = clipReadVBoxShCl(pCtx, VBOX_SHCL_FMT_BITMAP, &pv, &cb); if (RT_SUCCESS(rc) && (cb == 0)) rc = VERR_NO_DATA; if (RT_SUCCESS(rc) && (clipFormat == SHCLX11FMT_BMP)) { /* Create a full BMP from it */ rc = ShClDibToBmp(pv, cb, (void **)pValReturn, (size_t *)pcLenReturn); } else rc = VERR_NOT_SUPPORTED; if (RT_SUCCESS(rc)) { *atomTypeReturn = *atomTarget; *piFormatReturn = 8; } RTMemFree(pv); } else if ( (clipFormat == SHCLX11FMT_HTML) && (pCtx->vboxFormats & VBOX_SHCL_FMT_HTML)) { void *pv = NULL; uint32_t cb = 0; rc = clipReadVBoxShCl(pCtx, VBOX_SHCL_FMT_HTML, &pv, &cb); if (RT_SUCCESS(rc) && (cb == 0)) rc = VERR_NO_DATA; if (RT_SUCCESS(rc)) { /* * The common VBox HTML encoding will be - Utf8 * because it more general for HTML formats then UTF16 * X11 clipboard returns UTF-16, so before sending it we should * convert it to UTF8. * It's very strange but here we get UTF-16 from x11 clipboard * in same time we send UTF-8 to x11 clipboard and it's work. */ rc = clipWinHTMLToUtf8ForX11CB(XtDisplay(pCtx->pWidget), (const char*)pv, cb, atomTarget, atomTypeReturn, pValReturn, pcLenReturn, piFormatReturn); if (RT_SUCCESS(rc)) clipTrimTrailingNul(*(XtPointer *)pValReturn, pcLenReturn, clipFormat); RTMemFree(pv); } } #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS else if (pCtx->vboxFormats & VBOX_SHCL_FMT_URI_LIST) { switch (clipFormat) { case SHCLX11FMT_TEXT: RT_FALL_THROUGH(); case SHCLX11FMT_UTF8: RT_FALL_THROUGH(); case SHCLX11FMT_URI_LIST: { break; } default: rc = VERR_NOT_SUPPORTED; break; } } #endif else { *atomTypeReturn = XT_CONVERT_FAIL; *pValReturn = (XtPointer)NULL; *pcLenReturn = 0; *piFormatReturn = 0; rc = VERR_NOT_SUPPORTED; } if (RT_FAILURE(rc)) LogRel2(("Shared Clipboard: Converting format 0x%x for X11 (x11Format=%u, clipFormat=%u) failed, rc=%Rrc\n", pCtx->vboxFormats, x11Format, clipFormat, rc)); LogFlowFuncLeaveRC(rc); return rc; } /** * Returns VBox's clipboard data for an X11 client. * * @note Callback for XtOwnSelection. */ static Boolean clipXtConvertSelectionProc(Widget widget, Atom *atomSelection, Atom *atomTarget, Atom *atomTypeReturn, XtPointer *pValReturn, unsigned long *pcLenReturn, int *piFormatReturn) { LogFlowFuncEnter(); PSHCLX11CTX pCtx = clipLookupContext(widget); int rc = VINF_SUCCESS; if (!pCtx) return False; if (!clipIsSupportedSelectionType(pCtx, *atomSelection)) return False; if (*atomTarget == clipGetAtom(pCtx, "TARGETS")) rc = clipCreateX11Targets(pCtx, atomTypeReturn, pValReturn, pcLenReturn, piFormatReturn); else rc = clipConvertVBoxCBForX11(pCtx, atomTarget, atomTypeReturn, pValReturn, pcLenReturn, piFormatReturn); LogFlowFunc(("returning %RTbool, internal status code %Rrc\n", RT_SUCCESS(rc), rc)); return RT_SUCCESS(rc) ? True : False; } /** * Structure used to pass information about formats that VBox supports. */ typedef struct _CLIPNEWVBOXFORMATS { /** Context information for the X11 clipboard. */ PSHCLX11CTX pCtx; /** Formats supported by VBox. */ SHCLFORMATS Formats; } CLIPNEWVBOXFORMATS, *PCLIPNEWVBOXFORMATS; /** Invalidates the local cache of the data in the VBox clipboard. */ static void clipInvalidateVBoxCBCache(PSHCLX11CTX pCtx) { if (pCtx->pvUnicodeCache != NULL) { RTMemFree(pCtx->pvUnicodeCache); pCtx->pvUnicodeCache = NULL; } } /** * Takes possession of the X11 clipboard (and middle-button selection). */ static void clipGrabX11CB(PSHCLX11CTX pCtx, SHCLFORMATS Formats) { LogFlowFuncEnter(); if (XtOwnSelection(pCtx->pWidget, clipGetAtom(pCtx, "CLIPBOARD"), CurrentTime, clipXtConvertSelectionProc, NULL, 0)) { pCtx->vboxFormats = Formats; /* Grab the middle-button paste selection too. */ XtOwnSelection(pCtx->pWidget, clipGetAtom(pCtx, "PRIMARY"), CurrentTime, clipXtConvertSelectionProc, NULL, 0); #ifndef TESTCASE /* Xt suppresses these if we already own the clipboard, so send them * ourselves. */ XSetSelectionOwner(XtDisplay(pCtx->pWidget), clipGetAtom(pCtx, "CLIPBOARD"), XtWindow(pCtx->pWidget), CurrentTime); XSetSelectionOwner(XtDisplay(pCtx->pWidget), clipGetAtom(pCtx, "PRIMARY"), XtWindow(pCtx->pWidget), CurrentTime); #endif } } /** * Worker function for ShClX11ReportFormatsToX11 which runs on the * event thread. * * @param pUserData Pointer to a CLIPNEWVBOXFORMATS structure containing * information about the VBox formats available and the * clipboard context data. Must be freed by the worker. */ static void ShClX11ReportFormatsToX11Worker(void *pUserData, void * /* interval */) { CLIPNEWVBOXFORMATS *pFormats = (CLIPNEWVBOXFORMATS *)pUserData; PSHCLX11CTX pCtx = pFormats->pCtx; uint32_t fFormats = pFormats->Formats; RTMemFree(pFormats); LogFlowFunc (("fFormats=0x%x\n", fFormats)); clipInvalidateVBoxCBCache(pCtx); clipGrabX11CB(pCtx, fFormats); clipResetX11Formats(pCtx); LogFlowFuncLeave(); } /** * Announces new clipboard formats to the host. * * @returns VBox status code. * @param Formats Clipboard formats offered. */ int ShClX11ReportFormatsToX11(PSHCLX11CTX pCtx, uint32_t Formats) { /* * Immediately return if we are not connected to the X server. */ if (!pCtx->fHaveX11) return VINF_SUCCESS; int rc; /* This must be free'd by the worker callback. */ PCLIPNEWVBOXFORMATS pFormats = (PCLIPNEWVBOXFORMATS)RTMemAlloc(sizeof(CLIPNEWVBOXFORMATS)); if (pFormats) { pFormats->pCtx = pCtx; pFormats->Formats = Formats; rc = clipQueueToEventThread(pCtx, ShClX11ReportFormatsToX11Worker, (XtPointer) pFormats); } else rc = VERR_NO_MEMORY; LogFlowFuncLeaveRC(rc); return rc; } /** * Massages generic UTF-16 with CR end-of-lines into the format Windows expects * and return the result in a RTMemAlloc allocated buffer. * * @returns VBox status code. * @param pwcSrc The source as UTF-16. * @param cwcSrc The number of 16bit elements in @a pwcSrc, not counting * the terminating zero. * @param ppwszDest Where to store the buffer address. * @param pcbDest On success, where to store the number of bytes written. * Undefined otherwise. Optional. */ static int clipUtf16ToWinTxt(RTUTF16 *pwcSrc, size_t cwcSrc, PRTUTF16 *ppwszDest, uint32_t *pcbDest) { AssertPtrReturn(pwcSrc, VERR_INVALID_POINTER); AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER); LogFlowFunc(("pwcSrc=%p, cwcSrc=%u, ppwszDest=%p\n", pwcSrc, cwcSrc, ppwszDest)); if (pcbDest) *pcbDest = 0; PRTUTF16 pwszDest = NULL; size_t cwcDest; int rc = ShClUtf16GetWinSize(pwcSrc, cwcSrc + 1, &cwcDest); if (RT_SUCCESS(rc)) { pwszDest = (PRTUTF16)RTMemAlloc(cwcDest * sizeof(RTUTF16)); if (!pwszDest) rc = VERR_NO_MEMORY; } if (RT_SUCCESS(rc)) rc = ShClUtf16LinToWin(pwcSrc, cwcSrc + 1, pwszDest, cwcDest); if (RT_SUCCESS(rc)) { LogFlowFunc(("Converted string is %.*ls\n", cwcDest, pwszDest)); *ppwszDest = pwszDest; if (pcbDest) *pcbDest = cwcDest * sizeof(RTUTF16); } else RTMemFree(pwszDest); LogFlowFuncLeaveRC(rc); return rc; } /** * Converts UTF-8 text with CR end-of-lines into UTF-16 as Windows expects it * and return the result in a RTMemAlloc allocated buffer. * * @returns VBox status code. * @param pcSrc The source UTF-8. * @param cbSrc The size of the source in bytes, not counting the * terminating zero. * @param ppwszDest Where to store the buffer address. * @param pcbDest On success, where to store the number of bytes written. * Undefined otherwise. Optional. */ static int clipUtf8ToWinTxt(const char *pcSrc, unsigned cbSrc, PRTUTF16 *ppwszDest, uint32_t *pcbDest) { AssertPtrReturn(pcSrc, VERR_INVALID_POINTER); AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER); LogFlowFunc(("pcSrc=%p, cbSrc=%u, ppwszDest=%p\n", pcSrc, cbSrc, ppwszDest)); if (pcbDest) *pcbDest = 0; /* Intermediate conversion to UTF-16. */ size_t cwcTmp; PRTUTF16 pwcTmp = NULL; int rc = RTStrToUtf16Ex(pcSrc, cbSrc, &pwcTmp, 0, &cwcTmp); if (RT_SUCCESS(rc)) rc = clipUtf16ToWinTxt(pwcTmp, cwcTmp, ppwszDest, pcbDest); RTUtf16Free(pwcTmp); LogFlowFuncLeaveRC(rc); return rc; } /** * Converts Latin-1 text with CR end-of-lines into UTF-16 as Windows expects * it and return the result in a RTMemAlloc allocated buffer. * * @returns VBox status code. * @param pcSrc The source text. * @param cbSrc The size of the source in bytes, not counting the * terminating zero. * @param ppwszDest Where to store the buffer address. * @param pcbDest On success, where to store the number of bytes written. * Undefined otherwise. Optional. */ static int clipLatin1ToWinTxt(char *pcSrc, unsigned cbSrc, PRTUTF16 *ppwszDest, uint32_t *pcbDest) { AssertPtrReturn(pcSrc, VERR_INVALID_POINTER); AssertPtrReturn(ppwszDest, VERR_INVALID_POINTER); LogFlowFunc(("pcSrc=%.*s, cbSrc=%u, ppwszDest=%p\n", cbSrc, (char *) pcSrc, cbSrc, ppwszDest)); PRTUTF16 pwszDest = NULL; int rc = VINF_SUCCESS; /* Calculate the space needed. */ unsigned cwcDest = 0; for (unsigned i = 0; i < cbSrc && pcSrc[i] != '\0'; ++i) { if (pcSrc[i] == VBOX_SHCL_LINEFEED) cwcDest += 2; else ++cwcDest; } ++cwcDest; /* Leave space for the terminator. */ if (pcbDest) *pcbDest = cwcDest * sizeof(RTUTF16); pwszDest = (PRTUTF16) RTMemAlloc(cwcDest * sizeof(RTUTF16)); if (!pwszDest) rc = VERR_NO_MEMORY; /* And do the conversion, bearing in mind that Latin-1 expands "naturally" * to UTF-16. */ if (RT_SUCCESS(rc)) { for (unsigned i = 0, j = 0; i < cbSrc; ++i, ++j) { if (pcSrc[i] != VBOX_SHCL_LINEFEED) pwszDest[j] = pcSrc[i]; else { pwszDest[j] = VBOX_SHCL_CARRIAGERETURN; pwszDest[j + 1] = VBOX_SHCL_LINEFEED; ++j; } } pwszDest[cwcDest - 1] = '\0'; /* Make sure we are zero-terminated. */ LogFlowFunc(("Converted text is %.*ls\n", cwcDest, pwszDest)); } if (RT_SUCCESS(rc)) { *ppwszDest = pwszDest; } else RTMemFree(pwszDest); LogFlowFuncLeaveRC(rc); return rc; } /** * Converts UTF-16 text into UTF-8 as Windows expects * it and return the result in a RTMemAlloc allocated buffer. * * @returns VBox status code. * @param pcSrc The source text. * @param cbSrc The size of the source in bytes, not counting the * terminating zero. * @param ppwszDest Where to store the buffer address. * @param pcbDest On success, where to store the number of bytes written. * Undefined otherwise. Optional. */ static int clipUTF16ToWinHTML(RTUTF16 *pwcBuf, size_t cb, char **ppszOut, uint32_t *pcOut) { AssertPtrReturn(pwcBuf, VERR_INVALID_POINTER); AssertReturn (cb, VERR_INVALID_PARAMETER); AssertPtrReturn(ppszOut, VERR_INVALID_POINTER); AssertPtrReturn(pcOut, VERR_INVALID_POINTER); if (cb % 2) return VERR_INVALID_PARAMETER; size_t cwc = cb / 2; size_t i = 0; RTUTF16 *pwc = pwcBuf; char *pchRes = NULL; size_t cRes = 0; LogFlowFunc(("src= %ls cb=%d i=%i, %x %x\n", pwcBuf, cb, i, ppszOut, pcOut)); while (i < cwc) { /* find zero symbol (end of string) */ for (; i < cwc && pwcBuf[i] != 0; i++) ; LogFlowFunc(("skipped nulls i=%d cwc=%d\n", i, cwc)); /* convert found string */ char *psz = NULL; size_t cch = 0; int rc = RTUtf16ToUtf8Ex(pwc, cwc, &psz, pwc - pwcBuf, &cch); LogFlowFunc(("utf16toutf8 src= %ls res=%s i=%i\n", pwc, psz, i)); if (RT_FAILURE(rc)) { RTMemFree(pchRes); return rc; } /* append new substring */ char *pchNew = (char *)RTMemRealloc(pchRes, cRes + cch + 1); if (!pchNew) { RTMemFree(pchRes); RTStrFree(psz); return VERR_NO_MEMORY; } pchRes = pchNew; memcpy(pchRes + cRes, psz, cch + 1); LogFlowFunc(("Temp result res=%s\n", pchRes + cRes)); /* remove temporary buffer */ RTStrFree(psz); cRes += cch + 1; /* skip zero symbols */ for (; i < cwc && pwcBuf[i] == 0; i++) ; /* remember start of string */ pwc += i; } *ppszOut = pchRes; *pcOut = cRes; return VINF_SUCCESS; } /** * A structure containing information about where to store a request * for the X11 clipboard contents. */ typedef struct _CLIPREADX11CBREQ { /** The format VBox would like the data in. */ SHCLFORMAT mFormat; /** The format we requested from X11. */ SHCLX11FMTIDX mX11Format; /** The clipboard context this request is associated with. */ SHCLX11CTX *mpCtx; /** The request structure passed in from the backend. */ CLIPREADCBREQ *mpReq; } CLIPREADX11CBREQ; /** * Converts the data obtained from the X11 clipboard to the required format, * place it in the buffer supplied and signal that data has arrived. * * Converts the text obtained UTF-16LE with Windows EOLs. * Converts full BMP data to DIB format. * * @note Callback for XtGetSelectionValue, for use when * the X11 clipboard contains a format we understand. */ static void clipConvertDataFromX11CallbackWorker(void *pClient, void *pvSrc, unsigned cbSrc) { CLIPREADX11CBREQ *pReq = (CLIPREADX11CBREQ *)pClient; LogFlowFunc(("pReq->mFormat=%02X, pReq->mX11Format=%u, pReq->mCtx=%p\n", pReq->mFormat, pReq->mX11Format, pReq->mpCtx)); AssertPtr(pReq->mpCtx); Assert(pReq->mFormat != VBOX_SHCL_FMT_NONE); /* Sanity. */ int rc = VINF_SUCCESS; void *pvDst = NULL; uint32_t cbDst = 0; if (pvSrc == NULL) { /* The clipboard selection may have changed before we could get it. */ rc = VERR_NO_DATA; } else if (pReq->mFormat == VBOX_SHCL_FMT_UNICODETEXT) { /* In which format is the clipboard data? */ switch (clipRealFormatForX11Format(pReq->mX11Format)) { case SHCLX11FMT_UTF8: RT_FALL_THROUGH(); case SHCLX11FMT_TEXT: { /* If we are given broken UTF-8, we treat it as Latin1. */ /** @todo Is this acceptable? */ if (RT_SUCCESS(RTStrValidateEncodingEx((char *)pvSrc, cbSrc, 0))) { rc = clipUtf8ToWinTxt((const char *)pvSrc, cbSrc, (PRTUTF16 *)&pvDst, &cbDst); } else { rc = clipLatin1ToWinTxt((char *)pvSrc, cbSrc, (PRTUTF16 *)&pvDst, &cbDst); } break; } default: { rc = VERR_INVALID_PARAMETER; break; } } } else if (pReq->mFormat == VBOX_SHCL_FMT_BITMAP) { /* In which format is the clipboard data? */ switch (clipRealFormatForX11Format(pReq->mX11Format)) { case SHCLX11FMT_BMP: { const void *pDib; size_t cbDibSize; rc = ShClBmpGetDib((const void *)pvSrc, cbSrc, &pDib, &cbDibSize); if (RT_SUCCESS(rc)) { pvDst = RTMemAlloc(cbDibSize); if (!pvDst) rc = VERR_NO_MEMORY; else { memcpy(pvDst, pDib, cbDibSize); cbDst = cbDibSize; } } break; } default: { rc = VERR_INVALID_PARAMETER; break; } } } else if (pReq->mFormat == VBOX_SHCL_FMT_HTML) { /* In which format is the clipboard data? */ switch (clipRealFormatForX11Format(pReq->mX11Format)) { case SHCLX11FMT_HTML: { /* * The common VBox HTML encoding will be - UTF-8 * because it more general for HTML formats then UTF-16 * X11 clipboard returns UTF-16, so before sending it we should * convert it to UTF-8. */ pvDst = NULL; cbDst = 0; /* * Some applications sends data in UTF-16, some in UTF-8, * without indication it in MIME. * But in case of UTF-16, at least an OpenOffice adds Byte Order Mark - 0xfeff * at start of clipboard data. */ if ( cbSrc >= sizeof(RTUTF16) && *(PRTUTF16)pvSrc == 0xfeff) { LogFlowFunc((" \n")); rc = clipUTF16ToWinHTML((RTUTF16 *)pvSrc, cbSrc, (char**)&pvDst, &cbDst); } else { pvDst = RTMemAlloc(cbSrc); if(pvDst) { memcpy(pvDst, pvSrc, cbSrc); cbDst = cbSrc; } else { rc = VERR_NO_MEMORY; break; } } LogFlowFunc(("Source unicode %ls, cbSrc = %d\n, Byte Order Mark = %hx", pvSrc, cbSrc, ((PRTUTF16)pvSrc)[0])); LogFlowFunc(("converted to win unicode %s, cbDest = %d, rc = %Rrc\n", pvDst, cbDst, rc)); rc = VINF_SUCCESS; break; } default: { rc = VERR_INVALID_PARAMETER; break; } } } #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS else if (pReq->mFormat == VBOX_SHCL_FMT_URI_LIST) { /* In which format is the clipboard data? */ switch (clipRealFormatForX11Format(pReq->mX11Format)) { case SHCLX11FMT_URI_LIST: { /* For URI lists we only accept valid UTF-8 encodings. */ if (RT_SUCCESS(RTStrValidateEncodingEx((char *)pvSrc, cbSrc, 0))) { /* URI lists on X are string separated with "\r\n". */ RTCList lstRootEntries = RTCString((char *)pvSrc, cbSrc).split("\r\n"); for (size_t i = 0; i < lstRootEntries.size(); ++i) { char *pszEntry = RTUriFilePath(lstRootEntries.at(i).c_str()); AssertPtrBreakStmt(pszEntry, VERR_INVALID_PARAMETER); LogFlowFunc(("URI list entry '%s'\n", pszEntry)); rc = RTStrAAppend((char **)&pvDst, pszEntry); AssertRCBreakStmt(rc, VERR_NO_MEMORY); cbDst += (uint32_t)strlen(pszEntry); rc = RTStrAAppend((char **)&pvDst, "\r\n"); AssertRCBreakStmt(rc, VERR_NO_MEMORY); cbDst += (uint32_t)strlen("\r\n"); RTStrFree(pszEntry); } if (cbDst) cbDst++; /* Include final (zero) termination. */ LogFlowFunc(("URI list: cbDst=%RU32\n", cbDst)); } else rc = VERR_INVALID_PARAMETER; break; } default: { rc = VERR_INVALID_PARAMETER; break; } } } #endif else rc = VERR_NOT_SUPPORTED; ShClX11RequestFromX11CompleteCallback(pReq->mpCtx->pFrontend, rc, pReq->mpReq, pvDst, cbDst); RTMemFree(pvDst); RTMemFree(pReq); LogFlowFuncLeaveRC(rc); } #ifndef TESTCASE /** * Converts the data obtained from the X11 clipboard to the required format, * place it in the buffer supplied and signal that data has arrived. * * Converts the text obtained UTF-16LE with Windows EOLs. * Converts full BMP data to DIB format. * * @note Callback for XtGetSelectionValue(), for use when * the X11 clipboard contains a format we understand. */ static void clipConvertDataFromX11Callback(Widget widget, XtPointer pClient, Atom * /* selection */, Atom *atomType, XtPointer pvSrc, long unsigned int *pcLen, int *piFormat) { RT_NOREF(widget); if (*atomType == XT_CONVERT_FAIL) /* Xt timeout */ clipConvertDataFromX11CallbackWorker(pClient, NULL, 0); else clipConvertDataFromX11CallbackWorker(pClient, pvSrc, (*pcLen) * (*piFormat) / 8); XtFree((char *)pvSrc); } #endif #ifdef TESTCASE static void tstClipRequestData(SHCLX11CTX* pCtx, SHCLX11FMTIDX target, void *closure); #endif static int clipGetSelectionValue(PSHCLX11CTX pCtx, SHCLX11FMTIDX format, CLIPREADX11CBREQ *pReq) { #ifndef TESTCASE XtGetSelectionValue(pCtx->pWidget, clipGetAtom(pCtx, "CLIPBOARD"), clipAtomForX11Format(pCtx, format), clipConvertDataFromX11Callback, reinterpret_cast(pReq), CurrentTime); #else tstClipRequestData(pCtx, format, (void *)pReq); #endif return VINF_SUCCESS; /** @todo Return real rc. */ } /** * Worker function for ShClX11ReadDataFromX11 which runs on the event thread. */ static void ShClX11ReadDataFromX11Worker(void *pvUserData, void * /* interval */) { AssertPtrReturnVoid(pvUserData); CLIPREADX11CBREQ *pReq = (CLIPREADX11CBREQ *)pvUserData; SHCLX11CTX *pCtx = pReq->mpCtx; LogFlowFunc(("pReq->mFormat = %02x\n", pReq->mFormat)); int rc = VERR_NO_DATA; /* VBox thinks we have data and we don't. */ if (pReq->mFormat & VBOX_SHCL_FMT_UNICODETEXT) { pReq->mX11Format = pCtx->X11TextFormat; if (pReq->mX11Format != SHCLX11FMT_INVALID) { /* Send out a request for the data to the current clipboard owner. */ rc = clipGetSelectionValue(pCtx, pCtx->X11TextFormat, pReq); } } else if (pReq->mFormat & VBOX_SHCL_FMT_BITMAP) { pReq->mX11Format = pCtx->X11BitmapFormat; if (pReq->mX11Format != SHCLX11FMT_INVALID) { /* Send out a request for the data to the current clipboard owner. */ rc = clipGetSelectionValue(pCtx, pCtx->X11BitmapFormat, pReq); } } else if (pReq->mFormat & VBOX_SHCL_FMT_HTML) { pReq->mX11Format = pCtx->X11HTMLFormat; if (pReq->mX11Format != SHCLX11FMT_INVALID) { /* Send out a request for the data to the current clipboard owner. */ rc = clipGetSelectionValue(pCtx, pCtx->X11HTMLFormat, pReq); } } #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS else if (pReq->mFormat & VBOX_SHCL_FMT_URI_LIST) { pReq->mX11Format = pCtx->X11URIListFormat; if (pReq->mX11Format != SHCLX11FMT_INVALID) { /* Send out a request for the data to the current clipboard owner. */ rc = clipGetSelectionValue(pCtx, pCtx->X11URIListFormat, pReq); } } #endif else { rc = VERR_NOT_IMPLEMENTED; } if (RT_FAILURE(rc)) { /* The clipboard callback was never scheduled, so we must signal * that the request processing is finished and clean up ourselves. */ ShClX11RequestFromX11CompleteCallback(pReq->mpCtx->pFrontend, rc, pReq->mpReq, NULL /* pv */ ,0 /* cb */); RTMemFree(pReq); } LogFlowFuncLeaveRC(rc); } /** * Called when VBox wants to read the X11 clipboard. * * @returns VBox status code. * @param pCtx Context data for the clipboard backend. * @param Format The format that the VBox would like to receive the data in. * @param pReq Read callback request to use. Must be free'd in the callback. * * @note We allocate a request structure which must be freed by the worker. */ int ShClX11ReadDataFromX11(PSHCLX11CTX pCtx, SHCLFORMAT Format, CLIPREADCBREQ *pReq) { /* * Immediately return if we are not connected to the X server. */ if (!pCtx->fHaveX11) return VERR_NO_DATA; int rc = VINF_SUCCESS; CLIPREADX11CBREQ *pX11Req = (CLIPREADX11CBREQ *)RTMemAllocZ(sizeof(CLIPREADX11CBREQ)); if (pX11Req) { pX11Req->mpCtx = pCtx; pX11Req->mFormat = Format; pX11Req->mpReq = pReq; /* We use this to schedule a worker function on the event thread. */ rc = clipQueueToEventThread(pCtx, ShClX11ReadDataFromX11Worker, (XtPointer)pX11Req); } else rc = VERR_NO_MEMORY; LogFlowFuncLeaveRC(rc); return rc; } #ifdef TESTCASE /** @todo This unit test currently works by emulating the X11 and X toolkit * APIs to exercise the code, since I didn't want to rewrite the code too much * when I wrote the tests. However, this makes it rather ugly and hard to * understand. Anyone doing any work on the code should feel free to * rewrite the tests and the code to make them cleaner and more readable. */ #include #include #define TESTCASE_WIDGET_ID (Widget)0xffff /* For the purpose of the test case, we just execute the procedure to be * scheduled, as we are running single threaded. */ void tstClipQueueToEventThread(void (*proc)(void *, void *), void *client_data) { proc(client_data, NULL); } void XtFree(char *ptr) { RTMemFree((void *)ptr); } /* The data in the simulated VBox clipboard. */ static int g_tst_rcDataVBox = VINF_SUCCESS; static void *g_tst_pvDataVBox = NULL; static uint32_t g_tst_cbDataVBox = 0; /* Set empty data in the simulated VBox clipboard. */ static void tstClipEmptyVBox(PSHCLX11CTX pCtx, int retval) { g_tst_rcDataVBox = retval; RTMemFree(g_tst_pvDataVBox); g_tst_pvDataVBox = NULL; g_tst_cbDataVBox = 0; ShClX11ReportFormatsToX11(pCtx, 0); } /* Set the data in the simulated VBox clipboard. */ static int tstClipSetVBoxUtf16(PSHCLX11CTX pCtx, int retval, const char *pcszData, size_t cb) { PRTUTF16 pwszData = NULL; size_t cwData = 0; int rc = RTStrToUtf16Ex(pcszData, RTSTR_MAX, &pwszData, 0, &cwData); if (RT_FAILURE(rc)) return rc; AssertReturn(cb <= cwData * 2 + 2, VERR_BUFFER_OVERFLOW); void *pv = RTMemDup(pwszData, cb); RTUtf16Free(pwszData); if (pv == NULL) return VERR_NO_MEMORY; if (g_tst_pvDataVBox) RTMemFree(g_tst_pvDataVBox); g_tst_rcDataVBox = retval; g_tst_pvDataVBox = pv; g_tst_cbDataVBox = cb; ShClX11ReportFormatsToX11(pCtx, VBOX_SHCL_FMT_UNICODETEXT); return VINF_SUCCESS; } /* Return the data in the simulated VBox clipboard. */ DECLCALLBACK(int) ShClX11RequestDataForX11Callback(PSHCLCONTEXT pCtx, uint32_t Format, void **ppv, uint32_t *pcb) { RT_NOREF(pCtx, Format); *pcb = g_tst_cbDataVBox; if (g_tst_pvDataVBox != NULL) { void *pv = RTMemDup(g_tst_pvDataVBox, g_tst_cbDataVBox); *ppv = pv; return pv != NULL ? g_tst_rcDataVBox : VERR_NO_MEMORY; } *ppv = NULL; return g_tst_rcDataVBox; } Display *XtDisplay(Widget w) { NOREF(w); return (Display *) 0xffff; } void XtAppSetExitFlag(XtAppContext app_context) { NOREF(app_context); } void XtDestroyWidget(Widget w) { NOREF(w); } XtAppContext XtCreateApplicationContext(void) { return (XtAppContext)0xffff; } void XtDestroyApplicationContext(XtAppContext app_context) { NOREF(app_context); } void XtToolkitInitialize(void) {} Boolean XtToolkitThreadInitialize(void) { return True; } Display *XtOpenDisplay(XtAppContext app_context, _Xconst _XtString display_string, _Xconst _XtString application_name, _Xconst _XtString application_class, XrmOptionDescRec *options, Cardinal num_options, int *argc, char **argv) { RT_NOREF8(app_context, display_string, application_name, application_class, options, num_options, argc, argv); return (Display *)0xffff; } Widget XtVaAppCreateShell(_Xconst _XtString application_name, _Xconst _XtString application_class, WidgetClass widget_class, Display *display, ...) { RT_NOREF(application_name, application_class, widget_class, display); return TESTCASE_WIDGET_ID; } void XtSetMappedWhenManaged(Widget widget, _XtBoolean mapped_when_managed) { RT_NOREF(widget, mapped_when_managed); } void XtRealizeWidget(Widget widget) { NOREF(widget); } XtInputId XtAppAddInput(XtAppContext app_context, int source, XtPointer condition, XtInputCallbackProc proc, XtPointer closure) { RT_NOREF(app_context, source, condition, proc, closure); return 0xffff; } /* Atoms we need other than the formats we support. */ static const char *g_tst_apszSupAtoms[] = { "PRIMARY", "CLIPBOARD", "TARGETS", "MULTIPLE", "TIMESTAMP" }; /* This just looks for the atom names in a couple of tables and returns an * index with an offset added. */ Atom XInternAtom(Display *, const char *pcsz, int) { Atom atom = 0; for (unsigned i = 0; i < RT_ELEMENTS(g_aFormats); ++i) if (!strcmp(pcsz, g_aFormats[i].pcszAtom)) atom = (Atom) (i + 0x1000); for (unsigned i = 0; i < RT_ELEMENTS(g_tst_apszSupAtoms); ++i) if (!strcmp(pcsz, g_tst_apszSupAtoms[i])) atom = (Atom) (i + 0x2000); Assert(atom); /* Have we missed any atoms? */ return atom; } /* Take a request for the targets we are currently offering. */ static SHCLX11FMTIDX g_tst_aSelTargetsIdx[10] = { 0 }; static size_t g_tst_cTargets = 0; void tstRequestTargets(SHCLX11CTX* pCtx) { clipUpdateX11Targets(pCtx, g_tst_aSelTargetsIdx, g_tst_cTargets); } /* The current values of the X selection, which will be returned to the * XtGetSelectionValue callback. */ static Atom g_tst_atmSelType = 0; static const void *g_tst_pSelData = NULL; static unsigned long g_tst_cSelData = 0; static int g_tst_selFormat = 0; void tstClipRequestData(PSHCLX11CTX pCtx, SHCLX11FMTIDX target, void *closure) { RT_NOREF(pCtx); unsigned long count = 0; int format = 0; if (target != g_tst_aSelTargetsIdx[0]) { clipConvertDataFromX11CallbackWorker(closure, NULL, 0); /* Could not convert to target. */ return; } void *pValue = NULL; pValue = g_tst_pSelData ? RTMemDup(g_tst_pSelData, g_tst_cSelData) : NULL; count = g_tst_pSelData ? g_tst_cSelData : 0; format = g_tst_selFormat; if (!pValue) { count = 0; format = 0; } clipConvertDataFromX11CallbackWorker(closure, pValue, count * format / 8); if (pValue) RTMemFree(pValue); } /* The formats currently on offer from X11 via the shared clipboard. */ static uint32_t g_tst_uX11Formats = 0; DECLCALLBACK(void) ShClX11ReportFormatsCallback(PSHCLCONTEXT pCtx, SHCLFORMATS Formats) { RT_NOREF(pCtx); g_tst_uX11Formats = Formats; } static uint32_t tstClipQueryFormats(void) { return g_tst_uX11Formats; } static void tstClipInvalidateFormats(void) { g_tst_uX11Formats = ~0; } /* Does our clipboard code currently own the selection? */ static bool g_tst_fOwnsSel = false; /* The procedure that is called when we should convert the selection to a * given format. */ static XtConvertSelectionProc g_tst_pfnSelConvert = NULL; /* The procedure which is called when we lose the selection. */ static XtLoseSelectionProc g_tst_pfnSelLose = NULL; /* The procedure which is called when the selection transfer has completed. */ static XtSelectionDoneProc g_tst_pfnSelDone = NULL; Boolean XtOwnSelection(Widget widget, Atom selection, Time time, XtConvertSelectionProc convert, XtLoseSelectionProc lose, XtSelectionDoneProc done) { RT_NOREF(widget, time); if (selection != XInternAtom(NULL, "CLIPBOARD", 0)) return True; /* We don't really care about this. */ g_tst_fOwnsSel = true; /* Always succeed. */ g_tst_pfnSelConvert = convert; g_tst_pfnSelLose = lose; g_tst_pfnSelDone = done; return True; } void XtDisownSelection(Widget widget, Atom selection, Time time) { RT_NOREF(widget, time, selection); g_tst_fOwnsSel = false; g_tst_pfnSelConvert = NULL; g_tst_pfnSelLose = NULL; g_tst_pfnSelDone = NULL; } /* Request the shared clipboard to convert its data to a given format. */ static bool tstClipConvertSelection(const char *pcszTarget, Atom *type, XtPointer *value, unsigned long *length, int *format) { Atom target = XInternAtom(NULL, pcszTarget, 0); if (target == 0) return false; /* Initialise all return values in case we make a quick exit. */ *type = XA_STRING; *value = NULL; *length = 0; *format = 0; if (!g_tst_fOwnsSel) return false; if (!g_tst_pfnSelConvert) return false; Atom clipAtom = XInternAtom(NULL, "CLIPBOARD", 0); if (!g_tst_pfnSelConvert(TESTCASE_WIDGET_ID, &clipAtom, &target, type, value, length, format)) return false; if (g_tst_pfnSelDone) g_tst_pfnSelDone(TESTCASE_WIDGET_ID, &clipAtom, &target); return true; } /* Set the current X selection data */ static void tstClipSetSelectionValues(const char *pcszTarget, Atom type, const void *data, unsigned long count, int format) { Atom clipAtom = XInternAtom(NULL, "CLIPBOARD", 0); g_tst_aSelTargetsIdx[0] = clipFindX11FormatByAtomText(pcszTarget); g_tst_cTargets = 1; g_tst_atmSelType = type; g_tst_pSelData = data; g_tst_cSelData = count; g_tst_selFormat = format; if (g_tst_pfnSelLose) g_tst_pfnSelLose(TESTCASE_WIDGET_ID, &clipAtom); g_tst_fOwnsSel = false; } static void tstClipSendTargetUpdate(PSHCLX11CTX pCtx) { clipQueryX11FormatsCallback(pCtx); } /* Configure if and how the X11 TARGETS clipboard target will fail. */ static void tstClipSetTargetsFailure(void) { g_tst_cTargets = 0; } char *XtMalloc(Cardinal size) { return (char *) RTMemAlloc(size); } char *XGetAtomName(Display *display, Atom atom) { RT_NOREF(display); AssertReturn((unsigned)atom < RT_ELEMENTS(g_aFormats) + 1, NULL); const char *pcszName = NULL; if (atom < 0x1000) return NULL; if (0x1000 <= atom && atom < 0x2000) { unsigned index = atom - 0x1000; AssertReturn(index < RT_ELEMENTS(g_aFormats), NULL); pcszName = g_aFormats[index].pcszAtom; } else { unsigned index = atom - 0x2000; AssertReturn(index < RT_ELEMENTS(g_tst_apszSupAtoms), NULL); pcszName = g_tst_apszSupAtoms[index]; } return (char *)RTMemDup(pcszName, sizeof(pcszName) + 1); } int XFree(void *data) { RTMemFree(data); return 0; } void XFreeStringList(char **list) { if (list) RTMemFree(*list); RTMemFree(list); } #define TESTCASE_MAX_BUF_SIZE 256 static int g_tst_rcCompleted = VINF_SUCCESS; static int g_tst_cbCompleted = 0; static CLIPREADCBREQ *g_tst_pCompletedReq = NULL; static char g_tst_abCompletedBuf[TESTCASE_MAX_BUF_SIZE]; void ShClX11RequestFromX11CompleteCallback(PSHCLCONTEXT pCtx, int rc, CLIPREADCBREQ *pReq, void *pv, uint32_t cb) { RT_NOREF(pCtx); if (cb <= TESTCASE_MAX_BUF_SIZE) { g_tst_rcCompleted = rc; if (cb != 0) memcpy(g_tst_abCompletedBuf, pv, cb); } else g_tst_rcCompleted = VERR_BUFFER_OVERFLOW; g_tst_cbCompleted = cb; g_tst_pCompletedReq = pReq; } static void tstClipGetCompletedRequest(int *prc, char ** ppc, uint32_t *pcb, CLIPREADCBREQ **ppReq) { *prc = g_tst_rcCompleted; *ppc = g_tst_abCompletedBuf; *pcb = g_tst_cbCompleted; *ppReq = g_tst_pCompletedReq; } #ifdef RT_OS_SOLARIS_10 char XtStrings [] = ""; _WidgetClassRec* applicationShellWidgetClass; char XtShellStrings [] = ""; int XmbTextPropertyToTextList( Display* /* display */, XTextProperty* /* text_prop */, char*** /* list_return */, int* /* count_return */ ) { return 0; } #else const char XtStrings [] = ""; _WidgetClassRec* applicationShellWidgetClass; const char XtShellStrings [] = ""; #endif static void tstStringFromX11(RTTEST hTest, PSHCLX11CTX pCtx, const char *pcszExp, int rcExp) { bool retval = true; tstClipSendTargetUpdate(pCtx); if (tstClipQueryFormats() != VBOX_SHCL_FMT_UNICODETEXT) RTTestFailed(hTest, "Wrong targets reported: %02X\n", tstClipQueryFormats()); else { char *pc; CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL; ShClX11ReadDataFromX11(pCtx, VBOX_SHCL_FMT_UNICODETEXT, pReq); int rc = VINF_SUCCESS; uint32_t cbActual = 0; tstClipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet); if (rc != rcExp) RTTestFailed(hTest, "Wrong return code, expected %Rrc, got %Rrc\n", rcExp, rc); else if (pReqRet != pReq) RTTestFailed(hTest, "Wrong returned request data, expected %p, got %p\n", pReq, pReqRet); else if (RT_FAILURE(rcExp)) retval = true; else { RTUTF16 wcExp[TESTCASE_MAX_BUF_SIZE / 2]; RTUTF16 *pwcExp = wcExp; size_t cwc = 0; rc = RTStrToUtf16Ex(pcszExp, RTSTR_MAX, &pwcExp, RT_ELEMENTS(wcExp), &cwc); size_t cbExp = cwc * 2 + 2; AssertRC(rc); if (RT_SUCCESS(rc)) { if (cbActual != cbExp) { RTTestFailed(hTest, "Returned string is the wrong size, string \"%.*ls\", size %u, expected \"%s\", size %u\n", RT_MIN(TESTCASE_MAX_BUF_SIZE, cbActual), pc, cbActual, pcszExp, cbExp); } else { if (memcmp(pc, wcExp, cbExp) == 0) retval = true; else RTTestFailed(hTest, "Returned string \"%.*ls\" does not match expected string \"%s\"\n", TESTCASE_MAX_BUF_SIZE, pc, pcszExp); } } } } if (!retval) RTTestFailureDetails(hTest, "Expected: string \"%s\", rc %Rrc\n", pcszExp, rcExp); } static void tstLatin1FromX11(RTTEST hTest, PSHCLX11CTX pCtx, const char *pcszExp, int rcExp) { bool retval = false; tstClipSendTargetUpdate(pCtx); if (tstClipQueryFormats() != VBOX_SHCL_FMT_UNICODETEXT) RTTestFailed(hTest, "Wrong targets reported: %02X\n", tstClipQueryFormats()); else { char *pc; CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL; ShClX11ReadDataFromX11(pCtx, VBOX_SHCL_FMT_UNICODETEXT, pReq); int rc = VINF_SUCCESS; uint32_t cbActual = 0; tstClipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet); if (rc != rcExp) RTTestFailed(hTest, "Wrong return code, expected %Rrc, got %Rrc\n", rcExp, rc); else if (pReqRet != pReq) RTTestFailed(hTest, "Wrong returned request data, expected %p, got %p\n", pReq, pReqRet); else if (RT_FAILURE(rcExp)) retval = true; else { RTUTF16 wcExp[TESTCASE_MAX_BUF_SIZE / 2]; //RTUTF16 *pwcExp = wcExp; - unused size_t cwc; for (cwc = 0; cwc == 0 || pcszExp[cwc - 1] != '\0'; ++cwc) wcExp[cwc] = pcszExp[cwc]; size_t cbExp = cwc * 2; if (cbActual != cbExp) { RTTestFailed(hTest, "Returned string is the wrong size, string \"%.*ls\", size %u, expected \"%s\", size %u\n", RT_MIN(TESTCASE_MAX_BUF_SIZE, cbActual), pc, cbActual, pcszExp, cbExp); } else { if (memcmp(pc, wcExp, cbExp) == 0) retval = true; else RTTestFailed(hTest, "Returned string \"%.*ls\" does not match expected string \"%s\"\n", TESTCASE_MAX_BUF_SIZE, pc, pcszExp); } } } if (!retval) RTTestFailureDetails(hTest, "Expected: string \"%s\", rc %Rrc\n", pcszExp, rcExp); } static void tstStringFromVBox(RTTEST hTest, PSHCLX11CTX pCtx, const char *pcszTarget, Atom typeExp, const char *valueExp) { RT_NOREF(pCtx); bool retval = false; Atom type; XtPointer value = NULL; unsigned long length; int format; size_t lenExp = strlen(valueExp); if (tstClipConvertSelection(pcszTarget, &type, &value, &length, &format)) { if ( type != typeExp || length != lenExp || format != 8 || memcmp((const void *) value, (const void *)valueExp, lenExp)) { RTTestFailed(hTest, "Bad data: type %d, (expected %d), length %u, (%u), format %d (%d), value \"%.*s\" (\"%.*s\")\n", type, typeExp, length, lenExp, format, 8, RT_MIN(length, 20), value, RT_MIN(lenExp, 20), valueExp); } else retval = true; } else RTTestFailed(hTest, "Conversion failed\n"); XtFree((char *)value); if (!retval) RTTestFailureDetails(hTest, "Conversion to %s, expected \"%s\"\n", pcszTarget, valueExp); } static void tstNoX11(PSHCLX11CTX pCtx, const char *pcszTestCtx) { CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq; int rc = ShClX11ReadDataFromX11(pCtx, VBOX_SHCL_FMT_UNICODETEXT, pReq); RTTESTI_CHECK_MSG(rc == VERR_NO_DATA, ("context: %s\n", pcszTestCtx)); } static void tstStringFromVBoxFailed(RTTEST hTest, PSHCLX11CTX pCtx, const char *pcszTarget) { RT_NOREF(pCtx); Atom type; XtPointer value = NULL; unsigned long length; int format; RTTEST_CHECK_MSG(hTest, !tstClipConvertSelection(pcszTarget, &type, &value, &length, &format), (hTest, "Conversion to target %s, should have failed but didn't, returned type %d, length %u, format %d, value \"%.*s\"\n", pcszTarget, type, length, format, RT_MIN(length, 20), value)); XtFree((char *)value); } static void tstNoSelectionOwnership(PSHCLX11CTX pCtx, const char *pcszTestCtx) { RT_NOREF(pCtx); RTTESTI_CHECK_MSG(!g_tst_fOwnsSel, ("context: %s\n", pcszTestCtx)); } static void tstBadFormatRequestFromHost(RTTEST hTest, PSHCLX11CTX pCtx) { tstClipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world", sizeof("hello world"), 8); tstClipSendTargetUpdate(pCtx); if (tstClipQueryFormats() != VBOX_SHCL_FMT_UNICODETEXT) RTTestFailed(hTest, "Wrong targets reported: %02X\n", tstClipQueryFormats()); else { char *pc; CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL; ShClX11ReadDataFromX11(pCtx, 100, pReq); /* Bad format. */ int rc = VINF_SUCCESS; uint32_t cbActual = 0; tstClipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet); if (rc != VERR_NOT_IMPLEMENTED) RTTestFailed(hTest, "Wrong return code, expected VERR_NOT_IMPLEMENTED, got %Rrc\n", rc); tstClipSetSelectionValues("", XA_STRING, "", sizeof(""), 8); tstClipSendTargetUpdate(pCtx); if (tstClipQueryFormats() == VBOX_SHCL_FMT_UNICODETEXT) RTTestFailed(hTest, "Failed to report targets after bad host request.\n"); } } int main() { /* * Init the runtime, test and say hello. */ RTTEST hTest; int rc = RTTestInitAndCreate("tstClipboardX11", &hTest); if (rc) return rc; RTTestBanner(hTest); /* * Run the test. */ SHCLX11CTX X11Ctx; rc = ShClX11Init(&X11Ctx, NULL, false); AssertRCReturn(rc, 1); char *pc; uint32_t cbActual; CLIPREADCBREQ *pReq = (CLIPREADCBREQ *)&pReq, *pReqRet = NULL; rc = ShClX11ThreadStart(&X11Ctx, false /* fGrab */); AssertRCReturn(rc, 1); /*** UTF-8 from X11 ***/ RTTestSub(hTest, "reading UTF-8 from X11"); /* Simple test */ tstClipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world", sizeof("hello world"), 8); tstStringFromX11(hTest, &X11Ctx, "hello world", VINF_SUCCESS); /* With an embedded carriage return */ tstClipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING, "hello\nworld", sizeof("hello\nworld"), 8); tstStringFromX11(hTest, &X11Ctx, "hello\r\nworld", VINF_SUCCESS); /* With an embedded CRLF */ tstClipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING, "hello\r\nworld", sizeof("hello\r\nworld"), 8); tstStringFromX11(hTest, &X11Ctx, "hello\r\r\nworld", VINF_SUCCESS); /* With an embedded LFCR */ tstClipSetSelectionValues("text/plain;charset=UTF-8", XA_STRING, "hello\n\rworld", sizeof("hello\n\rworld"), 8); tstStringFromX11(hTest, &X11Ctx, "hello\r\n\rworld", VINF_SUCCESS); /* An empty string */ tstClipSetSelectionValues("text/plain;charset=utf-8", XA_STRING, "", sizeof(""), 8); tstStringFromX11(hTest, &X11Ctx, "", VINF_SUCCESS); /* With an embedded UTF-8 character. */ tstClipSetSelectionValues("STRING", XA_STRING, "100\xE2\x82\xAC" /* 100 Euro */, sizeof("100\xE2\x82\xAC"), 8); tstStringFromX11(hTest, &X11Ctx, "100\xE2\x82\xAC", VINF_SUCCESS); /* A non-zero-terminated string */ tstClipSetSelectionValues("TEXT", XA_STRING, "hello world", sizeof("hello world") - 1, 8); tstStringFromX11(hTest, &X11Ctx, "hello world", VINF_SUCCESS); /*** Latin1 from X11 ***/ RTTestSub(hTest, "reading Latin1 from X11"); /* Simple test */ tstClipSetSelectionValues("STRING", XA_STRING, "Georges Dupr\xEA", sizeof("Georges Dupr\xEA"), 8); tstLatin1FromX11(hTest, &X11Ctx, "Georges Dupr\xEA", VINF_SUCCESS); /* With an embedded carriage return */ tstClipSetSelectionValues("TEXT", XA_STRING, "Georges\nDupr\xEA", sizeof("Georges\nDupr\xEA"), 8); tstLatin1FromX11(hTest, &X11Ctx, "Georges\r\nDupr\xEA", VINF_SUCCESS); /* With an embedded CRLF */ tstClipSetSelectionValues("TEXT", XA_STRING, "Georges\r\nDupr\xEA", sizeof("Georges\r\nDupr\xEA"), 8); tstLatin1FromX11(hTest, &X11Ctx, "Georges\r\r\nDupr\xEA", VINF_SUCCESS); /* With an embedded LFCR */ tstClipSetSelectionValues("TEXT", XA_STRING, "Georges\n\rDupr\xEA", sizeof("Georges\n\rDupr\xEA"), 8); tstLatin1FromX11(hTest, &X11Ctx, "Georges\r\n\rDupr\xEA", VINF_SUCCESS); /* A non-zero-terminated string */ tstClipSetSelectionValues("text/plain", XA_STRING, "Georges Dupr\xEA!", sizeof("Georges Dupr\xEA!") - 1, 8); tstLatin1FromX11(hTest, &X11Ctx, "Georges Dupr\xEA!", VINF_SUCCESS); /*** Unknown X11 format ***/ RTTestSub(hTest, "handling of an unknown X11 format"); tstClipInvalidateFormats(); tstClipSetSelectionValues("CLIPBOARD", XA_STRING, "Test", sizeof("Test"), 8); tstClipSendTargetUpdate(&X11Ctx); RTTEST_CHECK_MSG(hTest, tstClipQueryFormats() == 0, (hTest, "Failed to send a format update notification\n")); /*** Timeout from X11 ***/ RTTestSub(hTest, "X11 timeout"); tstClipSetSelectionValues("UTF8_STRING", XT_CONVERT_FAIL, NULL,0, 8); tstStringFromX11(hTest, &X11Ctx, "", VERR_NO_DATA); /*** No data in X11 clipboard ***/ RTTestSub(hTest, "a data request from an empty X11 clipboard"); tstClipSetSelectionValues("UTF8_STRING", XA_STRING, NULL, 0, 8); ShClX11ReadDataFromX11(&X11Ctx, VBOX_SHCL_FMT_UNICODETEXT, pReq); tstClipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet); RTTEST_CHECK_MSG(hTest, rc == VERR_NO_DATA, (hTest, "Returned %Rrc instead of VERR_NO_DATA\n", rc)); RTTEST_CHECK_MSG(hTest, pReqRet == pReq, (hTest, "Wrong returned request data, expected %p, got %p\n", pReq, pReqRet)); /*** Ensure that VBox is notified when we return the CB to X11 ***/ RTTestSub(hTest, "notification of switch to X11 clipboard"); tstClipInvalidateFormats(); clipReportEmptyX11CB(&X11Ctx); RTTEST_CHECK_MSG(hTest, tstClipQueryFormats() == 0, (hTest, "Failed to send a format update (release) notification\n")); /*** request for an invalid VBox format from X11 ***/ RTTestSub(hTest, "a request for an invalid VBox format from X11"); ShClX11ReadDataFromX11(&X11Ctx, 0xffff, pReq); tstClipGetCompletedRequest(&rc, &pc, &cbActual, &pReqRet); RTTEST_CHECK_MSG(hTest, rc == VERR_NOT_IMPLEMENTED, (hTest, "Returned %Rrc instead of VERR_NOT_IMPLEMENTED\n", rc)); RTTEST_CHECK_MSG(hTest, pReqRet == pReq, (hTest, "Wrong returned request data, expected %p, got %p\n", pReq, pReqRet)); /*** Targets failure from X11 ***/ RTTestSub(hTest, "X11 targets conversion failure"); tstClipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world", sizeof("hello world"), 8); tstClipSetTargetsFailure(); Atom atom = XA_STRING; long unsigned int cLen = 0; int format = 8; clipConvertX11TargetsCallback(NULL, (XtPointer) &X11Ctx, NULL, &atom, NULL, &cLen, &format); RTTEST_CHECK_MSG(hTest, tstClipQueryFormats() == 0, (hTest, "Wrong targets reported: %02X\n", tstClipQueryFormats())); /*** X11 text format conversion ***/ RTTestSub(hTest, "handling of X11 selection targets"); RTTEST_CHECK_MSG(hTest, tstClipTextFormatConversion(&X11Ctx), (hTest, "failed to select the right X11 text formats\n")); /*** UTF-8 from VBox ***/ RTTestSub(hTest, "reading UTF-8 from VBox"); /* Simple test */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello world", sizeof("hello world") * 2); tstStringFromVBox(hTest, &X11Ctx, "UTF8_STRING", clipGetAtom(&X11Ctx, "UTF8_STRING"), "hello world"); /* With an embedded carriage return */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello\r\nworld", sizeof("hello\r\nworld") * 2); tstStringFromVBox(hTest, &X11Ctx, "text/plain;charset=UTF-8", clipGetAtom(&X11Ctx, "text/plain;charset=UTF-8"), "hello\nworld"); /* With an embedded CRCRLF */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello\r\r\nworld", sizeof("hello\r\r\nworld") * 2); tstStringFromVBox(hTest, &X11Ctx, "text/plain;charset=UTF-8", clipGetAtom(&X11Ctx, "text/plain;charset=UTF-8"), "hello\r\nworld"); /* With an embedded CRLFCR */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello\r\n\rworld", sizeof("hello\r\n\rworld") * 2); tstStringFromVBox(hTest, &X11Ctx, "text/plain;charset=UTF-8", clipGetAtom(&X11Ctx, "text/plain;charset=UTF-8"), "hello\n\rworld"); /* An empty string */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "", 2); tstStringFromVBox(hTest, &X11Ctx, "text/plain;charset=utf-8", clipGetAtom(&X11Ctx, "text/plain;charset=utf-8"), ""); /* With an embedded UTF-8 character. */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "100\xE2\x82\xAC" /* 100 Euro */, 10); tstStringFromVBox(hTest, &X11Ctx, "STRING", clipGetAtom(&X11Ctx, "STRING"), "100\xE2\x82\xAC"); /* A non-zero-terminated string */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello world", sizeof("hello world") * 2 - 2); tstStringFromVBox(hTest, &X11Ctx, "TEXT", clipGetAtom(&X11Ctx, "TEXT"), "hello world"); /*** Timeout from VBox ***/ RTTestSub(hTest, "reading from VBox with timeout"); tstClipEmptyVBox(&X11Ctx, VERR_TIMEOUT); tstStringFromVBoxFailed(hTest, &X11Ctx, "UTF8_STRING"); /*** No data in VBox clipboard ***/ RTTestSub(hTest, "an empty VBox clipboard"); tstClipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8); tstClipEmptyVBox(&X11Ctx, VINF_SUCCESS); RTTEST_CHECK_MSG(hTest, g_tst_fOwnsSel, (hTest, "VBox grabbed the clipboard with no data and we ignored it\n")); tstStringFromVBoxFailed(hTest, &X11Ctx, "UTF8_STRING"); /*** An unknown VBox format ***/ RTTestSub(hTest, "reading an unknown VBox format"); tstClipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8); tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "", 2); ShClX11ReportFormatsToX11(&X11Ctx, 0xa0000); RTTEST_CHECK_MSG(hTest, g_tst_fOwnsSel, (hTest, "VBox grabbed the clipboard with unknown data and we ignored it\n")); tstStringFromVBoxFailed(hTest, &X11Ctx, "UTF8_STRING"); /*** VBox requests a bad format ***/ RTTestSub(hTest, "recovery from a bad format request"); tstBadFormatRequestFromHost(hTest, &X11Ctx); rc = ShClX11ThreadStop(&X11Ctx); AssertRCReturn(rc, 1); ShClX11Destroy(&X11Ctx); /*** Headless clipboard tests ***/ rc = ShClX11Init(&X11Ctx, NULL, true); AssertRCReturn(rc, 1); rc = ShClX11ThreadStart(&X11Ctx, false /* fGrab */); AssertRCReturn(rc, 1); /*** Read from X11 ***/ RTTestSub(hTest, "reading from X11, headless clipboard"); /* Simple test */ tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "", sizeof("") * 2); tstClipSetSelectionValues("UTF8_STRING", XA_STRING, "hello world", sizeof("hello world"), 8); tstNoX11(&X11Ctx, "reading from X11, headless clipboard"); /*** Read from VBox ***/ RTTestSub(hTest, "reading from VBox, headless clipboard"); /* Simple test */ tstClipEmptyVBox(&X11Ctx, VERR_WRONG_ORDER); tstClipSetSelectionValues("TEXT", XA_STRING, "", sizeof(""), 8); tstClipSetVBoxUtf16(&X11Ctx, VINF_SUCCESS, "hello world", sizeof("hello world") * 2); tstNoSelectionOwnership(&X11Ctx, "reading from VBox, headless clipboard"); rc = ShClX11ThreadStop(&X11Ctx); AssertRCReturn(rc, 1); ShClX11Destroy(&X11Ctx); return RTTestSummaryAndDestroy(hTest); } #endif #ifdef SMOKETEST /* This is a simple test case that just starts a copy of the X11 clipboard * backend, checks the X11 clipboard and exits. If ever needed I will add an * interactive mode in which the user can read and copy to the clipboard from * the command line. */ # include # include DECLCALLBACK(int) ShClX11RequestDataForX11Callback(PSHCLCONTEXT pCtx, SHCLFORMAT Format, void **ppv, uint32_t *pcb) { RT_NOREF(pCtx, Format, ppv, pcb); return VERR_NO_DATA; } DECLCALLBACK(void) ShClX11ReportFormatsCallback(PSHCLCONTEXT pCtx, SHCLFORMATS Formats) { RT_NOREF(pCtx, Formats); } DECLCALLBACK(void) ShClX11RequestFromX11CompleteCallback(PSHCLCONTEXT pCtx, int rc, CLIPREADCBREQ *pReq, void *pv, uint32_t cb) { RT_NOREF(pCtx, rc, pReq, pv, cb); } int main() { /* * Init the runtime, test and say hello. */ RTTEST hTest; int rc = RTTestInitAndCreate("tstClipboardX11Smoke", &hTest); if (rc) return rc; RTTestBanner(hTest); /* * Run the test. */ rc = VINF_SUCCESS; /* We can't test anything without an X session, so just return success * in that case. */ if (!RTEnvExist("DISPLAY")) { RTTestPrintf(hTest, RTTESTLVL_INFO, "X11 not available, not running test\n"); return RTTestSummaryAndDestroy(hTest); } SHCLX11CTX X11Ctx; rc = ShClX11Init(&X11Ctx, NULL, false); AssertRCReturn(rc, 1); rc = ShClX11ThreadStart(&X11Ctx, false /* fGrab */); AssertRCReturn(rc, 1); /* Give the clipboard time to synchronise. */ RTThreadSleep(500); rc = ShClX11ThreadStop(&X11Ctx); AssertRCReturn(rc, 1); ShClX11Destroy(&X11Ctx); return RTTestSummaryAndDestroy(hTest); } #endif /* SMOKETEST defined */